// 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.schema; import static com.google.common.base.Preconditions.checkState; import com.google.common.collect.ImmutableList; import com.google.common.util.concurrent.CheckedFuture; import com.google.common.util.concurrent.Futures; import com.google.gerrit.reviewdb.client.Account; import com.google.gerrit.reviewdb.client.Change; import com.google.gerrit.reviewdb.client.ChangeMessage; import com.google.gerrit.reviewdb.client.PatchLineComment; import com.google.gerrit.reviewdb.client.PatchSet; import com.google.gerrit.reviewdb.client.PatchSetApproval; import com.google.gerrit.reviewdb.server.ChangeAccess; import com.google.gerrit.reviewdb.server.ChangeMessageAccess; import com.google.gerrit.reviewdb.server.PatchLineCommentAccess; import com.google.gerrit.reviewdb.server.PatchSetAccess; import com.google.gerrit.reviewdb.server.PatchSetApprovalAccess; import com.google.gerrit.reviewdb.server.ReviewDb; import com.google.gerrit.reviewdb.server.ReviewDbWrapper; import com.google.gwtorm.client.Key; import com.google.gwtorm.server.Access; import com.google.gwtorm.server.AtomicUpdate; import com.google.gwtorm.server.ListResultSet; import com.google.gwtorm.server.OrmException; import com.google.gwtorm.server.ResultSet; import java.util.Map; import java.util.function.Function; /** * Wrapper for ReviewDb that never calls the underlying change tables. * * <p>See {@link NotesMigrationSchemaFactory} for discussion. */ class NoChangesReviewDbWrapper extends ReviewDbWrapper { private static <T> ResultSet<T> empty() { return new ListResultSet<>(ImmutableList.of()); } private static <T, K extends Key<?>> CheckedFuture<T, OrmException> emptyFuture() { return Futures.immediateCheckedFuture(null); } private final ChangeAccess changes; private final PatchSetApprovalAccess patchSetApprovals; private final ChangeMessageAccess changeMessages; private final PatchSetAccess patchSets; private final PatchLineCommentAccess patchComments; private boolean inTransaction; NoChangesReviewDbWrapper(ReviewDb db) { super(db); changes = new Changes(this, delegate); patchSetApprovals = new PatchSetApprovals(this, delegate); changeMessages = new ChangeMessages(this, delegate); patchSets = new PatchSets(this, delegate); patchComments = new PatchLineComments(this, delegate); } @Override public boolean changesTablesEnabled() { return false; } @Override public ChangeAccess changes() { return changes; } @Override public PatchSetApprovalAccess patchSetApprovals() { return patchSetApprovals; } @Override public ChangeMessageAccess changeMessages() { return changeMessages; } @Override public PatchSetAccess patchSets() { return patchSets; } @Override public PatchLineCommentAccess patchComments() { return patchComments; } @Override public void commit() throws OrmException { if (!inTransaction) { // This reads a little weird, we're not in a transaction, so why are we calling commit? // Because we want to let the underlying ReviewDb do its normal thing in this case (which may // be throwing an exception, or not, depending on implementation). delegate.commit(); } } @Override public void rollback() throws OrmException { if (inTransaction) { inTransaction = false; } else { // See comment in commit(): we want to let the underlying ReviewDb do its thing. delegate.rollback(); } } private abstract static class AbstractDisabledAccess<T, K extends Key<?>> implements Access<T, K> { // Don't even hold a reference to delegate, so it's not possible to use it accidentally. private final NoChangesReviewDbWrapper wrapper; private final String relationName; private final int relationId; private final Function<T, K> primaryKey; private final Function<Iterable<T>, Map<K, T>> toMap; private AbstractDisabledAccess(NoChangesReviewDbWrapper wrapper, Access<T, K> delegate) { this.wrapper = wrapper; this.relationName = delegate.getRelationName(); this.relationId = delegate.getRelationID(); this.primaryKey = delegate::primaryKey; this.toMap = delegate::toMap; } @Override public final int getRelationID() { return relationId; } @Override public final String getRelationName() { return relationName; } @Override public final K primaryKey(T entity) { return primaryKey.apply(entity); } @Override public final Map<K, T> toMap(Iterable<T> iterable) { return toMap.apply(iterable); } @Override public final ResultSet<T> iterateAllEntities() { return empty(); } @Override public final CheckedFuture<T, OrmException> getAsync(K key) { return emptyFuture(); } @Override public final ResultSet<T> get(Iterable<K> keys) { return empty(); } @Override public final void insert(Iterable<T> instances) { // Do nothing. } @Override public final void update(Iterable<T> instances) { // Do nothing. } @Override public final void upsert(Iterable<T> instances) { // Do nothing. } @Override public final void deleteKeys(Iterable<K> keys) { // Do nothing. } @Override public final void delete(Iterable<T> instances) { // Do nothing. } @Override public final void beginTransaction(K key) { // Keep track of when we've started a transaction so that we can avoid calling commit/rollback // on the underlying ReviewDb. This is just a simple arm's-length approach, and may produce // slightly different results from a native ReviewDb in corner cases like: // * beginning transactions on different tables simultaneously // * doing work between commit and rollback // These kinds of things are already misuses of ReviewDb, and shouldn't be happening in // current code anyway. checkState(!wrapper.inTransaction, "already in transaction"); wrapper.inTransaction = true; } @Override public final T atomicUpdate(K key, AtomicUpdate<T> update) { return null; } @Override public final T get(K id) { return null; } } private static class Changes extends AbstractDisabledAccess<Change, Change.Id> implements ChangeAccess { private Changes(NoChangesReviewDbWrapper wrapper, ReviewDb db) { super(wrapper, db.changes()); } @Override public ResultSet<Change> all() { return empty(); } } private static class ChangeMessages extends AbstractDisabledAccess<ChangeMessage, ChangeMessage.Key> implements ChangeMessageAccess { private ChangeMessages(NoChangesReviewDbWrapper wrapper, ReviewDb db) { super(wrapper, db.changeMessages()); } @Override public ResultSet<ChangeMessage> byChange(Change.Id id) throws OrmException { return empty(); } @Override public ResultSet<ChangeMessage> byPatchSet(PatchSet.Id id) throws OrmException { return empty(); } @Override public ResultSet<ChangeMessage> all() throws OrmException { return empty(); } } private static class PatchSets extends AbstractDisabledAccess<PatchSet, PatchSet.Id> implements PatchSetAccess { private PatchSets(NoChangesReviewDbWrapper wrapper, ReviewDb db) { super(wrapper, db.patchSets()); } @Override public ResultSet<PatchSet> byChange(Change.Id id) { return empty(); } @Override public ResultSet<PatchSet> all() { return empty(); } } private static class PatchSetApprovals extends AbstractDisabledAccess<PatchSetApproval, PatchSetApproval.Key> implements PatchSetApprovalAccess { private PatchSetApprovals(NoChangesReviewDbWrapper wrapper, ReviewDb db) { super(wrapper, db.patchSetApprovals()); } @Override public ResultSet<PatchSetApproval> byChange(Change.Id id) { return empty(); } @Override public ResultSet<PatchSetApproval> byPatchSet(PatchSet.Id id) { return empty(); } @Override public ResultSet<PatchSetApproval> byPatchSetUser(PatchSet.Id patchSet, Account.Id account) { return empty(); } @Override public ResultSet<PatchSetApproval> all() { return empty(); } } private static class PatchLineComments extends AbstractDisabledAccess<PatchLineComment, PatchLineComment.Key> implements PatchLineCommentAccess { private PatchLineComments(NoChangesReviewDbWrapper wrapper, ReviewDb db) { super(wrapper, db.patchComments()); } @Override public ResultSet<PatchLineComment> byChange(Change.Id id) { return empty(); } @Override public ResultSet<PatchLineComment> byPatchSet(PatchSet.Id id) { return empty(); } @Override public ResultSet<PatchLineComment> publishedByChangeFile(Change.Id id, String file) { return empty(); } @Override public ResultSet<PatchLineComment> publishedByPatchSet(PatchSet.Id patchset) { return empty(); } @Override public ResultSet<PatchLineComment> draftByPatchSetAuthor( PatchSet.Id patchset, Account.Id author) { return empty(); } @Override public ResultSet<PatchLineComment> draftByChangeFileAuthor( Change.Id id, String file, Account.Id author) { return empty(); } @Override public ResultSet<PatchLineComment> draftByAuthor(Account.Id author) { return empty(); } @Override public ResultSet<PatchLineComment> all() { return empty(); } } }