package net.pterodactylus.sone.core;
import static com.google.common.base.Optional.of;
import static com.google.common.io.ByteStreams.toByteArray;
import static com.google.common.util.concurrent.MoreExecutors.sameThreadExecutor;
import static java.lang.System.currentTimeMillis;
import static org.hamcrest.MatcherAssert.assertThat;
import static org.hamcrest.Matchers.containsString;
import static org.hamcrest.Matchers.instanceOf;
import static org.hamcrest.Matchers.is;
import static org.hamcrest.Matchers.nullValue;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.ArgumentMatchers.anyString;
import static org.mockito.ArgumentMatchers.eq;
import static org.mockito.Mockito.doAnswer;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.never;
import static org.mockito.Mockito.times;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when;
import static org.mockito.hamcrest.MockitoHamcrest.argThat;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
import net.pterodactylus.sone.core.SoneInserter.ManifestCreator;
import net.pterodactylus.sone.core.event.InsertionDelayChangedEvent;
import net.pterodactylus.sone.core.event.SoneEvent;
import net.pterodactylus.sone.core.event.SoneInsertAbortedEvent;
import net.pterodactylus.sone.core.event.SoneInsertedEvent;
import net.pterodactylus.sone.core.event.SoneInsertingEvent;
import net.pterodactylus.sone.data.Album;
import net.pterodactylus.sone.data.Sone;
import net.pterodactylus.sone.main.SonePlugin;
import freenet.keys.FreenetURI;
import freenet.support.api.ManifestElement;
import com.google.common.base.Charsets;
import com.google.common.base.Optional;
import com.google.common.eventbus.AsyncEventBus;
import com.google.common.eventbus.EventBus;
import org.junit.Before;
import org.junit.Test;
import org.mockito.ArgumentCaptor;
import org.mockito.invocation.InvocationOnMock;
import org.mockito.stubbing.Answer;
/**
* Unit test for {@link SoneInserter} and its subclasses.
*
* @author <a href="mailto:bombe@pterodactylus.net">David ‘Bombe’ Roden</a>
*/
public class SoneInserterTest {
private final Core core = mock(Core.class);
private final EventBus eventBus = mock(EventBus.class);
private final FreenetInterface freenetInterface = mock(FreenetInterface.class);
@Before
public void setupCore() {
UpdateChecker updateChecker = mock(UpdateChecker.class);
when(core.getUpdateChecker()).thenReturn(updateChecker);
when(core.getSone(anyString())).thenReturn(Optional.<Sone>absent());
}
@Test
public void insertionDelayIsForwardedToSoneInserter() {
EventBus eventBus = new AsyncEventBus(sameThreadExecutor());
eventBus.register(new SoneInserter(core, eventBus, freenetInterface, "SoneId"));
eventBus.post(new InsertionDelayChangedEvent(15));
assertThat(SoneInserter.getInsertionDelay().get(), is(15));
}
private Sone createSone(FreenetURI insertUri, String fingerprint) {
Sone sone = mock(Sone.class);
when(sone.getInsertUri()).thenReturn(insertUri);
when(sone.getFingerprint()).thenReturn(fingerprint);
when(sone.getRootAlbum()).thenReturn(mock(Album.class));
when(core.getSone(anyString())).thenReturn(of(sone));
return sone;
}
@Test
public void isModifiedIsTrueIfModificationDetectorSaysSo() {
SoneModificationDetector soneModificationDetector = mock(SoneModificationDetector.class);
when(soneModificationDetector.isModified()).thenReturn(true);
SoneInserter soneInserter = new SoneInserter(core, eventBus, freenetInterface, "SoneId", soneModificationDetector, 1);
assertThat(soneInserter.isModified(), is(true));
}
@Test
public void isModifiedIsFalseIfModificationDetectorSaysSo() {
SoneModificationDetector soneModificationDetector = mock(SoneModificationDetector.class);
SoneInserter soneInserter = new SoneInserter(core, eventBus, freenetInterface, "SoneId", soneModificationDetector, 1);
assertThat(soneInserter.isModified(), is(false));
}
@Test
public void lastFingerprintIsStoredCorrectly() {
SoneInserter soneInserter = new SoneInserter(core, eventBus, freenetInterface, "SoneId");
soneInserter.setLastInsertFingerprint("last-fingerprint");
assertThat(soneInserter.getLastInsertFingerprint(), is("last-fingerprint"));
}
@Test
public void soneInserterStopsWhenItShould() {
SoneInserter soneInserter = new SoneInserter(core, eventBus, freenetInterface, "SoneId");
soneInserter.stop();
soneInserter.serviceRun();
}
@Test
public void soneInserterInsertsASoneIfItIsEligible() throws SoneException {
FreenetURI insertUri = mock(FreenetURI.class);
final FreenetURI finalUri = mock(FreenetURI.class);
String fingerprint = "fingerprint";
Sone sone = createSone(insertUri, fingerprint);
SoneModificationDetector soneModificationDetector = mock(SoneModificationDetector.class);
when(soneModificationDetector.isEligibleForInsert()).thenReturn(true);
when(freenetInterface.insertDirectory(eq(insertUri), any(HashMap.class), eq("index.html"))).thenReturn(finalUri);
final SoneInserter soneInserter = new SoneInserter(core, eventBus, freenetInterface, "SoneId", soneModificationDetector, 1);
doAnswer(new Answer<Void>() {
@Override
public Void answer(InvocationOnMock invocation) throws Throwable {
soneInserter.stop();
return null;
}
}).when(core).touchConfiguration();
soneInserter.serviceRun();
ArgumentCaptor<SoneEvent> soneEvents = ArgumentCaptor.forClass(SoneEvent.class);
verify(freenetInterface).insertDirectory(eq(insertUri), any(HashMap.class), eq("index.html"));
verify(eventBus, times(2)).post(soneEvents.capture());
assertThat(soneEvents.getAllValues().get(0), instanceOf(SoneInsertingEvent.class));
assertThat(soneEvents.getAllValues().get(0).sone(), is(sone));
assertThat(soneEvents.getAllValues().get(1), instanceOf(SoneInsertedEvent.class));
assertThat(soneEvents.getAllValues().get(1).sone(), is(sone));
}
@Test
public void soneInserterBailsOutIfItIsStoppedWhileInserting() throws SoneException {
FreenetURI insertUri = mock(FreenetURI.class);
final FreenetURI finalUri = mock(FreenetURI.class);
String fingerprint = "fingerprint";
Sone sone = createSone(insertUri, fingerprint);
SoneModificationDetector soneModificationDetector = mock(SoneModificationDetector.class);
when(soneModificationDetector.isEligibleForInsert()).thenReturn(true);
final SoneInserter soneInserter = new SoneInserter(core, eventBus, freenetInterface, "SoneId", soneModificationDetector, 1);
when(freenetInterface.insertDirectory(eq(insertUri), any(HashMap.class), eq("index.html"))).thenAnswer(new Answer<FreenetURI>() {
@Override
public FreenetURI answer(InvocationOnMock invocation) throws Throwable {
soneInserter.stop();
return finalUri;
}
});
soneInserter.serviceRun();
ArgumentCaptor<SoneEvent> soneEvents = ArgumentCaptor.forClass(SoneEvent.class);
verify(freenetInterface).insertDirectory(eq(insertUri), any(HashMap.class), eq("index.html"));
verify(eventBus, times(2)).post(soneEvents.capture());
assertThat(soneEvents.getAllValues().get(0), instanceOf(SoneInsertingEvent.class));
assertThat(soneEvents.getAllValues().get(0).sone(), is(sone));
assertThat(soneEvents.getAllValues().get(1), instanceOf(SoneInsertedEvent.class));
assertThat(soneEvents.getAllValues().get(1).sone(), is(sone));
verify(core, never()).touchConfiguration();
}
@Test
public void soneInserterDoesNotInsertSoneIfItIsNotEligible() throws SoneException {
FreenetURI insertUri = mock(FreenetURI.class);
String fingerprint = "fingerprint";
Sone sone = createSone(insertUri, fingerprint);
SoneModificationDetector soneModificationDetector = mock(SoneModificationDetector.class);
final SoneInserter soneInserter = new SoneInserter(core, eventBus, freenetInterface, "SoneId", soneModificationDetector, 1);
new Thread(new Runnable() {
@Override
public void run() {
try {
Thread.sleep(500);
} catch (InterruptedException ie1) {
throw new RuntimeException(ie1);
}
soneInserter.stop();
}
}).start();
soneInserter.serviceRun();
verify(freenetInterface, never()).insertDirectory(eq(insertUri), any(HashMap.class), eq("index.html"));
verify(eventBus, never()).post(argThat(org.hamcrest.Matchers.any(SoneEvent.class)));
}
@Test
public void soneInserterPostsAbortedEventIfAnExceptionOccurs() throws SoneException {
FreenetURI insertUri = mock(FreenetURI.class);
String fingerprint = "fingerprint";
Sone sone = createSone(insertUri, fingerprint);
SoneModificationDetector soneModificationDetector = mock(SoneModificationDetector.class);
when(soneModificationDetector.isEligibleForInsert()).thenReturn(true);
final SoneInserter soneInserter = new SoneInserter(core, eventBus, freenetInterface, "SoneId", soneModificationDetector, 1);
final SoneException soneException = new SoneException(new Exception());
when(freenetInterface.insertDirectory(eq(insertUri), any(HashMap.class), eq("index.html"))).thenAnswer(new Answer<FreenetURI>() {
@Override
public FreenetURI answer(InvocationOnMock invocation) throws Throwable {
soneInserter.stop();
throw soneException;
}
});
soneInserter.serviceRun();
ArgumentCaptor<SoneEvent> soneEvents = ArgumentCaptor.forClass(SoneEvent.class);
verify(freenetInterface).insertDirectory(eq(insertUri), any(HashMap.class), eq("index.html"));
verify(eventBus, times(2)).post(soneEvents.capture());
assertThat(soneEvents.getAllValues().get(0), instanceOf(SoneInsertingEvent.class));
assertThat(soneEvents.getAllValues().get(0).sone(), is(sone));
assertThat(soneEvents.getAllValues().get(1), instanceOf(SoneInsertAbortedEvent.class));
assertThat(soneEvents.getAllValues().get(1).sone(), is(sone));
verify(core, never()).touchConfiguration();
}
@Test
public void soneInserterExitsIfSoneIsUnknown() {
SoneModificationDetector soneModificationDetector =
mock(SoneModificationDetector.class);
SoneInserter soneInserter =
new SoneInserter(core, eventBus, freenetInterface, "SoneId",
soneModificationDetector, 1);
when(soneModificationDetector.isEligibleForInsert()).thenReturn(true);
when(core.getSone("SoneId")).thenReturn(Optional.<Sone>absent());
soneInserter.serviceRun();
}
@Test
public void soneInserterCatchesExceptionAndContinues() {
SoneModificationDetector soneModificationDetector =
mock(SoneModificationDetector.class);
final SoneInserter soneInserter =
new SoneInserter(core, eventBus, freenetInterface, "SoneId",
soneModificationDetector, 1);
Answer<Optional<Sone>> stopInserterAndThrowException =
new Answer<Optional<Sone>>() {
@Override
public Optional<Sone> answer(
InvocationOnMock invocation) {
soneInserter.stop();
throw new NullPointerException();
}
};
when(soneModificationDetector.isEligibleForInsert()).thenAnswer(
stopInserterAndThrowException);
soneInserter.serviceRun();
}
@Test
public void templateIsRenderedCorrectlyForManifestElement()
throws IOException {
Map<String, Object> soneProperties = new HashMap<String, Object>();
soneProperties.put("id", "SoneId");
ManifestCreator manifestCreator = new ManifestCreator(core, soneProperties);
long now = currentTimeMillis();
when(core.getStartupTime()).thenReturn(now);
ManifestElement manifestElement = manifestCreator.createManifestElement("test.txt", "plain/text; charset=utf-8", "sone-inserter-manifest.txt");
assertThat(manifestElement.getName(), is("test.txt"));
assertThat(manifestElement.getMimeTypeOverride(), is("plain/text; charset=utf-8"));
String templateContent = new String(toByteArray(manifestElement.getData().getInputStream()), Charsets.UTF_8);
assertThat(templateContent, containsString("Sone Version: " + SonePlugin.getPluginVersion() + "\n"));
assertThat(templateContent, containsString("Core Startup: " + now + "\n"));
assertThat(templateContent, containsString("Sone ID: " + "SoneId" + "\n"));
}
@Test
public void invalidTemplateReturnsANullManifestElement() {
Map<String, Object> soneProperties = new HashMap<String, Object>();
ManifestCreator manifestCreator = new ManifestCreator(core, soneProperties);
assertThat(manifestCreator.createManifestElement("test.txt",
"plain/text; charset=utf-8",
"sone-inserter-invalid-manifest.txt"),
nullValue());
}
@Test
public void errorWhileRenderingTemplateReturnsANullManifestElement() {
Map<String, Object> soneProperties = new HashMap<String, Object>();
ManifestCreator manifestCreator = new ManifestCreator(core, soneProperties);
when(core.toString()).thenThrow(NullPointerException.class);
assertThat(manifestCreator.createManifestElement("test.txt",
"plain/text; charset=utf-8",
"sone-inserter-faulty-manifest.txt"),
nullValue());
}
}