1 package net.pterodactylus.fcp.quelaton;
3 import java.io.IOException;
4 import java.util.Optional;
5 import java.util.concurrent.ExecutionException;
6 import java.util.concurrent.ExecutorService;
7 import java.util.concurrent.atomic.AtomicBoolean;
8 import java.util.concurrent.atomic.AtomicReference;
10 import net.pterodactylus.fcp.ModifyPeer;
11 import net.pterodactylus.fcp.Peer;
12 import net.pterodactylus.fcp.UnknownNodeIdentifier;
14 import com.google.common.util.concurrent.ListenableFuture;
15 import com.google.common.util.concurrent.ListeningExecutorService;
16 import com.google.common.util.concurrent.MoreExecutors;
19 * Default {@link ModifyPeerCommand} implementation based on {@link FcpDialog}.
21 * @author <a href="mailto:bombe@freenetproject.org">David ‘Bombe’ Roden</a>
23 public class ModifyPeerCommandImpl implements ModifyPeerCommand {
25 private final ListeningExecutorService threadPool;
26 private final ConnectionSupplier connectionSupplier;
27 private final AtomicReference<String> nodeIdentifier = new AtomicReference<>();
28 private final AtomicReference<Boolean> enabled = new AtomicReference<>();
29 private final AtomicReference<Boolean> allowLocalAddresses = new AtomicReference<>();
30 private final AtomicReference<Boolean> burstOnly = new AtomicReference<>();
32 public ModifyPeerCommandImpl(ExecutorService threadPool, ConnectionSupplier connectionSupplier) {
33 this.threadPool = MoreExecutors.listeningDecorator(threadPool);
34 this.connectionSupplier = connectionSupplier;
38 public ModifyPeerCommand enable() {
44 public ModifyPeerCommand disable() {
50 public ModifyPeerCommand allowLocalAddresses() {
51 allowLocalAddresses.set(true);
56 public ModifyPeerCommand disallowLocalAddresses() {
57 allowLocalAddresses.set(false);
62 public ModifyPeerCommand setBurstOnly() {
68 public Executable<Optional<Peer>> byName(String name) {
69 nodeIdentifier.set(name);
74 public Executable<Optional<Peer>> byIdentity(String nodeIdentity) {
75 nodeIdentifier.set(nodeIdentity);
80 public Executable<Optional<Peer>> byHostAndPort(String host, int port) {
81 nodeIdentifier.set(String.format("%s:%d", host, port));
85 private ListenableFuture<Optional<Peer>> execute() {
86 return threadPool.submit(this::executeSequence);
89 private Optional<Peer> executeSequence() throws IOException, ExecutionException, InterruptedException {
90 ModifyPeer modifyPeer = new ModifyPeer(new RandomIdentifierGenerator().generate(), nodeIdentifier.get());
91 Optional.ofNullable(enabled.get()).ifPresent(enabled -> modifyPeer.setEnabled(enabled));
92 Optional.ofNullable(allowLocalAddresses.get()).ifPresent(allowed -> modifyPeer.setAllowLocalAddresses(allowed));
93 Optional.ofNullable(burstOnly.get()).ifPresent(burstOnly -> modifyPeer.setBurstOnly(burstOnly));
94 try (ModifyPeerDialog modifyPeerDialog = new ModifyPeerDialog()) {
95 return modifyPeerDialog.send(modifyPeer).get();
99 private class ModifyPeerDialog extends FcpDialog<Optional<Peer>> {
101 private final AtomicBoolean finished = new AtomicBoolean();
102 private final AtomicReference<Peer> peer = new AtomicReference<>();
104 public ModifyPeerDialog() throws IOException {
105 super(threadPool, connectionSupplier.get());
109 protected boolean isFinished() {
110 return finished.get();
114 protected Optional<Peer> getResult() {
115 return Optional.ofNullable(peer.get());
119 protected void consumePeer(Peer peer) {
125 protected void consumeUnknownNodeIdentifier(UnknownNodeIdentifier unknownNodeIdentifier) {