blob: 63db9396fb60fd3d7e888f77c280021724b488aa [file] [log] [blame]
// Copyright (C) 2018 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.ericsson.gerrit.plugins.highavailability.forwarder;
import static com.google.common.truth.Truth.assertThat;
import static org.junit.Assert.fail;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.doAnswer;
import static org.mockito.Mockito.doThrow;
import static org.mockito.Mockito.times;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when;
import com.ericsson.gerrit.plugins.highavailability.forwarder.ForwardedIndexingHandler.Operation;
import com.google.gerrit.reviewdb.client.Change;
import com.google.gerrit.reviewdb.server.ReviewDb;
import com.google.gerrit.server.change.ChangeFinder;
import com.google.gerrit.server.index.change.ChangeIndexer;
import com.google.gerrit.server.notedb.ChangeNotes;
import com.google.gerrit.server.project.NoSuchChangeException;
import com.google.gerrit.server.util.time.TimeUtil;
import com.google.gwtorm.server.OrmException;
import com.google.gwtorm.server.SchemaFactory;
import java.io.IOException;
import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.ExpectedException;
import org.junit.runner.RunWith;
import org.mockito.Mock;
import org.mockito.junit.MockitoJUnitRunner;
import org.mockito.stubbing.Answer;
@RunWith(MockitoJUnitRunner.class)
public class ForwardedIndexChangeHandlerTest {
private static final int TEST_CHANGE_NUMBER = 123;
private static String TEST_PROJECT = "test/project";
private static String TEST_CHANGE_ID = TEST_PROJECT + "~" + TEST_CHANGE_NUMBER;
private static final boolean CHANGE_EXISTS = true;
private static final boolean CHANGE_DOES_NOT_EXIST = false;
private static final boolean DO_NOT_THROW_IO_EXCEPTION = false;
private static final boolean DO_NOT_THROW_ORM_EXCEPTION = false;
private static final boolean THROW_IO_EXCEPTION = true;
private static final boolean THROW_ORM_EXCEPTION = true;
@Rule public ExpectedException exception = ExpectedException.none();
@Mock private ChangeIndexer indexerMock;
@Mock private SchemaFactory<ReviewDb> schemaFactoryMock;
@Mock private ReviewDb dbMock;
@Mock private ChangeFinder changeFinderMock;
@Mock private ChangeNotes changeNotes;
private ForwardedIndexChangeHandler handler;
private Change.Id id;
private Change change;
@Before
public void setUp() throws Exception {
when(schemaFactoryMock.open()).thenReturn(dbMock);
id = new Change.Id(TEST_CHANGE_NUMBER);
change = new Change(null, id, null, null, TimeUtil.nowTs());
when(changeNotes.getChange()).thenReturn(change);
handler = new ForwardedIndexChangeHandler(indexerMock, schemaFactoryMock, changeFinderMock);
}
@Test
public void changeIsIndexed() throws Exception {
setupChangeAccessRelatedMocks(CHANGE_EXISTS);
handler.index(TEST_CHANGE_ID, Operation.INDEX);
verify(indexerMock, times(1)).index(any(ReviewDb.class), any(Change.class));
}
@Test
public void changeIsDeletedFromIndex() throws Exception {
handler.index(TEST_CHANGE_ID, Operation.DELETE);
verify(indexerMock, times(1)).delete(id);
}
@Test
public void changeToIndexDoesNotExist() throws Exception {
setupChangeAccessRelatedMocks(CHANGE_DOES_NOT_EXIST);
handler.index(TEST_CHANGE_ID, Operation.INDEX);
verify(indexerMock, times(1)).delete(id);
}
@Test
public void schemaThrowsExceptionWhenLookingUpForChange() throws Exception {
setupChangeAccessRelatedMocks(CHANGE_EXISTS, THROW_ORM_EXCEPTION);
exception.expect(OrmException.class);
handler.index(TEST_CHANGE_ID, Operation.INDEX);
}
@Test
public void indexerThrowsNoSuchChangeExceptionTryingToPostChange() throws Exception {
doThrow(new NoSuchChangeException(id)).when(schemaFactoryMock).open();
handler.index(TEST_CHANGE_ID, Operation.INDEX);
verify(indexerMock, times(1)).delete(id);
}
@Test
public void indexerThrowsNestedNoSuchChangeExceptionTryingToPostChange() throws Exception {
OrmException e = new OrmException("test", new NoSuchChangeException(id));
doThrow(e).when(schemaFactoryMock).open();
handler.index(TEST_CHANGE_ID, Operation.INDEX);
verify(indexerMock, times(1)).delete(id);
}
@Test
public void indexerThrowsIOExceptionTryingToIndexChange() throws Exception {
setupChangeAccessRelatedMocks(CHANGE_EXISTS, DO_NOT_THROW_ORM_EXCEPTION, THROW_IO_EXCEPTION);
exception.expect(IOException.class);
handler.index(TEST_CHANGE_ID, Operation.INDEX);
}
@Test
public void shouldSetAndUnsetForwardedContext() throws Exception {
setupChangeAccessRelatedMocks(CHANGE_EXISTS);
// this doAnswer is to allow to assert that context is set to forwarded
// while cache eviction is called.
doAnswer(
(Answer<Void>)
invocation -> {
assertThat(Context.isForwardedEvent()).isTrue();
return null;
})
.when(indexerMock)
.index(any(ReviewDb.class), any(Change.class));
assertThat(Context.isForwardedEvent()).isFalse();
handler.index(TEST_CHANGE_ID, Operation.INDEX);
assertThat(Context.isForwardedEvent()).isFalse();
verify(indexerMock, times(1)).index(any(ReviewDb.class), any(Change.class));
}
@Test
public void shouldSetAndUnsetForwardedContextEvenIfExceptionIsThrown() throws Exception {
setupChangeAccessRelatedMocks(CHANGE_EXISTS);
doAnswer(
(Answer<Void>)
invocation -> {
assertThat(Context.isForwardedEvent()).isTrue();
throw new IOException("someMessage");
})
.when(indexerMock)
.index(any(ReviewDb.class), any(Change.class));
assertThat(Context.isForwardedEvent()).isFalse();
try {
handler.index(TEST_CHANGE_ID, Operation.INDEX);
fail("should have thrown an IOException");
} catch (IOException e) {
assertThat(e.getMessage()).isEqualTo("someMessage");
}
assertThat(Context.isForwardedEvent()).isFalse();
verify(indexerMock, times(1)).index(any(ReviewDb.class), any(Change.class));
}
private void setupChangeAccessRelatedMocks(boolean changeExist) throws Exception {
setupChangeAccessRelatedMocks(
changeExist, DO_NOT_THROW_ORM_EXCEPTION, DO_NOT_THROW_IO_EXCEPTION);
}
private void setupChangeAccessRelatedMocks(boolean changeExist, boolean ormException)
throws OrmException, IOException {
setupChangeAccessRelatedMocks(changeExist, ormException, DO_NOT_THROW_IO_EXCEPTION);
}
private void setupChangeAccessRelatedMocks(
boolean changeExists, boolean ormException, boolean ioException)
throws OrmException, IOException {
if (ormException) {
doThrow(new OrmException("")).when(schemaFactoryMock).open();
} else {
when(schemaFactoryMock.open()).thenReturn(dbMock);
if (changeExists) {
when(changeFinderMock.findOne(TEST_CHANGE_ID)).thenReturn(changeNotes);
if (ioException) {
doThrow(new IOException("io-error"))
.when(indexerMock)
.index(any(ReviewDb.class), any(Change.class));
}
} else {
when(changeFinderMock.findOne(TEST_CHANGE_ID)).thenReturn(null);
}
}
}
}