Add basic tests for VersionedMetaData updates
Change-Id: I96e0b5d09f37375276ba67a94cbcc78879f465bf
This commit is contained in:
@@ -0,0 +1,200 @@
|
|||||||
|
// Copyright (C) 2017 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.server.git;
|
||||||
|
|
||||||
|
import static com.google.common.base.Preconditions.checkArgument;
|
||||||
|
import static com.google.common.collect.ImmutableList.toImmutableList;
|
||||||
|
import static com.google.common.truth.Truth.assertThat;
|
||||||
|
|
||||||
|
import com.google.common.collect.ImmutableList;
|
||||||
|
import com.google.common.collect.Streams;
|
||||||
|
import com.google.gerrit.common.TimeUtil;
|
||||||
|
import com.google.gerrit.reviewdb.client.Project;
|
||||||
|
import com.google.gerrit.server.extensions.events.GitReferenceUpdated;
|
||||||
|
import com.google.gerrit.testing.TestTimeUtil;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.Optional;
|
||||||
|
import java.util.TimeZone;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import org.eclipse.jgit.errors.ConfigInvalidException;
|
||||||
|
import org.eclipse.jgit.internal.storage.dfs.DfsRepositoryDescription;
|
||||||
|
import org.eclipse.jgit.internal.storage.dfs.InMemoryRepository;
|
||||||
|
import org.eclipse.jgit.lib.CommitBuilder;
|
||||||
|
import org.eclipse.jgit.lib.Config;
|
||||||
|
import org.eclipse.jgit.lib.PersonIdent;
|
||||||
|
import org.eclipse.jgit.lib.Ref;
|
||||||
|
import org.eclipse.jgit.lib.Repository;
|
||||||
|
import org.eclipse.jgit.revwalk.RevCommit;
|
||||||
|
import org.eclipse.jgit.revwalk.RevSort;
|
||||||
|
import org.eclipse.jgit.revwalk.RevWalk;
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
public class VersionedMetaDataTest {
|
||||||
|
// If you're considering fleshing out this test and making it more comprehensive, please consider
|
||||||
|
// instead coming up with a replacement interface for
|
||||||
|
// VersionedMetaData/BatchMetaDataUpdate/MetaDataUpdate that is easier to use correctly.
|
||||||
|
|
||||||
|
private static final TimeZone TZ = TimeZone.getTimeZone("America/Los_Angeles");
|
||||||
|
|
||||||
|
private Project.NameKey project;
|
||||||
|
private Repository repo;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setUp() {
|
||||||
|
TestTimeUtil.resetWithClockStep(1, TimeUnit.SECONDS);
|
||||||
|
project = new Project.NameKey("repo");
|
||||||
|
repo = new InMemoryRepository(new DfsRepositoryDescription(project.get()));
|
||||||
|
}
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void tearDown() {
|
||||||
|
TestTimeUtil.useSystemTime();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void singleUpdate() throws Exception {
|
||||||
|
MyMetaData d = load(0);
|
||||||
|
d.setIncrement(3);
|
||||||
|
d.commit(newMetaDataUpdate());
|
||||||
|
assertMyMetaData(3, "Increment conf.value by 3");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void noOpNoSetter() throws Exception {
|
||||||
|
MyMetaData d = load(0);
|
||||||
|
d.commit(newMetaDataUpdate());
|
||||||
|
assertMyMetaData(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void noOpWithSetter() throws Exception {
|
||||||
|
MyMetaData d = load(0);
|
||||||
|
d.setIncrement(0);
|
||||||
|
d.commit(newMetaDataUpdate());
|
||||||
|
// First commit is actually not a no-op because it creates an empty config file.
|
||||||
|
assertMyMetaData(0, "Increment conf.value by 0");
|
||||||
|
|
||||||
|
d = load(0);
|
||||||
|
d.setIncrement(0);
|
||||||
|
d.commit(newMetaDataUpdate());
|
||||||
|
assertMyMetaData(0, "Increment conf.value by 0");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void multipleSeparateUpdatesWithSameObject() throws Exception {
|
||||||
|
MyMetaData d = load(0);
|
||||||
|
d.setIncrement(1);
|
||||||
|
d.commit(newMetaDataUpdate());
|
||||||
|
assertMyMetaData(1, "Increment conf.value by 1");
|
||||||
|
d.setIncrement(2);
|
||||||
|
d.commit(newMetaDataUpdate());
|
||||||
|
assertMyMetaData(3, "Increment conf.value by 1", "Increment conf.value by 2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void multipleSeparateUpdatesWithDifferentObject() throws Exception {
|
||||||
|
MyMetaData d = load(0);
|
||||||
|
d.setIncrement(1);
|
||||||
|
d.commit(newMetaDataUpdate());
|
||||||
|
assertMyMetaData(1, "Increment conf.value by 1");
|
||||||
|
|
||||||
|
d = load(1);
|
||||||
|
d.setIncrement(2);
|
||||||
|
d.commit(newMetaDataUpdate());
|
||||||
|
assertMyMetaData(3, "Increment conf.value by 1", "Increment conf.value by 2");
|
||||||
|
}
|
||||||
|
|
||||||
|
private MyMetaData load(int expectedValue) throws Exception {
|
||||||
|
MyMetaData d = new MyMetaData();
|
||||||
|
d.load(repo);
|
||||||
|
assertThat(d.getValue()).isEqualTo(expectedValue);
|
||||||
|
return d;
|
||||||
|
}
|
||||||
|
|
||||||
|
private MetaDataUpdate newMetaDataUpdate() {
|
||||||
|
MetaDataUpdate u = new MetaDataUpdate(GitReferenceUpdated.DISABLED, project, repo, null);
|
||||||
|
PersonIdent author = new PersonIdent("J. Author", "author@example.com", TimeUtil.nowTs(), TZ);
|
||||||
|
u.getCommitBuilder().setAuthor(author);
|
||||||
|
u.getCommitBuilder()
|
||||||
|
.setCommitter(
|
||||||
|
new PersonIdent(
|
||||||
|
"M. Committer", "committer@example.com", author.getWhen(), author.getTimeZone()));
|
||||||
|
return u;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertMyMetaData(int expectedValue, String... expectedLog) throws Exception {
|
||||||
|
MyMetaData d = load(expectedValue);
|
||||||
|
assertThat(log(d)).containsExactlyElementsIn(Arrays.asList(expectedLog)).inOrder();
|
||||||
|
}
|
||||||
|
|
||||||
|
private ImmutableList<String> log(MyMetaData d) throws Exception {
|
||||||
|
try (RevWalk rw = new RevWalk(repo)) {
|
||||||
|
Ref ref = repo.exactRef(d.getRefName());
|
||||||
|
if (ref == null) {
|
||||||
|
return ImmutableList.of();
|
||||||
|
}
|
||||||
|
rw.sort(RevSort.REVERSE);
|
||||||
|
rw.setRetainBody(true);
|
||||||
|
rw.markStart(rw.parseCommit(ref.getObjectId()));
|
||||||
|
return Streams.stream(rw).map(RevCommit::getFullMessage).collect(toImmutableList());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class MyMetaData extends VersionedMetaData {
|
||||||
|
private static final String CONFIG_FILE = "my.config";
|
||||||
|
private static final String SECTION = "conf";
|
||||||
|
private static final String NAME = "value";
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected String getRefName() {
|
||||||
|
return "refs/my/config";
|
||||||
|
}
|
||||||
|
|
||||||
|
private int curr;
|
||||||
|
private Optional<Integer> increment = Optional.empty();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onLoad() throws IOException, ConfigInvalidException {
|
||||||
|
Config cfg = readConfig(CONFIG_FILE);
|
||||||
|
curr = cfg.getInt(SECTION, null, NAME, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
int getValue() {
|
||||||
|
return curr;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setIncrement(int increment) {
|
||||||
|
checkArgument(increment >= 0, "increment must be positive: %s", increment);
|
||||||
|
this.increment = Optional.of(increment);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected boolean onSave(CommitBuilder cb) throws IOException, ConfigInvalidException {
|
||||||
|
// Two ways to produce a no-op: don't call setIncrement, and call setIncrement(0);
|
||||||
|
if (!increment.isPresent()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
Config cfg = readConfig(CONFIG_FILE);
|
||||||
|
cfg.setInt(SECTION, null, NAME, cfg.getInt(SECTION, null, NAME, 0) + increment.get());
|
||||||
|
cb.setMessage(String.format("Increment %s.%s by %d", SECTION, NAME, increment.get()));
|
||||||
|
saveConfig(CONFIG_FILE, cfg);
|
||||||
|
increment = Optional.empty();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user