org.projectnessie.versioned.persist.tests.AbstractReferences Maven / Gradle / Ivy
/*
* Copyright (C) 2020 Dremio
*
* 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.projectnessie.versioned.persist.tests;
import static org.assertj.core.api.Assertions.assertThat;
import static org.assertj.core.api.Assertions.assertThatThrownBy;
import static org.junit.jupiter.api.Assertions.assertAll;
import static org.projectnessie.versioned.store.DefaultStoreWorker.payloadForContent;
import static org.projectnessie.versioned.testworker.OnRefOnly.onRef;
import com.google.protobuf.ByteString;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Map;
import java.util.Optional;
import java.util.function.IntFunction;
import java.util.stream.Collectors;
import java.util.stream.IntStream;
import java.util.stream.Stream;
import org.junit.jupiter.api.Test;
import org.projectnessie.model.ContentKey;
import org.projectnessie.versioned.BranchName;
import org.projectnessie.versioned.GetNamedRefsParams;
import org.projectnessie.versioned.Hash;
import org.projectnessie.versioned.NamedRef;
import org.projectnessie.versioned.ReferenceAlreadyExistsException;
import org.projectnessie.versioned.ReferenceConflictException;
import org.projectnessie.versioned.ReferenceInfo;
import org.projectnessie.versioned.ReferenceNotFoundException;
import org.projectnessie.versioned.TagName;
import org.projectnessie.versioned.persist.adapter.ContentId;
import org.projectnessie.versioned.persist.adapter.DatabaseAdapter;
import org.projectnessie.versioned.persist.adapter.ImmutableCommitParams;
import org.projectnessie.versioned.persist.adapter.KeyWithBytes;
import org.projectnessie.versioned.store.DefaultStoreWorker;
import org.projectnessie.versioned.testworker.OnRefOnly;
/** Verifies handling of repo-description in the database-adapters. */
public abstract class AbstractReferences {
private final DatabaseAdapter databaseAdapter;
protected AbstractReferences(DatabaseAdapter databaseAdapter) {
this.databaseAdapter = databaseAdapter;
}
@Test
void createBranch() throws Exception {
BranchName create = BranchName.of("createBranch");
createNamedRef(create, TagName.of(create.getName()));
}
@Test
void createTag() throws Exception {
TagName create = TagName.of("createTag");
createNamedRef(create, BranchName.of(create.getName()));
}
private void createNamedRef(NamedRef create, NamedRef opposite) throws Exception {
BranchName branch = BranchName.of("main");
try (Stream> refs =
databaseAdapter.namedRefs(GetNamedRefsParams.DEFAULT)) {
assertThat(refs.map(ReferenceInfo::getNamedRef)).containsExactlyInAnyOrder(branch);
}
Hash mainHash = databaseAdapter.hashOnReference(branch, Optional.empty());
assertThatThrownBy(() -> databaseAdapter.hashOnReference(create, Optional.empty()))
.isInstanceOf(ReferenceNotFoundException.class);
Hash createHash =
databaseAdapter.create(create, databaseAdapter.hashOnReference(branch, Optional.empty()));
assertThat(createHash).isEqualTo(mainHash);
try (Stream> refs =
databaseAdapter.namedRefs(GetNamedRefsParams.DEFAULT)) {
assertThat(refs.map(ReferenceInfo::getNamedRef)).containsExactlyInAnyOrder(branch, create);
}
assertThatThrownBy(
() ->
databaseAdapter.create(
create, databaseAdapter.hashOnReference(branch, Optional.empty())))
.isInstanceOf(ReferenceAlreadyExistsException.class);
assertThat(databaseAdapter.hashOnReference(create, Optional.empty())).isEqualTo(createHash);
assertThatThrownBy(() -> databaseAdapter.hashOnReference(opposite, Optional.empty()))
.isInstanceOf(ReferenceNotFoundException.class);
assertThatThrownBy(
() ->
databaseAdapter.create(
BranchName.of(create.getName()),
databaseAdapter.hashOnReference(branch, Optional.empty())))
.isInstanceOf(ReferenceAlreadyExistsException.class);
assertThatThrownBy(
() -> databaseAdapter.delete(create, Optional.of(Hash.of("dead00004242fee18eef"))))
.isInstanceOf(ReferenceConflictException.class);
assertThatThrownBy(() -> databaseAdapter.delete(opposite, Optional.of(createHash)))
.isInstanceOf(ReferenceNotFoundException.class);
assertThat(databaseAdapter.delete(create, Optional.of(createHash))).isEqualTo(createHash);
assertThatThrownBy(() -> databaseAdapter.hashOnReference(create, Optional.empty()))
.isInstanceOf(ReferenceNotFoundException.class);
try (Stream> refs =
databaseAdapter.namedRefs(GetNamedRefsParams.DEFAULT)) {
assertThat(refs.map(ReferenceInfo::getNamedRef)).containsExactlyInAnyOrder(branch);
}
}
@Test
void verifyNotFoundAndConflictExceptionsForUnreachableCommit() throws Exception {
BranchName main = BranchName.of("main");
BranchName unreachable = BranchName.of("unreachable");
BranchName helper = BranchName.of("helper");
databaseAdapter.create(unreachable, databaseAdapter.hashOnReference(main, Optional.empty()));
Hash helperHead =
databaseAdapter.create(helper, databaseAdapter.hashOnReference(main, Optional.empty()));
OnRefOnly hello = onRef("hello", "contentId");
Hash unreachableHead =
databaseAdapter.commit(
ImmutableCommitParams.builder()
.toBranch(unreachable)
.commitMetaSerialized(ByteString.copyFromUtf8("commit meta"))
.addPuts(
KeyWithBytes.of(
ContentKey.of("foo"),
ContentId.of(hello.getId()),
payloadForContent(hello),
hello.serialized()))
.build());
assertAll(
() ->
assertThatThrownBy(
() -> databaseAdapter.hashOnReference(main, Optional.of(unreachableHead)))
.isInstanceOf(ReferenceNotFoundException.class)
.hasMessage(
String.format(
"Could not find commit '%s' in reference '%s'.",
unreachableHead.asString(), main.getName())),
() ->
assertThatThrownBy(
() -> {
OnRefOnly noNo = onRef("hello", "contentId-no-no");
databaseAdapter.commit(
ImmutableCommitParams.builder()
.toBranch(helper)
.expectedHead(Optional.of(unreachableHead))
.commitMetaSerialized(ByteString.copyFromUtf8("commit meta"))
.addPuts(
KeyWithBytes.of(
ContentKey.of("bar"),
ContentId.of(noNo.getId()),
payloadForContent(noNo),
noNo.serialized()))
.build());
})
.isInstanceOf(ReferenceNotFoundException.class)
.hasMessage(
String.format(
"Could not find commit '%s' in reference '%s'.",
unreachableHead.asString(), helper.getName())),
() ->
assertThatThrownBy(
() ->
databaseAdapter.assign(
helper,
Optional.of(unreachableHead),
databaseAdapter.hashOnReference(main, Optional.empty())))
.isInstanceOf(ReferenceConflictException.class)
.hasMessage(
String.format(
"Named-reference '%s' is not at expected hash '%s', but at '%s'.",
helper.getName(), unreachableHead.asString(), helperHead.asString())));
}
@Test
void assign() throws Exception {
BranchName main = BranchName.of("main");
TagName tag = TagName.of("tag");
TagName branch = TagName.of("branch");
databaseAdapter.create(branch, databaseAdapter.hashOnReference(main, Optional.empty()));
databaseAdapter.create(tag, databaseAdapter.hashOnReference(main, Optional.empty()));
Hash beginning = databaseAdapter.hashOnReference(main, Optional.empty());
Hash[] commits = new Hash[3];
for (int i = 0; i < commits.length; i++) {
OnRefOnly hello = onRef("hello " + i, "contentId-" + i);
commits[i] =
databaseAdapter.commit(
ImmutableCommitParams.builder()
.toBranch(main)
.commitMetaSerialized(ByteString.copyFromUtf8("commit meta " + i))
.addPuts(
KeyWithBytes.of(
ContentKey.of("bar-" + i),
ContentId.of(hello.getId()),
payloadForContent(hello),
hello.serialized()))
.build());
}
Hash expect = beginning;
for (Hash commit : commits) {
assertThat(
Arrays.asList(
databaseAdapter.hashOnReference(branch, Optional.empty()),
databaseAdapter.hashOnReference(tag, Optional.empty())))
.containsExactly(expect, expect);
databaseAdapter.assign(tag, Optional.of(expect), commit);
databaseAdapter.assign(branch, Optional.of(expect), commit);
expect = commit;
}
assertThat(
Arrays.asList(
databaseAdapter.hashOnReference(branch, Optional.empty()),
databaseAdapter.hashOnReference(tag, Optional.empty())))
.containsExactly(commits[commits.length - 1], commits[commits.length - 1]);
}
@Test
void recreateDefaultBranch() throws Exception {
// note: the default branch cannot be deleted through the TreeApi,
// but the underlying DatabaseAdapter should still support this operation.
BranchName main = BranchName.of("main");
Hash mainHead = databaseAdapter.hashOnReference(main, Optional.empty());
assertThat(databaseAdapter.delete(main, Optional.of(mainHead))).isEqualTo(mainHead);
assertThatThrownBy(() -> databaseAdapter.hashOnReference(main, Optional.empty()))
.isInstanceOf(ReferenceNotFoundException.class);
databaseAdapter.create(main, null);
databaseAdapter.hashOnReference(main, Optional.empty());
}
@Test
void deleteReferences() throws Exception {
BranchName main = BranchName.of("main");
Hash mainHead = databaseAdapter.hashOnReference(main, Optional.empty());
BranchName delete1 = BranchName.of("delete1");
TagName delete2 = TagName.of("delete2");
BranchName delete3 = BranchName.of("delete3");
TagName delete4 = TagName.of("delete4");
databaseAdapter.create(delete1, mainHead);
databaseAdapter.create(delete2, mainHead);
databaseAdapter.create(delete3, mainHead);
databaseAdapter.create(delete4, mainHead);
assertThat(databaseAdapter.delete(delete1, Optional.of(mainHead))).isEqualTo(mainHead);
assertThat(databaseAdapter.delete(delete2, Optional.of(mainHead))).isEqualTo(mainHead);
assertThat(databaseAdapter.delete(delete3, Optional.empty())).isEqualTo(mainHead);
assertThat(databaseAdapter.delete(delete4, Optional.empty())).isEqualTo(mainHead);
assertThatThrownBy(() -> databaseAdapter.hashOnReference(delete1, Optional.empty()))
.isInstanceOf(ReferenceNotFoundException.class);
assertThatThrownBy(() -> databaseAdapter.hashOnReference(delete2, Optional.empty()))
.isInstanceOf(ReferenceNotFoundException.class);
assertThatThrownBy(() -> databaseAdapter.hashOnReference(delete3, Optional.empty()))
.isInstanceOf(ReferenceNotFoundException.class);
assertThatThrownBy(() -> databaseAdapter.hashOnReference(delete4, Optional.empty()))
.isInstanceOf(ReferenceNotFoundException.class);
}
/** Validates that multiple reference-name segments work. */
@Test
void manyReferences() throws Exception {
IntFunction refGen =
i -> {
StringBuilder sb = new StringBuilder(120);
sb.append("manyReferencesTest-").append(i).append('-');
while (sb.length() < 100) {
sb.append('x');
}
String name = sb.toString();
return (i & 1) == 1 ? TagName.of(name) : BranchName.of(name);
};
Map refHeads = new HashMap<>();
for (int i = 0; i < 50; i++) {
NamedRef ref = refGen.apply(i);
assertThat(databaseAdapter.create(ref, databaseAdapter.noAncestorHash()))
.isEqualTo(databaseAdapter.noAncestorHash());
refHeads.put(ref, databaseAdapter.noAncestorHash());
assertThat(databaseAdapter.namedRef(ref.getName(), GetNamedRefsParams.DEFAULT).getNamedRef())
.isEqualTo(ref);
}
// Verify that the HEADs of all references point to the no-ancestor-hash
try (Stream> refs =
databaseAdapter.namedRefs(GetNamedRefsParams.DEFAULT)) {
assertThat(refs.filter(ri -> ri.getNamedRef().getName().startsWith("manyReferencesTest-")))
.containsExactlyInAnyOrderElementsOf(
IntStream.range(0, 50)
.mapToObj(refGen)
.map(ref -> ReferenceInfo.of(databaseAdapter.noAncestorHash(), ref))
.collect(Collectors.toList()));
}
// add 50 commits to every branch (crossing the number of parents per commit log entry)
for (int commit = 0; commit < 50; commit++) {
for (int i = 0; i < 50; i++) {
NamedRef ref = refGen.apply(i);
if (ref instanceof BranchName) {
Hash newHead =
databaseAdapter.commit(
ImmutableCommitParams.builder()
.toBranch((BranchName) ref)
.commitMetaSerialized(ByteString.copyFromUtf8("foo on " + ref.getName()))
.expectedHead(Optional.of(refHeads.get(ref)))
.addPuts(
KeyWithBytes.of(
ContentKey.of("table-" + commit),
ContentId.of("c" + commit),
payloadForContent(OnRefOnly.ON_REF_ONLY),
DefaultStoreWorker.instance()
.toStoreOnReferenceState(
OnRefOnly.newOnRef("c" + commit), att -> {})))
.build());
refHeads.put(ref, newHead);
}
}
}
// drop every 3rd reference
for (int i = 2; i < 50; i += 3) {
NamedRef ref = refGen.apply(i);
databaseAdapter.delete(ref, Optional.empty());
assertThatThrownBy(() -> databaseAdapter.namedRef(ref.getName(), GetNamedRefsParams.DEFAULT))
.isInstanceOf(ReferenceNotFoundException.class);
}
// Verify HEAD hashes for remaining branches + tags
try (Stream> refs =
databaseAdapter.namedRefs(GetNamedRefsParams.DEFAULT)) {
assertThat(refs.filter(ri -> ri.getNamedRef().getName().startsWith("manyReferencesTest-")))
.containsExactlyInAnyOrderElementsOf(
IntStream.range(0, 50)
.filter(i -> (i - 2) % 3 != 0)
.mapToObj(refGen)
.map(
ref ->
ReferenceInfo.of(
refHeads.getOrDefault(ref, databaseAdapter.noAncestorHash()), ref))
.collect(Collectors.toList()));
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy