X-Git-Url: https://git.pterodactylus.net/?a=blobdiff_plain;f=src%2Fmain%2Fjava%2Fnet%2Fpterodactylus%2Fsone%2Fcore%2FFreenetInterface.java;h=e22f407488ee7ca6e468671381031d4ca7132ead;hb=0521aa52048c343e9f2ea2920d5f9ac65d724272;hp=dae6f6853119fcffe07af46c219333502f29d256;hpb=adf590c0b33f1cc3f136d8d3a714f7051407535f;p=Sone.git diff --git a/src/main/java/net/pterodactylus/sone/core/FreenetInterface.java b/src/main/java/net/pterodactylus/sone/core/FreenetInterface.java index dae6f68..e22f407 100644 --- a/src/main/java/net/pterodactylus/sone/core/FreenetInterface.java +++ b/src/main/java/net/pterodactylus/sone/core/FreenetInterface.java @@ -18,14 +18,15 @@ package net.pterodactylus.sone.core; import java.net.MalformedURLException; +import java.util.Collections; import java.util.HashMap; import java.util.Map; import java.util.logging.Level; import java.util.logging.Logger; import net.pterodactylus.sone.data.Sone; +import net.pterodactylus.util.collection.Pair; import net.pterodactylus.util.logging.Logging; -import net.pterodactylus.util.service.AbstractService; import com.db4o.ObjectContainer; @@ -46,7 +47,7 @@ import freenet.node.RequestStarter; * * @author David ‘Bombe’ Roden */ -public class FreenetInterface extends AbstractService { +public class FreenetInterface { /** The logger. */ private static final Logger logger = Logging.getLogger(FreenetInterface.class); @@ -60,18 +61,18 @@ public class FreenetInterface extends AbstractService { /** The USK callbacks. */ private final Map soneUskCallbacks = new HashMap(); + /** The not-Sone-related USK callbacks. */ + private final Map uriUskCallbacks = Collections.synchronizedMap(new HashMap()); + /** * Creates a new Freenet interface. * * @param node * The node to interact with - * @param client - * The high-level client */ - public FreenetInterface(Node node, HighLevelSimpleClient client) { - super("Sone Freenet Interface"); + public FreenetInterface(Node node) { this.node = node; - this.client = client; + this.client = node.clientCore.makeClient(RequestStarter.INTERACTIVE_PRIORITY_CLASS, false, true); } // @@ -85,17 +86,22 @@ public class FreenetInterface extends AbstractService { * The URI to fetch * @return The result of the fetch, or {@code null} if an error occured */ - public FetchResult fetchUri(FreenetURI uri) { - logger.entering(FreenetInterface.class.getName(), "fetchUri(FreenetURI)", uri); + public Pair fetchUri(FreenetURI uri) { FetchResult fetchResult = null; - try { - fetchResult = client.fetch(uri); - } catch (FetchException fe1) { - logger.log(Level.WARNING, "Could not fetch “" + uri + "”!", fe1); - } finally { - logger.exiting(FreenetInterface.class.getName(), "fetchUri(FreenetURI)", fetchResult); + FreenetURI currentUri = new FreenetURI(uri); + while (true) { + try { + fetchResult = client.fetch(currentUri); + return new Pair(currentUri, fetchResult); + } catch (FetchException fe1) { + if (fe1.getMode() == FetchException.PERMANENT_REDIRECT) { + currentUri = fe1.newURI; + continue; + } + logger.log(Level.WARNING, "Could not fetch “" + uri + "”!", fe1); + return null; + } } - return fetchResult; } /** @@ -146,11 +152,17 @@ public class FreenetInterface extends AbstractService { @Override @SuppressWarnings("synthetic-access") public void onFoundEdition(long edition, USK key, ObjectContainer objectContainer, ClientContext clientContext, boolean metadata, short codec, byte[] data, boolean newKnownGood, boolean newSlotToo) { - logger.log(Level.FINE, "Found USK update for Sone “%s” at %s.", new Object[] { sone, key }); - if (newKnownGood) { - sone.updateUris(key.getURI()); + logger.log(Level.FINE, "Found USK update for Sone “%s” at %s, new known good: %s, new slot too: %s.", new Object[] { sone, key, newKnownGood, newSlotToo }); + if (edition > sone.getLatestEdition()) { + sone.setLatestEdition(edition); + new Thread(new Runnable() { + + @Override + public void run() { + soneDownloader.fetchSone(sone); + } + }, "Sone Downloader").start(); } - soneDownloader.fetchSone(sone); } @Override @@ -164,7 +176,7 @@ public class FreenetInterface extends AbstractService { } }; soneUskCallbacks.put(sone.getId(), uskCallback); - node.clientCore.uskManager.subscribe(USK.create(sone.getRequestUri()), uskCallback, true, (HighLevelSimpleClientImpl) client); + node.clientCore.uskManager.subscribe(USK.create(sone.getRequestUri()), uskCallback, (System.currentTimeMillis() - sone.getTime()) < 7 * 24 * 60 * 60 * 1000, (HighLevelSimpleClientImpl) client); } catch (MalformedURLException mue1) { logger.log(Level.WARNING, "Could not subscribe USK “" + sone.getRequestUri() + "”!", mue1); } @@ -182,10 +194,90 @@ public class FreenetInterface extends AbstractService { return; } try { + logger.log(Level.FINEST, "Unsubscribing from USK for %s…", new Object[] { sone }); node.clientCore.uskManager.unsubscribe(USK.create(sone.getRequestUri()), uskCallback); } catch (MalformedURLException mue1) { logger.log(Level.FINE, "Could not unsubscribe USK “" + sone.getRequestUri() + "”!", mue1); } } + /** + * Registers an arbitrary URI and calls the given callback if a new edition + * is found. + * + * @param uri + * The URI to watch + * @param callback + * The callback to call + */ + public void registerUsk(FreenetURI uri, final Callback callback) { + USKCallback uskCallback = new USKCallback() { + + @Override + public void onFoundEdition(long edition, USK key, ObjectContainer objectContainer, ClientContext clientContext, boolean metadata, short codec, byte[] data, boolean newKnownGood, boolean newSlotToo) { + callback.editionFound(key.getURI(), edition, newKnownGood, newSlotToo); + } + + @Override + public short getPollingPriorityNormal() { + return RequestStarter.PREFETCH_PRIORITY_CLASS; + } + + @Override + public short getPollingPriorityProgress() { + return RequestStarter.INTERACTIVE_PRIORITY_CLASS; + } + + }; + try { + node.clientCore.uskManager.subscribe(USK.create(uri), uskCallback, true, (HighLevelSimpleClientImpl) client); + uriUskCallbacks.put(uri, uskCallback); + } catch (MalformedURLException mue1) { + logger.log(Level.WARNING, "Could not subscribe to USK: " + uri, uri); + } + } + + /** + * Unregisters the USK watcher for the given URI. + * + * @param uri + * The URI to unregister the USK watcher for + */ + public void unregisterUsk(FreenetURI uri) { + USKCallback uskCallback = uriUskCallbacks.remove(uri); + if (uskCallback == null) { + logger.log(Level.INFO, "Could not unregister unknown USK: " + uri); + return; + } + try { + node.clientCore.uskManager.unsubscribe(USK.create(uri), uskCallback); + } catch (MalformedURLException mue1) { + logger.log(Level.INFO, "Could not unregister invalid USK: " + uri); + } + } + + /** + * Callback for USK watcher events. + * + * @author David ‘Bombe’ Roden + */ + public static interface Callback { + + /** + * Notifies a listener that a new edition was found for a URI. + * + * @param uri + * The URI that a new edition was found for + * @param edition + * The found edition + * @param newKnownGood + * Whether the found edition was actually fetched + * @param newSlot + * Whether the found edition is higher than all previously + * found editions + */ + public void editionFound(FreenetURI uri, long edition, boolean newKnownGood, boolean newSlot); + + } + }