bt.tracker.CompactPeerInfo Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of bt-core Show documentation
Show all versions of bt-core Show documentation
BitTorrent Client Library (Core)
The newest version!
/*
* Copyright (c) 2016—2021 Andrei Tomashpolskiy and individual contributors.
*
* 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 bt.tracker;
import bt.BtException;
import bt.net.InetPeer;
import bt.net.Peer;
import bt.peer.PeerOptions;
import bt.protocol.crypto.EncryptionPolicy;
import java.net.InetAddress;
import java.net.UnknownHostException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Iterator;
import java.util.List;
import java.util.NoSuchElementException;
import java.util.Objects;
import java.util.Optional;
/**
* Wrapper for binary representation of a list of peers,
* which is used by the majority of HTTP trackers and all UDP trackers.
* See BEP-23 for more details.
*
* Decoding is performed lazily when {@link Iterable#iterator()} is used.
* Results are cached, so it's only done once per instance of this class.
*
* @since 1.0
*/
public class CompactPeerInfo implements Iterable {
/**
* Address family
*
* @since 1.0
*/
public enum AddressType {
/**
* Internet Protocol v4
*
* @since 1.0
*/
IPV4(4),
/**
* Internet Protocol v6
*
* @since 1.0
*/
IPV6(16);
private int length;
AddressType(int length) {
this.length = length;
}
/**
* @return Address length in bytes
* @since 1.0
*/
public int length() {
return length;
}
}
private static final int PORT_LENGTH = 2;
private final int addressLength;
private final byte[] peers;
private final Optional cryptoFlags;
private final List peerList;
/**
* Create a list of peers from its' binary representation,
* using the specified address type for decoding individual addresses.
*
* @since 1.0
*/
public CompactPeerInfo(byte[] peers, AddressType addressType) {
this(peers, addressType, null);
}
/**
* Create a list of peers from its' binary representation,
* using the specified address type for decoding individual addresses.
*
* @param cryptoFlags Byte array, where each byte indicates
* whether the corresponding peer from {@code peers} supports MSE:
* 1 if peer supports encryption, 0 otherwise.
* @since 1.2
*/
public CompactPeerInfo(byte[] peers, AddressType addressType, byte[] cryptoFlags) {
Objects.requireNonNull(peers);
Objects.requireNonNull(addressType);
int peerLength = addressType.length() + PORT_LENGTH;
if (peers.length % peerLength != 0) {
throw new IllegalArgumentException("Invalid peers string (" + addressType.name() + ") -- length (" +
peers.length + ") is not divisible by " + peerLength);
}
int numOfPeers = peers.length / peerLength;
if (cryptoFlags != null && cryptoFlags.length != numOfPeers) {
throw new IllegalArgumentException("Number of peers (" + numOfPeers +
") is different from the number of crypto flags (" + cryptoFlags.length + ")");
}
this.addressLength = addressType.length();
this.peers = peers;
this.cryptoFlags = Optional.ofNullable(cryptoFlags);
this.peerList = new ArrayList<>();
}
@Override
public Iterator iterator() {
if (!peerList.isEmpty()) {
return peerList.iterator();
}
return new Iterator() {
private int pos, index;
@Override
public boolean hasNext() {
return pos < peers.length;
}
@Override
public Peer next() {
if (!hasNext()) {
throw new NoSuchElementException("No more peers left");
}
int from, to;
InetAddress inetAddress;
int port;
from = pos;
to = pos = pos + addressLength;
try {
inetAddress = InetAddress.getByAddress(Arrays.copyOfRange(peers, from, to));
} catch (UnknownHostException e) {
throw new BtException("Failed to get next peer", e);
}
from = to;
to = pos = pos + PORT_LENGTH;
port = (((peers[from] << 8) & 0xFF00) + (peers[to - 1] & 0x00FF));
PeerOptions options = PeerOptions.defaultOptions();
boolean requiresEncryption = cryptoFlags.isPresent() && cryptoFlags.get()[index] == 1;
if (requiresEncryption) {
options = options.withEncryptionPolicy(EncryptionPolicy.PREFER_ENCRYPTED);
}
Peer peer = InetPeer.builder(inetAddress, port).options(options).build();
peerList.add(peer);
index++;
return peer;
}
};
}
}