From 82fb65193f3f685e9eb7e52f5cad0832e2f32b71 Mon Sep 17 00:00:00 2001
From: Dmitry Panin
Type.PIECE
messages.
+ *
+ * + * To impose rate limits, we only want to throttle when + * processing PIECE messages. All other peer messages + * should be exchanged as quickly as possible. + *
+ * + * @author ptgoetz + * + */ + private abstract class RateLimitThread extends Thread{ + protected Rate rate = new Rate(); + protected long sleep = 1000; + + protected void rateLimit(double maxRate, long messageSize, PeerMessage message){ + if(message.getType() == Type.PIECE && maxRate > 0){ + try { + this.rate.add(messageSize); + // continuously adjust the sleep time to try to hit our + // target rate limit + if(rate.get() > (maxRate * 1024)){ + Thread.sleep(this.sleep); + this.sleep += 50; + } else { + this.sleep -= 50; + } + if(this.sleep < 0){ + this.sleep = 0; + } + } catch (InterruptedException e) { + // not critical + } + } + } + } /** * Incoming messages thread. @@ -200,9 +239,7 @@ public void close() { * * @author mpetazzoni */ - private class IncomingThread extends Thread { - private Rate rate = new Rate(); - private long sleep = 1000; + private class IncomingThread extends RateLimitThread { @Override public void run() { @@ -249,23 +286,7 @@ public void run() { PeerMessage message = PeerMessage.parse(buffer, torrent); logger.trace("Received {} from {}", message, peer); - // throttling - if(message.getType() == Type.PIECE && PeerExchange.this.torrent.getMaxDownloadRate() > 0){ - try { - rate.add(size); - if(rate.get() > (PeerExchange.this.torrent.getMaxDownloadRate() * 1024)){ - Thread.sleep(this.sleep); - this.sleep += 50; - } else { - this.sleep -= 50; - } - if(this.sleep < 0){ - this.sleep = 0; - } - } catch (InterruptedException e) { - // not critical - } - } + rateLimit(PeerExchange.this.torrent.getMaxDownloadRate(), size, message); for (MessageListener listener : listeners) { listener.handleMessage(message); @@ -301,9 +322,7 @@ public void run() { * * @author mpetazzoni */ - private class OutgoingThread extends Thread { - private Rate rate = new Rate(); - private long sleep = 1000; + private class OutgoingThread extends RateLimitThread { @Override public void run() { @@ -338,22 +357,7 @@ public void run() { } } - if(message.getType() == Type.PIECE && PeerExchange.this.torrent.getMaxUploadRate() > 0){ - try { - rate.add(size); - if(rate.get() > (PeerExchange.this.torrent.getMaxUploadRate() * 1024)){ - Thread.sleep(this.sleep); - this.sleep += 50; - } else { - this.sleep -= 50; - } - if(this.sleep < 0){ - this.sleep = 0; - } - } catch (InterruptedException e) { - // not critical - } - } + rateLimit(PeerExchange.this.torrent.getMaxUploadRate(), size, message); } catch (InterruptedException ie) { // Ignore and potentially terminate } From 3ef9e9b1545cdc96135e2bfa72dda6ca96368d0a Mon Sep 17 00:00:00 2001 From: "P. Taylor Goetz"
+ * The algorithm is functional, but could certainly be
+ * improved upon. One obvious drawback is that with large
+ * changes in maxRate
, it will take a while
+ * for the sleep time to adjust and the throttled rate
+ * to "smooth out."
+ *
+ * Ideally, it would calculate the optimal sleep time + * necessary to hit a desired throughput rather than + * continuously adjust toward a goal. + *
+ * + * @param maxRate the target rate in kB/second + * @param messageSize the size, in bytes, of the last message read/written + * @param message the lastPeerMessage
read/written
+ */
+ protected void rateLimit(double maxRate, long messageSize, PeerMessage message) {
+ if(message.getType() == Type.PIECE && maxRate > 0) {
try {
this.rate.add(messageSize);
// continuously adjust the sleep time to try to hit our
// target rate limit
- if(rate.get() > (maxRate * 1024)){
+ if(rate.get() > (maxRate * 1024)) {
Thread.sleep(this.sleep);
this.sleep += 50;
} else {
this.sleep -= 50;
}
- if(this.sleep < 0){
+ if(this.sleep < 0) {
this.sleep = 0;
}
} catch (InterruptedException e) {
From 00bc8f287aa361850c7b8638ce243bd182747025 Mon Sep 17 00:00:00 2001
From: "P. Taylor Goetz" Type.PIECE
messages.
+ * for PIECE
messages.
*
* - * To impose rate limits, we only want to throttle when - * processing PIECE messages. All other peer messages - * should be exchanged as quickly as possible. + * To impose rate limits, we only want to throttle when processing PIECE + * messages. All other peer messages should be exchanged as quickly as + * possible. *
* * @author ptgoetz - * */ private abstract class RateLimitThread extends Thread { - protected Rate rate = new Rate(); + + protected final Rate rate = new Rate(); protected long sleep = 1000; - + /** - * Dynamically determines an amount of time to sleep, based - * on the average read/write throughput. + * Dynamically determines an amount of time to sleep, based on the + * average read/write throughput. * *
- * The algorithm is functional, but could certainly be
- * improved upon. One obvious drawback is that with large
- * changes in maxRate
, it will take a while
- * for the sleep time to adjust and the throttled rate
- * to "smooth out."
+ * The algorithm is functional, but could certainly be improved upon.
+ * One obvious drawback is that with large changes in
+ * maxRate
, it will take a while for the sleep time to
+ * adjust and the throttled rate to "smooth out."
*
- * Ideally, it would calculate the optimal sleep time - * necessary to hit a desired throughput rather than - * continuously adjust toward a goal. + * Ideally, it would calculate the optimal sleep time necessary to hit + * a desired throughput rather than continuously adjust toward a goal. *
* - * @param maxRate the target rate in kB/second - * @param messageSize the size, in bytes, of the last message read/written - * @param message the lastPeerMessage
read/written
+ * @param maxRate the target rate in kB/second.
+ * @param messageSize the size, in bytes, of the last message read/written.
+ * @param message the last PeerMessage
read/written.
*/
protected void rateLimit(double maxRate, long messageSize, PeerMessage message) {
- if(message.getType() == Type.PIECE && maxRate > 0) {
- try {
- this.rate.add(messageSize);
- // continuously adjust the sleep time to try to hit our
- // target rate limit
- if(rate.get() > (maxRate * 1024)) {
- Thread.sleep(this.sleep);
- this.sleep += 50;
- } else {
- this.sleep -= 50;
- }
- if(this.sleep < 0) {
- this.sleep = 0;
- }
- } catch (InterruptedException e) {
- // not critical
+ if (message.getType() != Type.PIECE || maxRate <= 0) {
+ return;
+ }
+
+ try {
+ this.rate.add(messageSize);
+
+ // Continuously adjust the sleep time to try to hit our target
+ // rate limit.
+ if (rate.get() > (maxRate * 1024)) {
+ Thread.sleep(this.sleep);
+ this.sleep += 50;
+ } else {
+ this.sleep = this.sleep > 50
+ ? this.sleep - 50
+ : 0;
}
+ } catch (InterruptedException e) {
+ // Not critical, eat it.
}
}
}
@@ -307,8 +307,10 @@ public void run() {
try {
PeerMessage message = PeerMessage.parse(buffer, torrent);
logger.trace("Received {} from {}", message, peer);
-
- rateLimit(PeerExchange.this.torrent.getMaxDownloadRate(), size, message);
+
+ // Wait if needed to reach configured download rate.
+ this.rateLimit(PeerExchange.this.torrent.getMaxDownloadRate(),
+ size, message);
for (MessageListener listener : listeners) {
listener.handleMessage(message);
@@ -345,7 +347,7 @@ public void run() {
* @author mpetazzoni
*/
private class OutgoingThread extends RateLimitThread {
-
+
@Override
public void run() {
try {
@@ -378,8 +380,10 @@ public void run() {
"Reached end of stream while writing");
}
}
-
- rateLimit(PeerExchange.this.torrent.getMaxUploadRate(), size, message);
+
+ // Wait if needed to reach configured upload rate.
+ this.rateLimit(PeerExchange.this.torrent.getMaxUploadRate(),
+ size, message);
} catch (InterruptedException ie) {
// Ignore and potentially terminate
}
From 64b886552678578d48edee5a860c32387d132246 Mon Sep 17 00:00:00 2001
From: Dan Oxlade
* When the download is complete, the client switches to seeding mode for
* as long as requested in the share()
call, if seeding was
- * requested. If not, the StopSeedingTask will execute immediately to stop
- * the client's main loop.
+ * requested. If not, the {@link ClientShutdown} will execute
+ * immediately to stop the client's main loop.
*
- * This method is called by {@link #stop()} to make sure all connections + * This method is called by {@link Announce#stop()} to make sure all connections * are correctly closed when the announce thread is asked to stop. *
*/ diff --git a/src/main/java/com/turn/ttorrent/client/announce/UDPTrackerClient.java b/src/main/java/com/turn/ttorrent/client/announce/UDPTrackerClient.java index 01b1c0420..5df3229b0 100644 --- a/src/main/java/com/turn/ttorrent/client/announce/UDPTrackerClient.java +++ b/src/main/java/com/turn/ttorrent/client/announce/UDPTrackerClient.java @@ -239,7 +239,7 @@ public void announce(AnnounceRequestMessage.RequestEvent event, * ** Verifies the transaction ID of the message before passing it over to - * {@link Announce#handleTrackerAnnounceResponse()}. + * any registered {@link AnnounceResponseListener}. *
* * @param message The message received from the tracker in response to the @@ -352,7 +352,7 @@ private void send(ByteBuffer data) { * * @param attempt The attempt number, used to calculate the timeout for the * receive operation. - * @retun Returns a {@link ByteBuffer} containing the packet data. + * @return Returns a {@link ByteBuffer} containing the packet data. */ private ByteBuffer recv(int attempt) throws IOException, SocketException, SocketTimeoutException { diff --git a/src/main/java/com/turn/ttorrent/common/Torrent.java b/src/main/java/com/turn/ttorrent/common/Torrent.java index e468dc560..de346067c 100644 --- a/src/main/java/com/turn/ttorrent/common/Torrent.java +++ b/src/main/java/com/turn/ttorrent/common/Torrent.java @@ -133,7 +133,6 @@ public TorrentFile(File file, long size) { * BitTorrent specification) and create a Torrent object from it. * * @param torrent The meta-info byte data. - * @param parent The parent directory or location of the torrent files. * @param seeder Whether we'll be seeding for this torrent or not. * @throws IOException When the info dictionary can't be read or * encoded and hashed back to create the torrent's SHA-1 hash. @@ -592,7 +591,7 @@ public static Torrent create(File source, List+ * If an interface name is given, return the first usable IPv4 address for + * that interface. If no interface name is given or if that interface + * doesn't have an IPv4 address, return's localhost address (if IPv4). + *
+ * + *+ * It is understood this makes the client IPv4 only, but it is important to + * remember that most BitTorrent extensions (like compact peer lists from + * trackers and UDP tracker support) are IPv4-only anyway. + *
+ * + * @param iface The network interface name. + * @return A usable IPv4 address as a {@link Inet4Address}. + * @throws UnsupportedAddressTypeException If no IPv4 address was available + * to bind on. + */ + private static Inet4Address getIPv4Address(String iface) + throws SocketException, UnsupportedAddressTypeException, + UnknownHostException { + if (iface != null) { + Enumeration+ * You can use the {@code main()} function of this class to read or create + * torrent files. See usage for details. + *
+ * + * TODO: support multiple announce URLs. + */ + public static void main(String[] args) { + BasicConfigurator.configure(new ConsoleAppender( + new PatternLayout("%-5p: %m%n"))); + + CmdLineParser parser = new CmdLineParser(); + CmdLineParser.Option help = parser.addBooleanOption('h', "help"); + CmdLineParser.Option filename = parser.addStringOption('t', "torrent"); + CmdLineParser.Option create = parser.addBooleanOption('c', "create"); + CmdLineParser.Option announce = parser.addStringOption('a', "announce"); + + try { + parser.parse(args); + } catch (CmdLineParser.OptionException oe) { + System.err.println(oe.getMessage()); + usage(System.err); + System.exit(1); + } + + // Display help and exit if requested + if (Boolean.TRUE.equals((Boolean)parser.getOptionValue(help))) { + usage(System.out); + System.exit(0); + } + + String filenameValue = (String)parser.getOptionValue(filename); + if (filenameValue == null) { + usage(System.err, "Torrent file must be provided!"); + System.exit(1); + } + + Boolean createFlag = (Boolean)parser.getOptionValue(create); + String announceURL = (String)parser.getOptionValue(announce); + + String[] otherArgs = parser.getRemainingArgs(); + + if (Boolean.TRUE.equals(createFlag) && + (otherArgs.length != 1 || announceURL == null)) { + usage(System.err, "Announce URL and a file or directory must be " + + "provided to create a torrent file!"); + System.exit(1); + } + + OutputStream fos = null; + try { + if (Boolean.TRUE.equals(createFlag)) { + if (filenameValue != null) { + fos = new FileOutputStream(filenameValue); + } else { + fos = System.out; + } + + URI announceURI = new URI(announceURL); + File source = new File(otherArgs[0]); + if (!source.exists() || !source.canRead()) { + throw new IllegalArgumentException( + "Cannot access source file or directory " + + source.getName()); + } + + String creator = String.format("%s (ttorrent)", + System.getProperty("user.name")); + + Torrent torrent = null; + if (source.isDirectory()) { + File[] files = source.listFiles(); + Arrays.sort(files); + torrent = Torrent.create(source, Arrays.asList(files), + announceURI, creator); + } else { + torrent = Torrent.create(source, announceURI, creator); + } + + torrent.save(fos); + } else { + Torrent.load(new File(filenameValue), true); + } + } catch (Exception e) { + logger.error("{}", e.getMessage(), e); + System.exit(2); + } finally { + if (fos != null && fos != System.out) { + try { + fos.close(); + } catch (IOException ioe) { + } + } + } + } +} diff --git a/src/main/java/com/turn/ttorrent/cli/TrackerMain.java b/src/main/java/com/turn/ttorrent/cli/TrackerMain.java new file mode 100644 index 000000000..bd3ffa473 --- /dev/null +++ b/src/main/java/com/turn/ttorrent/cli/TrackerMain.java @@ -0,0 +1,118 @@ +/** + * Copyright (C) 2011-2013 Turn, Inc. + * + * 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.turn.ttorrent.cli; + +import com.turn.ttorrent.tracker.TrackedTorrent; +import com.turn.ttorrent.tracker.Tracker; + +import java.io.File; +import java.io.FilenameFilter; +import java.io.PrintStream; +import java.net.InetSocketAddress; + +import jargs.gnu.CmdLineParser; +import org.apache.log4j.BasicConfigurator; +import org.apache.log4j.ConsoleAppender; +import org.apache.log4j.PatternLayout; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * Command-line entry-point for starting a {@link Tracker} + */ +public class TrackerMain { + + private static final Logger logger = + LoggerFactory.getLogger(TrackerMain.class); + + /** + * Display program usage on the given {@link PrintStream}. + */ + private static void usage(PrintStream s) { + s.println("usage: Tracker [options] [directory]"); + s.println(); + s.println("Available options:"); + s.println(" -h,--help Show this help and exit."); + s.println(" -p,--port PORT Bind to port PORT."); + s.println(); + } + + /** + * Main function to start a tracker. + */ + public static void main(String[] args) { + BasicConfigurator.configure(new ConsoleAppender( + new PatternLayout("%d [%-25t] %-5p: %m%n"))); + + CmdLineParser parser = new CmdLineParser(); + CmdLineParser.Option help = parser.addBooleanOption('h', "help"); + CmdLineParser.Option port = parser.addIntegerOption('p', "port"); + + try { + parser.parse(args); + } catch (CmdLineParser.OptionException oe) { + System.err.println(oe.getMessage()); + usage(System.err); + System.exit(1); + } + + // Display help and exit if requested + if (Boolean.TRUE.equals((Boolean)parser.getOptionValue(help))) { + usage(System.out); + System.exit(0); + } + + Integer portValue = (Integer)parser.getOptionValue(port, + Integer.valueOf(Tracker.DEFAULT_TRACKER_PORT)); + + String[] otherArgs = parser.getRemainingArgs(); + + if (otherArgs.length > 1) { + usage(System.err); + System.exit(1); + } + + // Get directory from command-line argument or default to current + // directory + String directory = otherArgs.length > 0 + ? otherArgs[0] + : "."; + + FilenameFilter filter = new FilenameFilter() { + @Override + public boolean accept(File dir, String name) { + return name.endsWith(".torrent"); + } + }; + + try { + Tracker t = new Tracker(new InetSocketAddress(portValue.intValue())); + + File parent = new File(directory); + for (File f : parent.listFiles(filter)) { + logger.info("Loading torrent from " + f.getName()); + t.announce(TrackedTorrent.load(f)); + } + + logger.info("Starting tracker with {} announced torrents...", + t.getTrackedTorrents().size()); + t.start(); + } catch (Exception e) { + logger.error("{}", e.getMessage(), e); + System.exit(2); + } + } +} diff --git a/src/main/java/com/turn/ttorrent/client/Client.java b/src/main/java/com/turn/ttorrent/client/Client.java index f16d23c71..00325f078 100644 --- a/src/main/java/com/turn/ttorrent/client/Client.java +++ b/src/main/java/com/turn/ttorrent/client/Client.java @@ -19,26 +19,19 @@ import com.turn.ttorrent.client.announce.AnnounceException; import com.turn.ttorrent.client.announce.AnnounceResponseListener; import com.turn.ttorrent.client.peer.PeerActivityListener; +import com.turn.ttorrent.client.peer.SharingPeer; import com.turn.ttorrent.common.Peer; import com.turn.ttorrent.common.Torrent; import com.turn.ttorrent.common.protocol.PeerMessage; import com.turn.ttorrent.common.protocol.TrackerMessage; -import com.turn.ttorrent.client.peer.SharingPeer; -import java.io.File; import java.io.IOException; -import java.io.PrintStream; -import java.net.Inet4Address; import java.net.InetAddress; -import java.net.NetworkInterface; -import java.net.SocketException; import java.net.UnknownHostException; import java.nio.ByteBuffer; import java.nio.channels.SocketChannel; -import java.nio.channels.UnsupportedAddressTypeException; import java.util.BitSet; import java.util.Comparator; -import java.util.Enumeration; import java.util.HashSet; import java.util.List; import java.util.Observable; @@ -51,11 +44,6 @@ import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentMap; -import jargs.gnu.CmdLineParser; - -import org.apache.log4j.BasicConfigurator; -import org.apache.log4j.ConsoleAppender; -import org.apache.log4j.PatternLayout; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -99,9 +87,6 @@ public class Client extends Observable implements Runnable, private static final int RATE_COMPUTATION_ITERATIONS = 2; private static final int MAX_DOWNLOADERS_UNCHOKE = 4; - /** Default data output directory. */ - private static final String DEFAULT_OUTPUT_DIRECTORY = "/tmp"; - public enum ClientState { WAITING, VALIDATING, @@ -985,12 +970,12 @@ private synchronized void seed() { * * @author mpetazzoni */ - private static class ClientShutdown extends TimerTask { + public static class ClientShutdown extends TimerTask { private final Client client; private final Timer timer; - ClientShutdown(Client client, Timer timer) { + public ClientShutdown(Client client, Timer timer) { this.client = client; this.timer = timer; } @@ -1003,130 +988,4 @@ public void run() { } } }; - - /** - * Display program usage on the given {@link PrintStream}. - */ - private static void usage(PrintStream s) { - s.println("usage: Client [options]- * If an interface name is given, return the first usable IPv4 address for - * that interface. If no interface name is given or if that interface - * doesn't have an IPv4 address, return's localhost address (if IPv4). - *
- * - *- * It is understood this makes the client IPv4 only, but it is important to - * remember that most BitTorrent extensions (like compact peer lists from - * trackers and UDP tracker support) are IPv4-only anyway. - *
- * - * @param iface The network interface name. - * @return A usable IPv4 address as a {@link Inet4Address}. - * @throws UnsupportedAddressTypeException If no IPv4 address was available - * to bind on. - */ - private static Inet4Address getIPv4Address(String iface) - throws SocketException, UnsupportedAddressTypeException, - UnknownHostException { - if (iface != null) { - Enumeration- * You can use the {@code main()} function of this {@link Torrent} class to - * read or create torrent files. See usage for details. - *
- * - * TODO: support multiple announce URLs. - */ - public static void main(String[] args) { - BasicConfigurator.configure(new ConsoleAppender( - new PatternLayout("%-5p: %m%n"))); - - CmdLineParser parser = new CmdLineParser(); - CmdLineParser.Option help = parser.addBooleanOption('h', "help"); - CmdLineParser.Option filename = parser.addStringOption('t', "torrent"); - CmdLineParser.Option create = parser.addBooleanOption('c', "create"); - CmdLineParser.Option announce = parser.addStringOption('a', "announce"); - - try { - parser.parse(args); - } catch (CmdLineParser.OptionException oe) { - System.err.println(oe.getMessage()); - usage(System.err); - System.exit(1); - } - - // Display help and exit if requested - if (Boolean.TRUE.equals((Boolean)parser.getOptionValue(help))) { - usage(System.out); - System.exit(0); - } - - String filenameValue = (String)parser.getOptionValue(filename); - if (filenameValue == null) { - usage(System.err, "Torrent file must be provided!"); - System.exit(1); - } - - Boolean createFlag = (Boolean)parser.getOptionValue(create); - String announceURL = (String)parser.getOptionValue(announce); - - String[] otherArgs = parser.getRemainingArgs(); - - if (Boolean.TRUE.equals(createFlag) && - (otherArgs.length != 1 || announceURL == null)) { - usage(System.err, "Announce URL and a file or directory must be " + - "provided to create a torrent file!"); - System.exit(1); - } - - OutputStream fos = null; - try { - if (Boolean.TRUE.equals(createFlag)) { - if (filenameValue != null) { - fos = new FileOutputStream(filenameValue); - } else { - fos = System.out; - } - - URI announceURI = new URI(announceURL); - File source = new File(otherArgs[0]); - if (!source.exists() || !source.canRead()) { - throw new IllegalArgumentException( - "Cannot access source file or directory " + - source.getName()); - } - - String creator = String.format("%s (ttorrent)", - System.getProperty("user.name")); - - Torrent torrent = null; - if (source.isDirectory()) { - File[] files = source.listFiles(); - Arrays.sort(files); - torrent = Torrent.create(source, Arrays.asList(files), - announceURI, creator); - } else { - torrent = Torrent.create(source, announceURI, creator); - } - - torrent.save(fos); - } else { - Torrent.load(new File(filenameValue), true); - } - } catch (Exception e) { - logger.error("{}", e.getMessage(), e); - System.exit(2); - } finally { - if (fos != null && fos != System.out) { - try { - fos.close(); - } catch (IOException ioe) { - } - } - } - } } diff --git a/src/main/java/com/turn/ttorrent/tracker/Tracker.java b/src/main/java/com/turn/ttorrent/tracker/Tracker.java index 6ba05d8b1..638fae259 100644 --- a/src/main/java/com/turn/ttorrent/tracker/Tracker.java +++ b/src/main/java/com/turn/ttorrent/tracker/Tracker.java @@ -17,30 +17,21 @@ import com.turn.ttorrent.common.Torrent; -import java.io.File; -import java.io.FilenameFilter; import java.io.IOException; -import java.io.PrintStream; import java.net.InetAddress; import java.net.InetSocketAddress; import java.net.MalformedURLException; import java.net.URL; +import java.util.Collection; import java.util.Timer; import java.util.TimerTask; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentMap; -import jargs.gnu.CmdLineParser; - -import org.apache.log4j.BasicConfigurator; -import org.apache.log4j.ConsoleAppender; -import org.apache.log4j.PatternLayout; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - import org.simpleframework.transport.connect.Connection; import org.simpleframework.transport.connect.SocketConnection; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; /** * BitTorrent tracker. @@ -180,6 +171,13 @@ public void stop() { } } + /** + * Returns the list of tracker's torrents + */ + public Collection
@@ -82,7 +76,6 @@ public class SharedTorrent extends Torrent implements PeerActivityListener {
*/
private static final float ENG_GAME_COMPLETION_RATIO = 0.95f;
- private Random random;
private boolean stop;
private long uploaded;
@@ -99,6 +92,7 @@ public class SharedTorrent extends Torrent implements PeerActivityListener {
private SortedSet
+ * This will recreate a SharedTorrent object from the provided Torrent
+ * object's encoded meta-info data.
+ * null
if no piece is interesting
+ */
+ Piece choosePiece(SortedSet> announceList,
+ public static Torrent create(File source, int pieceLength, List
> announceList,
String createdBy) throws InterruptedException, IOException {
- return Torrent.create(source, null, null, announceList, createdBy);
+ return Torrent.create(source, null, pieceLength,
+ null, announceList, createdBy);
}
/**
@@ -572,10 +578,11 @@ public static Torrent create(File source, List
> announceList,
* @param createdBy The creator's name, or any string identifying the
* torrent's creator.
*/
- public static Torrent create(File source, List
> announceList, String createdBy)
throws InterruptedException, IOException {
- return Torrent.create(source, files, null, announceList, createdBy);
+ return Torrent.create(source, files, pieceLength,
+ null, announceList, createdBy);
}
/**
@@ -597,8 +604,8 @@ public static Torrent create(File source, List
> announceList, String createdBy)
+ private static Torrent create(File parent, List
> announceList, String createdBy)
throws InterruptedException, IOException {
if (files == null || files.isEmpty()) {
logger.info("Creating single-file torrent for {}...",
@@ -630,11 +637,11 @@ private static Torrent create(File parent, List
- * Hash the given file to create the {@link Torrent} object representing - * the Torrent metainfo about this file, needed for announcing and/or - * sharing said file. - *
- * - * @param source The file to use in the torrent. - * @param announce The announce URI that will be used for this torrent. - * @param createdBy The creator's name, or any string identifying the - * torrent's creator. - * - * @deprecated use {@link Builder} instead - */ - @Deprecated - public static Torrent create(File source, URI announce, String createdBy) - throws InterruptedException, IOException { - return Torrent.create(source, null, DEFAULT_PIECE_LENGTH, - announce, null, createdBy, false); - } - - /** - * Create a {@link Torrent} object for a set of files. - * - *- * Hash the given files to create the multi-file {@link Torrent} object - * representing the Torrent meta-info about them, needed for announcing - * and/or sharing these files. Since we created the torrent, we're - * considering we'll be a full initial seeder for it. - *
- * - * @param parent The parent directory or location of the torrent files, - * also used as the torrent's name. - * @param files The files to add into this torrent. - * @param announce The announce URI that will be used for this torrent. - * @param createdBy The creator's name, or any string identifying the - * torrent's creator. - * - * @deprecated use {@link Builder} instead - */ - @Deprecated - public static Torrent create(File parent, List- * Hash the given file to create the {@link Torrent} object representing - * the Torrent metainfo about this file, needed for announcing and/or - * sharing said file. - *
- * - * @param source The file to use in the torrent. - * @param announceList The announce URIs organized as tiers that will - * be used for this torrent - * @param createdBy The creator's name, or any string identifying the - * torrent's creator. - * - * @deprecated use {@link Builder} instead - */ - @Deprecated - public static Torrent create(File source, int pieceLength, List- * Hash the given files to create the multi-file {@link Torrent} object - * representing the Torrent meta-info about them, needed for announcing - * and/or sharing these files. Since we created the torrent, we're - * considering we'll be a full initial seeder for it. - *
- * - * @param source The parent directory or location of the torrent files, - * also used as the torrent's name. - * @param files The files to add into this torrent. - * @param announceList The announce URIs organized as tiers that will - * be used for this torrent - * @param createdBy The creator's name, or any string identifying the - * torrent's creator. - * - * @deprecated use {@link Builder} instead - */ - @Deprecated - public static Torrent create(File source, List- * Hashes the given file piece by piece using the default Torrent piece - * length (see {@link #PIECE_LENGTH}) and returns the concatenation of - * these hashes, as a string. + * Hashes the given file piece by piece using the given piece length and + * returns the concatenation of these hashes, as a string. *
* *
@@ -835,7 +731,7 @@ public String call() throws UnsupportedEncodingException {
*/
private static String hashFile(File file, int pieceLenght)
throws InterruptedException, IOException {
- return Torrent.hashFiles(Arrays.asList(new File[] { file }), pieceLenght);
+ return Torrent.hashFiles(Arrays.asList(file), pieceLenght);
}
private static String hashFiles(List
* Hashes the given file piece by piece using the given piece length and
- * returns the concatenation of these hashes, as a string.
+ * returns the concatenation of these hashes, as a string.
*
> announceURIList;
-
- private String createdBy;
- private boolean privateFlag = false;
-
- private int pieceLength = DEFAULT_PIECE_LENGTH;
-
- public Builder withSharedSource(File source) {
- if (source != null && source.isDirectory()) {
- return withSharedDirectory(source);
- }
- return withSharedFile(source);
- }
-
- public Builder withSharedFile(File file) {
- checkSource(file, true);
- this.source = file;
- return this;
- }
-
- public Builder withSharedDirectory(File directory) {
- return withSharedDirectory(directory, null);
- }
-
- public Builder withSharedDirectory(File directory, List
>(1);
- }
- this.announceURIList.add(announceURIs);
- return this;
- }
-
- public Builder withPieceLength(int length) {
- this.pieceLength = length;
- return this;
- }
-
- public Builder withCreator(String creator) {
- this.createdBy = creator;
- return this;
- }
-
- public Torrent build() throws IOException, InterruptedException {
- return create(source, fileList, pieceLength, announceURI, announceURIList, createdBy, privateFlag);
- }
- }
+ /**
+ * Builder for Torrent Object
+ *
+ * @author jdeketelaere
+ */
+ public static final class Builder {
+
+ private File source;
+ private List
> announceURIList;
+
+ private String createdBy;
+ private boolean privateFlag = false;
+
+ private int pieceLength = DEFAULT_PIECE_LENGTH;
+
+ public Builder withSharedSource(File source) {
+ if (source != null && source.isDirectory()) {
+ return withSharedDirectory(source);
+ }
+ return withSharedFile(source);
+ }
+
+ public Builder withSharedFile(File file) {
+ checkSource(file, true);
+ this.source = file;
+ return this;
+ }
+
+ public Builder withSharedDirectory(File directory) {
+ return withSharedDirectory(directory, null);
+ }
+
+ public Builder withSharedDirectory(File directory, List
>(1);
+ }
+ this.announceURIList.add(announceURIs);
+ return this;
+ }
+
+ public Builder withPieceLength(int length) {
+ this.pieceLength = length;
+ return this;
+ }
+
+ public Builder withCreator(String creator) {
+ this.createdBy = creator;
+ return this;
+ }
+
+ public Torrent build() throws IOException, InterruptedException {
+ return create(source, fileList, pieceLength, announceURI, announceURIList, createdBy, privateFlag);
+ }
+ }
protected final byte[] encoded;
protected final byte[] encoded_info;
@@ -207,7 +207,7 @@ public Torrent build() throws IOException, InterruptedException {
private final String comment;
private final String createdBy;
private final String name;
- private final boolean privateFlag;
+ private final boolean privateFlag;
private final long size;
private final int pieceLength;
@@ -304,7 +304,7 @@ public Torrent(byte[] torrent, boolean seeder) throws IOException {
? this.decoded.get("created by").getString()
: null;
this.name = this.decoded_info.get("name").getString();
- this.privateFlag = this.decoded_info.containsKey("private") && this.decoded_info.get("private").getBoolean();
+ this.privateFlag = this.decoded_info.containsKey("private") && this.decoded_info.get("private").getBoolean();
this.pieceLength = this.decoded_info.get("piece length").getInt();
this.files = new LinkedList
> announceList, String createdBy, boolean privateFlag)
@@ -650,7 +650,7 @@ private static Torrent create(File parent, List