[Author Prev][Author Next][Thread Prev][Thread Next][Author Index][Thread Index]
[or-cvs] [puppetor/gsoc2008] Change the package name to org.torproject.puppetor.
The old name was related to the University of Bamberg where Karsten used to
work.
---
doc/DistributedPuppeTorReadme | 4 +-
doc/howto.tex | 2 +-
doc/howtosecurermi.txt | 4 +-
.../wiai/lspi/puppetor/ClientApplication.java | 100 --
.../uniba/wiai/lspi/puppetor/ClientEventType.java | 64 --
src/de/uniba/wiai/lspi/puppetor/DirectoryNode.java | 49 -
src/de/uniba/wiai/lspi/puppetor/Event.java | 54 -
src/de/uniba/wiai/lspi/puppetor/EventListener.java | 25 -
src/de/uniba/wiai/lspi/puppetor/EventManager.java | 231 -----
src/de/uniba/wiai/lspi/puppetor/EventType.java | 26 -
src/de/uniba/wiai/lspi/puppetor/HiddenService.java | 61 --
.../wiai/lspi/puppetor/HiddenServiceEventType.java | 215 ----
src/de/uniba/wiai/lspi/puppetor/Network.java | 657 ------------
.../uniba/wiai/lspi/puppetor/NetworkFactory.java | 82 --
src/de/uniba/wiai/lspi/puppetor/NodeEventType.java | 60 --
src/de/uniba/wiai/lspi/puppetor/NodeState.java | 44 -
src/de/uniba/wiai/lspi/puppetor/ProxyNode.java | 269 -----
.../wiai/lspi/puppetor/PuppeTorException.java | 65 --
src/de/uniba/wiai/lspi/puppetor/RouterNode.java | 55 -
.../wiai/lspi/puppetor/ServerApplication.java | 58 --
.../uniba/wiai/lspi/puppetor/ServerEventType.java | 41 -
.../examples/AccessingPublicWebServerOverTor.java | 119 ---
...ccessingHiddenServiceOverPrivateTorNetwork.java | 132 ---
...AccessingHiddenServiceOverPublicTorNetwork.java | 149 ---
...AdvertisingHiddenServiceToPublicTorNetwork.java | 110 --
.../lspi/puppetor/impl/ClientApplicationImpl.java | 457 ---------
.../wiai/lspi/puppetor/impl/DirectoryNodeImpl.java | 431 --------
.../uniba/wiai/lspi/puppetor/impl/EventImpl.java | 122 ---
.../wiai/lspi/puppetor/impl/EventManagerImpl.java | 729 -------------
.../wiai/lspi/puppetor/impl/HiddenServiceImpl.java | 166 ---
.../uniba/wiai/lspi/puppetor/impl/NetworkImpl.java | 1082 -------------------
.../wiai/lspi/puppetor/impl/ProxyNodeImpl.java | 734 -------------
.../wiai/lspi/puppetor/impl/RouterNodeImpl.java | 402 --------
.../lspi/puppetor/impl/ServerApplicationImpl.java | 351 -------
.../lspi/puppetor/rmi/AbstractMasterFactory.java | 78 --
.../lspi/puppetor/rmi/AbstractSlaveFactory.java | 88 --
.../uniba/wiai/lspi/puppetor/rmi/LocalMaster.java | 36 -
src/de/uniba/wiai/lspi/puppetor/rmi/Master.java | 15 -
.../wiai/lspi/puppetor/rmi/MasterConnector.java | 31 -
src/de/uniba/wiai/lspi/puppetor/rmi/Network.java | 79 --
.../wiai/lspi/puppetor/rmi/NetworkDescription.java | 16 -
.../uniba/wiai/lspi/puppetor/rmi/RemoteMaster.java | 47 -
src/de/uniba/wiai/lspi/puppetor/rmi/Slave.java | 51 -
src/de/uniba/wiai/lspi/puppetor/rmi/Task.java | 38 -
.../rmi/TaskExecutionNotSuccessfulException.java | 23 -
.../uniba/wiai/lspi/puppetor/rmi/TaskResult.java | 39 -
src/de/uniba/wiai/lspi/puppetor/rmi/Test.java | 37 -
.../uniba/wiai/lspi/puppetor/rmi/TestExecutor.java | 114 --
.../uniba/wiai/lspi/puppetor/rmi/TestResult.java | 15 -
.../uniba/wiai/lspi/puppetor/rmi/TorInstance.java | 57 -
.../rmi/execute/PuppeTorMasterProgram.java | 193 ----
.../puppetor/rmi/execute/PuppeTorSlaveProgram.java | 215 ----
.../lspi/puppetor/rmi/impl/AbstractTaskImpl.java | 61 --
.../lspi/puppetor/rmi/impl/AbstractTestImpl.java | 52 -
.../wiai/lspi/puppetor/rmi/impl/MasterImpl.java | 104 --
.../lspi/puppetor/rmi/impl/MasterImplFactory.java | 55 -
.../puppetor/rmi/impl/NetworkDescriptionImpl.java | 43 -
.../wiai/lspi/puppetor/rmi/impl/NetworkImpl.java | 125 ---
.../wiai/lspi/puppetor/rmi/impl/SlaveImpl.java | 131 ---
.../lspi/puppetor/rmi/impl/SlaveImplFactory.java | 31 -
.../lspi/puppetor/rmi/impl/TaskResultImpl.java | 51 -
.../lspi/puppetor/rmi/impl/TestExecutorImpl.java | 554 ----------
.../lspi/puppetor/rmi/impl/TorInstanceImpl.java | 174 ----
.../lspi/puppetor/rmi/tasks/AccessGoogleTask.java | 73 --
.../puppetor/rmi/tasks/AddHiddenServiceTask.java | 105 --
.../lspi/puppetor/rmi/tasks/BuildCircuitsTask.java | 40 -
.../puppetor/rmi/tasks/ConfigureAsPrivateTask.java | 49 -
.../rmi/tasks/CreateDirectoryAuthorityTask.java | 47 -
.../lspi/puppetor/rmi/tasks/CreateNetworkTask.java | 30 -
.../lspi/puppetor/rmi/tasks/CreateProxyTask.java | 47 -
.../lspi/puppetor/rmi/tasks/CreateRouterTask.java | 47 -
.../lspi/puppetor/rmi/tasks/ShutdownNodesTask.java | 37 -
.../lspi/puppetor/rmi/tasks/StartNodesTask.java | 41 -
.../lspi/puppetor/rmi/tasks/TerminateTask.java | 29 -
.../tests/AccessGoogleOverPublicTorNetwork.java | 39 -
...singHiddenServiceOverPrivateTorNetworkTest.java | 40 -
.../lspi/puppetor/rmi/tests/TestRegistration.java | 33 -
src/org/torproject/puppetor/ClientApplication.java | 100 ++
src/org/torproject/puppetor/ClientEventType.java | 64 ++
src/org/torproject/puppetor/DirectoryNode.java | 49 +
src/org/torproject/puppetor/Event.java | 54 +
src/org/torproject/puppetor/EventListener.java | 25 +
src/org/torproject/puppetor/EventManager.java | 231 +++++
src/org/torproject/puppetor/EventType.java | 26 +
src/org/torproject/puppetor/HiddenService.java | 61 ++
.../puppetor/HiddenServiceEventType.java | 215 ++++
src/org/torproject/puppetor/Network.java | 657 ++++++++++++
src/org/torproject/puppetor/NetworkFactory.java | 83 ++
src/org/torproject/puppetor/NodeEventType.java | 60 ++
src/org/torproject/puppetor/NodeState.java | 44 +
src/org/torproject/puppetor/ProxyNode.java | 269 +++++
src/org/torproject/puppetor/PuppeTorException.java | 65 ++
src/org/torproject/puppetor/RouterNode.java | 55 +
src/org/torproject/puppetor/ServerApplication.java | 58 ++
src/org/torproject/puppetor/ServerEventType.java | 41 +
.../examples/AccessingPublicWebServerOverTor.java | 119 +++
...ccessingHiddenServiceOverPrivateTorNetwork.java | 132 +++
...AccessingHiddenServiceOverPublicTorNetwork.java | 149 +++
...AdvertisingHiddenServiceToPublicTorNetwork.java | 110 ++
.../puppetor/impl/ClientApplicationImpl.java | 458 +++++++++
.../puppetor/impl/DirectoryNodeImpl.java | 432 ++++++++
src/org/torproject/puppetor/impl/EventImpl.java | 123 +++
.../torproject/puppetor/impl/EventManagerImpl.java | 730 +++++++++++++
.../puppetor/impl/HiddenServiceImpl.java | 167 +++
src/org/torproject/puppetor/impl/NetworkImpl.java | 1083 ++++++++++++++++++++
.../torproject/puppetor/impl/ProxyNodeImpl.java | 735 +++++++++++++
.../torproject/puppetor/impl/RouterNodeImpl.java | 403 ++++++++
.../puppetor/impl/ServerApplicationImpl.java | 352 +++++++
.../puppetor/rmi/AbstractMasterFactory.java | 79 ++
.../puppetor/rmi/AbstractSlaveFactory.java | 89 ++
src/org/torproject/puppetor/rmi/LocalMaster.java | 36 +
src/org/torproject/puppetor/rmi/Master.java | 15 +
.../torproject/puppetor/rmi/MasterConnector.java | 31 +
src/org/torproject/puppetor/rmi/Network.java | 79 ++
.../puppetor/rmi/NetworkDescription.java | 16 +
src/org/torproject/puppetor/rmi/RemoteMaster.java | 47 +
src/org/torproject/puppetor/rmi/Slave.java | 51 +
src/org/torproject/puppetor/rmi/Task.java | 38 +
.../rmi/TaskExecutionNotSuccessfulException.java | 23 +
src/org/torproject/puppetor/rmi/TaskResult.java | 39 +
src/org/torproject/puppetor/rmi/Test.java | 37 +
src/org/torproject/puppetor/rmi/TestExecutor.java | 114 ++
src/org/torproject/puppetor/rmi/TestResult.java | 15 +
src/org/torproject/puppetor/rmi/TorInstance.java | 57 +
.../rmi/execute/PuppeTorMasterProgram.java | 194 ++++
.../puppetor/rmi/execute/PuppeTorSlaveProgram.java | 216 ++++
.../puppetor/rmi/impl/AbstractTaskImpl.java | 62 ++
.../puppetor/rmi/impl/AbstractTestImpl.java | 52 +
.../torproject/puppetor/rmi/impl/MasterImpl.java | 105 ++
.../puppetor/rmi/impl/MasterImplFactory.java | 56 +
.../puppetor/rmi/impl/NetworkDescriptionImpl.java | 44 +
.../torproject/puppetor/rmi/impl/NetworkImpl.java | 126 +++
.../torproject/puppetor/rmi/impl/SlaveImpl.java | 132 +++
.../puppetor/rmi/impl/SlaveImplFactory.java | 32 +
.../puppetor/rmi/impl/TaskResultImpl.java | 51 +
.../puppetor/rmi/impl/TestExecutorImpl.java | 555 ++++++++++
.../puppetor/rmi/impl/TorInstanceImpl.java | 175 ++++
.../puppetor/rmi/tasks/AccessGoogleTask.java | 74 ++
.../puppetor/rmi/tasks/AddHiddenServiceTask.java | 106 ++
.../puppetor/rmi/tasks/BuildCircuitsTask.java | 41 +
.../puppetor/rmi/tasks/ConfigureAsPrivateTask.java | 50 +
.../rmi/tasks/CreateDirectoryAuthorityTask.java | 48 +
.../puppetor/rmi/tasks/CreateNetworkTask.java | 31 +
.../puppetor/rmi/tasks/CreateProxyTask.java | 48 +
.../puppetor/rmi/tasks/CreateRouterTask.java | 48 +
.../puppetor/rmi/tasks/ShutdownNodesTask.java | 38 +
.../puppetor/rmi/tasks/StartNodesTask.java | 42 +
.../puppetor/rmi/tasks/TerminateTask.java | 30 +
.../tests/AccessGoogleOverPublicTorNetwork.java | 40 +
...singHiddenServiceOverPrivateTorNetworkTest.java | 41 +
.../puppetor/rmi/tests/TestRegistration.java | 33 +
151 files changed, 10391 insertions(+), 10355 deletions(-)
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/ClientApplication.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/ClientEventType.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/DirectoryNode.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/Event.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/EventListener.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/EventManager.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/EventType.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/HiddenService.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/HiddenServiceEventType.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/Network.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/NetworkFactory.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/NodeEventType.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/NodeState.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/ProxyNode.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/PuppeTorException.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/RouterNode.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/ServerApplication.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/ServerEventType.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/examples/AccessingPublicWebServerOverTor.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/examples/AdvertisingAndAccessingHiddenServiceOverPrivateTorNetwork.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/examples/AdvertisingAndAccessingHiddenServiceOverPublicTorNetwork.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/examples/AdvertisingHiddenServiceToPublicTorNetwork.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/impl/ClientApplicationImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/impl/DirectoryNodeImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/impl/EventImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/impl/EventManagerImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/impl/HiddenServiceImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/impl/NetworkImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/impl/ProxyNodeImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/impl/RouterNodeImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/impl/ServerApplicationImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/AbstractMasterFactory.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/AbstractSlaveFactory.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/LocalMaster.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/Master.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/MasterConnector.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/Network.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/NetworkDescription.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/RemoteMaster.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/Slave.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/Task.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/TaskExecutionNotSuccessfulException.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/TaskResult.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/Test.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/TestExecutor.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/TestResult.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/TorInstance.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/execute/PuppeTorMasterProgram.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/execute/PuppeTorSlaveProgram.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/impl/AbstractTaskImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/impl/AbstractTestImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/impl/MasterImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/impl/MasterImplFactory.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/impl/NetworkDescriptionImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/impl/NetworkImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/impl/SlaveImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/impl/SlaveImplFactory.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/impl/TaskResultImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/impl/TestExecutorImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/impl/TorInstanceImpl.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/tasks/AccessGoogleTask.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/tasks/AddHiddenServiceTask.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/tasks/BuildCircuitsTask.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/tasks/ConfigureAsPrivateTask.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/tasks/CreateDirectoryAuthorityTask.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/tasks/CreateNetworkTask.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/tasks/CreateProxyTask.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/tasks/CreateRouterTask.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/tasks/ShutdownNodesTask.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/tasks/StartNodesTask.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/tasks/TerminateTask.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/tests/AccessGoogleOverPublicTorNetwork.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/tests/StartingAndAccessingHiddenServiceOverPrivateTorNetworkTest.java
delete mode 100644 src/de/uniba/wiai/lspi/puppetor/rmi/tests/TestRegistration.java
create mode 100644 src/org/torproject/puppetor/ClientApplication.java
create mode 100644 src/org/torproject/puppetor/ClientEventType.java
create mode 100644 src/org/torproject/puppetor/DirectoryNode.java
create mode 100644 src/org/torproject/puppetor/Event.java
create mode 100644 src/org/torproject/puppetor/EventListener.java
create mode 100644 src/org/torproject/puppetor/EventManager.java
create mode 100644 src/org/torproject/puppetor/EventType.java
create mode 100644 src/org/torproject/puppetor/HiddenService.java
create mode 100644 src/org/torproject/puppetor/HiddenServiceEventType.java
create mode 100644 src/org/torproject/puppetor/Network.java
create mode 100644 src/org/torproject/puppetor/NetworkFactory.java
create mode 100644 src/org/torproject/puppetor/NodeEventType.java
create mode 100644 src/org/torproject/puppetor/NodeState.java
create mode 100644 src/org/torproject/puppetor/ProxyNode.java
create mode 100644 src/org/torproject/puppetor/PuppeTorException.java
create mode 100644 src/org/torproject/puppetor/RouterNode.java
create mode 100644 src/org/torproject/puppetor/ServerApplication.java
create mode 100644 src/org/torproject/puppetor/ServerEventType.java
create mode 100644 src/org/torproject/puppetor/examples/AccessingPublicWebServerOverTor.java
create mode 100644 src/org/torproject/puppetor/examples/AdvertisingAndAccessingHiddenServiceOverPrivateTorNetwork.java
create mode 100644 src/org/torproject/puppetor/examples/AdvertisingAndAccessingHiddenServiceOverPublicTorNetwork.java
create mode 100644 src/org/torproject/puppetor/examples/AdvertisingHiddenServiceToPublicTorNetwork.java
create mode 100644 src/org/torproject/puppetor/impl/ClientApplicationImpl.java
create mode 100644 src/org/torproject/puppetor/impl/DirectoryNodeImpl.java
create mode 100644 src/org/torproject/puppetor/impl/EventImpl.java
create mode 100644 src/org/torproject/puppetor/impl/EventManagerImpl.java
create mode 100644 src/org/torproject/puppetor/impl/HiddenServiceImpl.java
create mode 100644 src/org/torproject/puppetor/impl/NetworkImpl.java
create mode 100644 src/org/torproject/puppetor/impl/ProxyNodeImpl.java
create mode 100644 src/org/torproject/puppetor/impl/RouterNodeImpl.java
create mode 100644 src/org/torproject/puppetor/impl/ServerApplicationImpl.java
create mode 100644 src/org/torproject/puppetor/rmi/AbstractMasterFactory.java
create mode 100644 src/org/torproject/puppetor/rmi/AbstractSlaveFactory.java
create mode 100644 src/org/torproject/puppetor/rmi/LocalMaster.java
create mode 100644 src/org/torproject/puppetor/rmi/Master.java
create mode 100644 src/org/torproject/puppetor/rmi/MasterConnector.java
create mode 100644 src/org/torproject/puppetor/rmi/Network.java
create mode 100644 src/org/torproject/puppetor/rmi/NetworkDescription.java
create mode 100644 src/org/torproject/puppetor/rmi/RemoteMaster.java
create mode 100644 src/org/torproject/puppetor/rmi/Slave.java
create mode 100644 src/org/torproject/puppetor/rmi/Task.java
create mode 100644 src/org/torproject/puppetor/rmi/TaskExecutionNotSuccessfulException.java
create mode 100644 src/org/torproject/puppetor/rmi/TaskResult.java
create mode 100644 src/org/torproject/puppetor/rmi/Test.java
create mode 100644 src/org/torproject/puppetor/rmi/TestExecutor.java
create mode 100644 src/org/torproject/puppetor/rmi/TestResult.java
create mode 100644 src/org/torproject/puppetor/rmi/TorInstance.java
create mode 100644 src/org/torproject/puppetor/rmi/execute/PuppeTorMasterProgram.java
create mode 100644 src/org/torproject/puppetor/rmi/execute/PuppeTorSlaveProgram.java
create mode 100644 src/org/torproject/puppetor/rmi/impl/AbstractTaskImpl.java
create mode 100644 src/org/torproject/puppetor/rmi/impl/AbstractTestImpl.java
create mode 100644 src/org/torproject/puppetor/rmi/impl/MasterImpl.java
create mode 100644 src/org/torproject/puppetor/rmi/impl/MasterImplFactory.java
create mode 100644 src/org/torproject/puppetor/rmi/impl/NetworkDescriptionImpl.java
create mode 100644 src/org/torproject/puppetor/rmi/impl/NetworkImpl.java
create mode 100644 src/org/torproject/puppetor/rmi/impl/SlaveImpl.java
create mode 100644 src/org/torproject/puppetor/rmi/impl/SlaveImplFactory.java
create mode 100644 src/org/torproject/puppetor/rmi/impl/TaskResultImpl.java
create mode 100644 src/org/torproject/puppetor/rmi/impl/TestExecutorImpl.java
create mode 100644 src/org/torproject/puppetor/rmi/impl/TorInstanceImpl.java
create mode 100644 src/org/torproject/puppetor/rmi/tasks/AccessGoogleTask.java
create mode 100644 src/org/torproject/puppetor/rmi/tasks/AddHiddenServiceTask.java
create mode 100644 src/org/torproject/puppetor/rmi/tasks/BuildCircuitsTask.java
create mode 100644 src/org/torproject/puppetor/rmi/tasks/ConfigureAsPrivateTask.java
create mode 100644 src/org/torproject/puppetor/rmi/tasks/CreateDirectoryAuthorityTask.java
create mode 100644 src/org/torproject/puppetor/rmi/tasks/CreateNetworkTask.java
create mode 100644 src/org/torproject/puppetor/rmi/tasks/CreateProxyTask.java
create mode 100644 src/org/torproject/puppetor/rmi/tasks/CreateRouterTask.java
create mode 100644 src/org/torproject/puppetor/rmi/tasks/ShutdownNodesTask.java
create mode 100644 src/org/torproject/puppetor/rmi/tasks/StartNodesTask.java
create mode 100644 src/org/torproject/puppetor/rmi/tasks/TerminateTask.java
create mode 100644 src/org/torproject/puppetor/rmi/tests/AccessGoogleOverPublicTorNetwork.java
create mode 100644 src/org/torproject/puppetor/rmi/tests/StartingAndAccessingHiddenServiceOverPrivateTorNetworkTest.java
create mode 100644 src/org/torproject/puppetor/rmi/tests/TestRegistration.java
diff --git a/doc/DistributedPuppeTorReadme b/doc/DistributedPuppeTorReadme
index 62588da..98c2f2a 100644
--- a/doc/DistributedPuppeTorReadme
+++ b/doc/DistributedPuppeTorReadme
@@ -48,7 +48,7 @@ computer that runs the master can also run a slave without a problem).
To start testing, go ahead and start all the Slaves. Go to the puppetor
directory, and simply execute
- java -cp bin/ de.uniba.wiai.lspi.puppetor.rmi.execute.PuppeTorSlaveProgram
+ java -cp bin/ org.torproject.puppetor.rmi.execute.PuppeTorSlaveProgram
This will run the PuppeTorSlaveProgram in a loop that tries connecting to the
specified master once a minute. After it has made a connection, it will go back
to this mode of continuously trying to reach the master, so you don't have to
@@ -58,7 +58,7 @@ large. Make sure you clean it occasionally.
After all the slaves are started, fire up the Master by going into the puppetor
directory, and executing
- java -cp bin/ de.uniba.wiai.lspi.puppetor.rmi.execute.PuppeTorMasterProgram
+ java -cp bin/ org.torproject.puppetor.rmi.execute.PuppeTorMasterProgram
You will see some output at the master to indicate connected slaves and the
success of the execution of tests. Note that there are no good logging
diff --git a/doc/howto.tex b/doc/howto.tex
index dbe4d4d..8ddd951 100644
--- a/doc/howto.tex
+++ b/doc/howto.tex
@@ -109,7 +109,7 @@ base directory of this framework:
\begin{verbatim}
java -cp bin:lib/torctl.jar
- de.uniba.wiai.lspi.puppetor.examples.
+ org.torproject.puppetor.examples.
AccessingPublicWebServerOverTor
\end{verbatim}
diff --git a/doc/howtosecurermi.txt b/doc/howtosecurermi.txt
index ccb8164..dffb918 100644
--- a/doc/howtosecurermi.txt
+++ b/doc/howtosecurermi.txt
@@ -30,12 +30,12 @@ Setting up the application:
everything into bin/. Then you can start the master like this:
java -cp bin/:lib/torctl.jar \
- de.uniba.wiai.lspi.puppetor.rmi.execute.PuppeTorMasterProgram
+ org.torproject.puppetor.rmi.execute.PuppeTorMasterProgram
and the slave like this:
java -cp bin/:lib/torctl.jar \
- de.uniba.wiai.lspi.puppetor.rmi.execute.PuppeTorSlaveProgram
+ org.torproject.puppetor.rmi.execute.PuppeTorSlaveProgram
Optionally, add the option "-Djavax.net.debug=all" to receive debug output in
the console window.
diff --git a/src/de/uniba/wiai/lspi/puppetor/ClientApplication.java b/src/de/uniba/wiai/lspi/puppetor/ClientApplication.java
deleted file mode 100644
index c6de5b6..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/ClientApplication.java
+++ /dev/null
@@ -1,100 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-/**
- * The <code>ClientApplication</code> can be used to simulate simple
- * <code>HTTP GET</code> requests by a virtual local client. Therefore, an
- * address and a port are given to which the client shall connect. Requests are
- * performed by a background thread, so that multiple requests could be
- * performed at the same time.
- *
- * @author kloesing
- */
-public interface ClientApplication {
-
- /**
- * <p>
- * Performs one or more HTTP requests to a previously provided address and
- * port. All requests are performed by a thread in the background, so that
- * this method returns immediately. That thread will try for
- * <code>retries</code> times to make the request with a timeout of
- * <code>timeoutForEachRetry</code> milliseconds each. If an attempt is
- * not successful, the thread nevertheless waits for the timeout to expire
- * before performing the next attempt. If <code>stopOnSuccess</code> is
- * set to <code>true</code>, the thread will quit performing requests
- * immediately after the first successful request.
- * </p>
- *
- * <p>
- * For each sent request the application fires a
- * <event>ClientEventType.CLIENT_SENDING_REQUEST</code> event. On receiving
- * a reply it fires an event of type <code>ClientEventType.CLIENT_REPLY_RECEIVED</code>,
- * if a request is not successful or times out, a <code>ClientEventType.CLIENT_GAVE_UP_REQUEST</code>
- * event is fired. After all requests have been performed (either
- * successfully, or not) an event of type <code>ClientEventType.CLIENT_REQUESTS_PERFORMED</code>
- * is fired.
- * </p>
- *
- * TODO may this method only be invoked once?!
- *
- * @param retries
- * The number of retries that this client will perform. Must be
- * <code>1</code> or greater.
- * @param timeoutForEachRetry
- * The timeout for each request. If a request is not successful,
- * the thread nevertheless waits for the timeout to expire. Must
- * not be negative.
- * @param stopOnSuccess
- * If set to <code>true</code>, the client quits performing
- * requests after the first successful request, if <code>false</code>
- * it continues until all <code>retries</code> have been
- * processed.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given for either of the
- * parameters.
- */
- public abstract void startRequests(int retries, long timeoutForEachRetry,
- boolean stopOnSuccess);
-
- /**
- * Stops all requests that are currently running.
- *
- * @throws IllegalStateException
- * Thrown if no requests have been started before.
- */
- public abstract void stopRequest();
-
- /**
- * Returns the name of this client.
- *
- * @return The name of this client.
- */
- public abstract String getClientApplicationName();
-
- /**
- * Returns the SOCKS port of the local Tor node to which requests are sent.
- *
- * @return The SOCKS port of the local Tor node to which requests are sent.
- */
- public abstract int getSocksPort();
-
- /**
- * Returns the target name for the requests sent by this client; can be
- * either a server name/address or an onion address.
- *
- * @return The target name for the requests sent by this client.
- */
- public abstract String getTargetName();
-
- /**
- * Returns the target port for the requests sent by this client; can be
- * either a server port or a virtual port of a hidden service.
- *
- * @return The target port for the requests sent by this client.
- */
- public abstract int getTargetPort();
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/ClientEventType.java b/src/de/uniba/wiai/lspi/puppetor/ClientEventType.java
deleted file mode 100644
index d6df5f3..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/ClientEventType.java
+++ /dev/null
@@ -1,64 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-/**
- * Event types that can be fired by a client application running as thread in
- * the background.
- */
-@SuppressWarnings("serial")
-public class ClientEventType implements EventType {
-
- /**
- * String identifying the type of the event type.
- */
- String typeString;
-
- /**
- * Creates a new event type with the given type string.
- *
- * @param typeString
- * String identifying the type of the event type.
- */
- public ClientEventType(final String typeString) {
- this.typeString = typeString;
- }
-
- public String getTypeName() {
- return typeString;
- }
-
- /**
- * The client application is sending a request; this event is fired
- * internally and not parsed from a log statement from Tor.
- */
- public static final ClientEventType CLIENT_SENDING_REQUEST =
- new ClientEventType("CLIENT_SENDING_REQUEST");
-
- /**
- * The client application has received a reply to a previously sent request;
- * this event is fired internally and not parsed from a log statement from
- * Tor.
- */
- public static final ClientEventType CLIENT_REPLY_RECEIVED =
- new ClientEventType("CLIENT_REPLY_RECEIVED");
-
- /**
- * The client application has given up waiting for the reply to a previously
- * sent request; this event is fired internally and not parsed from a log
- * statement from Tor.
- */
- public static final ClientEventType CLIENT_GAVE_UP_REQUEST =
- new ClientEventType("CLIENT_GAVE_UP_REQUEST");
-
- /**
- * The client application has completed a series of requests, whether they
- * were successful or not; this event is fired internally and not parsed
- * from a log statement from Tor.
- */
- public static final ClientEventType CLIENT_REQUESTS_PERFORMED =
- new ClientEventType("CLIENT_REQUESTS_PERFORMED");
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/DirectoryNode.java b/src/de/uniba/wiai/lspi/puppetor/DirectoryNode.java
deleted file mode 100644
index 37f363a..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/DirectoryNode.java
+++ /dev/null
@@ -1,49 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-import java.util.Set;
-
-/**
- * A <code>DirectoryNode</code> represents a Tor process that acts as
- * <code>RouterNode</code> and which is further a directory authoritative
- * server for the (private) Tor network. It inherits most of the configuration
- * and behavior from <code>RouterNode</code> and adds some directory-specific
- * configurations and behavior.
- *
- * @author kloesing
- */
-public interface DirectoryNode extends RouterNode {
-
- /**
- * Combines the fingerprint of this node to a <code>DirServer</code>
- * string that can be used to configure this or other nodes to use this node
- * as directory server.
- *
- * @return <code>DirServer</code> string to configure a node to use this
- * node as directory server.
- * @throws PuppeTorException
- * Thrown if a problem occurs when determining the fingerprint
- * of this node.
- */
- public abstract String getDirServerString() throws PuppeTorException;
-
- /**
- * Adds the given (possibly empty) set of onion router fingerprints to the
- * set of approved routers to confirm to directory clients, that the given
- * routers can be trusted. Changes are only stored locally and not written
- * to the <code>approved-routers</code> file to disk which will be done
- * when writing the configuration of this node.
- *
- * @param approvedRouters
- * The set of approved routers to be added. Each provided string
- * must be formatted as
- * <code>nickname 0000 0000 0000 0000 0000 0000 0000 0000 0000 0000</code>.
- * @throws IllegalArgumentException
- * Thrown if <code>null</code> is passed as parameter.
- */
- public void addApprovedRouters(Set<String> approvedRouters);
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/Event.java b/src/de/uniba/wiai/lspi/puppetor/Event.java
deleted file mode 100644
index 3cfa8c8..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/Event.java
+++ /dev/null
@@ -1,54 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-import java.io.Serializable;
-
-/**
- * An <code>Event</code> is created for every state change of an asynchronous
- * system component, e.g. a Tor process or a client/server application running
- * as thread in the background. In contrast to <code>NodeState</code> an
- * <code>Event</code> cannot be a pre- or postconditions for a method
- * invocation. There is no prescribed order in which events are fired by a
- * certain process or application. Some events can be fired only once, others
- * possibly multiple times. All management operations for events are contained
- * in the <code>EventManager</code>.
- *
- * @author kloesing
- */
-public interface Event extends Serializable {
-
- /**
- * Returns the name of the source of this event, which can be a Tor process
- * or a client/server application running as thread in the background.
- *
- * @return The event source.
- */
- public abstract String getSource();
-
- /**
- * Returns the event type.
- *
- * @return The event type.
- */
- public abstract EventType getType();
-
- /**
- * Returns the original message that lead to firing this event.
- *
- * @return The original message.
- */
- public abstract String getMessage();
-
- /**
- * Returns the occurrence time of this event, which is either the parsed
- * time from a Tor log statement, or the current system time when a
- * client/server application fired this event.
- *
- * @return The occurrence time.
- */
- public abstract long getOccurrenceTime();
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/EventListener.java b/src/de/uniba/wiai/lspi/puppetor/EventListener.java
deleted file mode 100644
index 3473d20..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/EventListener.java
+++ /dev/null
@@ -1,25 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-/**
- * This interface must be implemented by any object in a test application that
- * shall be registered as event listener.
- *
- * @author kloesing
- */
-public interface EventListener {
-
- /**
- * Is invoked when an asynchronous event is fired by the source (or one of
- * the sources) for which this listener was registered.
- *
- * @param event
- * The event that was fired.
- */
- public void handleEvent(Event event);
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/EventManager.java b/src/de/uniba/wiai/lspi/puppetor/EventManager.java
deleted file mode 100644
index 761d1a9..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/EventManager.java
+++ /dev/null
@@ -1,231 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-import java.util.List;
-
-/**
- * The <code>EventManager</code> is the central place for a test run to manage
- * asynchronous events by Tor processes and client or server applications
- * running as threads in the background. A test application can either register
- * event listeners to be notified asynchronously about events when they occur,
- * or synchronize with an event by being blocked until a certain event occurs.
- *
- * @author kloesing
- */
-public interface EventManager {
-
- /**
- * Registers the given <code>listener</code> as event listener for events
- * originating from the given <code>source</code>. This method returns a
- * list of all previously fired events by this source, so that each event
- * fired by this source is either included in the returned list or
- * signalized in a later invocation on the event listener, but not in both.
- * This prevents race conditions by eliminating the gap between registration
- * of an event handler and asking if an event has been fired before
- * registering.
- *
- * @param source
- * The name of the source of events that the listener is
- * interested in. May not be <code>null</code> and must be the
- * name of a previously created node, client, or server.
- * @param listener
- * The listener that wants to be notified about events from the
- * given <code>source</code>. If the <code>listener</code>
- * is already registered for the same <code>source</code>,
- * nothing happens, i.e. the <code>listener</code> will not
- * receive multiple invocations for the same event. May not be
- * <code>null</code>.
- * @return A list of all previously fired events for the given
- * <code>source</code>. If no event has been fired before, an
- * empty list is returned instead of <code>null</code>.
- * @throws IllegalArgumentException
- * Thrown if <code>null</code> is passed for either of the
- * parameters or if the <code>source</code> is unknown.
- */
- public abstract List<Event> addEventListener(String source,
- EventListener listener);
-
- /**
- * Registers the given <code>listener</code> as event listener for future
- * events originating from any source.
- *
- * @param listener
- * The listener that wants to be notified about events from the
- * given <code>source</code>. If the <code>listener</code>
- * is already registered for all sources, nothing happens, i.e.
- * the <code>listener</code> will not receive multiple
- * invocations for the same event. May not be <code>null</code>.
- * @throws IllegalArgumentException
- * Thrown if <code>null</code> is passed for the parameter.
- */
- public abstract void addEventListener(EventListener listener);
-
- /**
- * Returns the list of all previously observed events from the given
- * <code>source</code>.
- *
- * @param source
- * The source of the events that the invoking thread is
- * interested in. May not be <code>null</code> and must be the
- * name of a previously created node, client, or server.
- * @return List of all previously observed events from the given
- * <code>source</code>.
- * @throws IllegalArgumentException
- * Thrown if <code>null</code> is passed as parameter or if
- * the <code>source</code> is unknown.
- */
- public abstract List<Event> getEventHistory(String source);
-
- /**
- * Returns whether the given <code>eventType</code> has been observed from
- * the given <code>source</code> before, or not.
- *
- * @param source
- * The source of the event that the invoking thread is interested
- * in. May not be <code>null</code> and must be the name of a
- * previously created node, client, or server.
- * @param eventType
- * The event type that the invoking thread is interested in. May
- * not be <code>null</code>.
- * @throws IllegalArgumentException
- * Thrown if <code>null</code> is passed for either of the
- * parameters or if <code>source</code> is unknown.
- * @return <code>true</code> if the event has been observed from the
- * source before, <code>false</code> otherwise.
- */
- public abstract boolean hasEventOccured(String source, EventType eventType);
-
- /**
- * Removes the given <code>listener</code> as event listener from all
- * previously registered sources. If this listener is not registered for any
- * source, nothing happens.
- *
- * @param listener
- * The listener that shall be removed from the list of registered
- * listeners. May not be <code>null</code>.
- * @throws IllegalArgumentException
- * Thrown if <code>null</code> is passed as parameter.
- */
- public abstract void removeEventListener(EventListener listener);
-
- /**
- * Checks if the given <code>eventType</code> has been observed from the
- * given <code>source</code> before; if not, blocks the invoking thread
- * until the next event of this type is fired from that source. Note that
- * this method does not restrict waiting to a timeout, so that it could
- * potentially block forever!
- *
- * @param source
- * The source of the event that the invoking thread is willing to
- * wait for. May not be <code>null</code> and must be the name
- * of a previously created node, client, or server.
- * @param eventType
- * The event type that the invoking thread is willing to wait for
- * from the given <code>source</code>. May not be
- * <code>null</code>.
- * @throws IllegalArgumentException
- * Thrown if <code>null</code> is passed for either of the
- * parameters or if the <code>source</code> is unknown.
- */
- public abstract void waitForAnyOccurence(String source, EventType eventType);
-
- /**
- * Checks if the given <code>eventType</code> has been observed from the
- * given <code>source</code> before; if not, blocks the invoking thread
- * until the next event of this type is fired from that source or the given
- * timeout of <code>maximumTimeToWaitInMillis</code> milliseconds has
- * expired.
- *
- * @param source
- * The source of the event that the invoking thread is willing to
- * wait for. May not be <code>null</code> and must be the name
- * of a previously created node, client, or server.
- * @param eventType
- * The event type that the invoking thread is willing to wait for
- * from the given <code>source</code>. May not be
- * <code>null</code>.
- * @param maximumTimeToWaitInMillis
- * The maximum time to wait in milliseconds. A positive value or
- * zero restricts waiting to this time. If this value is
- * negative, we will wait potentially forever.
- * @return <code>true</code> if an event of the given type has been fired
- * by the <code>source</code> within the given timeout,
- * <code>false</code> otherwise.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is passed for either of the
- * parameters or if the <code>source</code> is unknown.
- */
- public abstract boolean waitForAnyOccurence(String source,
- EventType eventType, long maximumTimeToWaitInMillis);
-
- /**
- * Blocks the invoking thread until the next <code>event</code> is fired
- * from the given <code>source</code>. This method only waits for the
- * next occurence of an event, regardless of previous occurrences. Note that
- * this method does not restrict waiting to a timeout, so that it could
- * potentially block forever!
- *
- * @param source
- * The source of the event that the invoking thread is willing to
- * wait for. May not be <code>null</code> and must be the name
- * of a previously created node, client, or server.
- * @param eventType
- * The event type that the invoking thread is willing to wait for
- * from the given <code>source</code>. May not be
- * <code>null</code>.
- * @throws IllegalArgumentException
- * Thrown if <code>null</code> is passed for either of the
- * parameters or if the <code>source</code> is unknown.
- */
- public abstract void waitForNextOccurence(String source, EventType eventType);
-
- /**
- * Blocks the invoking thread until the next <code>event</code> is fired
- * from the given <code>source</code> or the given timeout of
- * <code>maximumTimeToWaitInMillis</code> milliseconds has expired. This
- * method only waits for the next occurence of an event, regardless of
- * previous occurrences.
- *
- * @param source
- * The source of the event that the invoking thread is willing to
- * wait for. May not be <code>null</code> and must be the name
- * of a previously created node, client, or server.
- * @param eventType
- * The event type that the invoking thread is willing to wait for
- * from the given <code>source</code>. May not be
- * <code>null</code>.
- * @param maximumTimeToWaitInMillis
- * The maximum time to wait in milliseconds. A positive value or
- * zero restricts waiting to this time. If this value is
- * negative, we will wait potentially forever.
- * @return <code>true</code> if an event of the given type has been fired
- * by the <code>source</code> within the given timeout,
- * <code>false</code> otherwise.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is passed for either of the
- * parameters or if the <code>source</code> is unknown.
- */
- public abstract boolean waitForNextOccurence(String source,
- EventType eventType, long maximumTimeToWaitInMillis);
-
- /**
- * Registers a new event type by passing a pattern string that can be
- * applied to a regular expression when parsing Tor log statements. This is
- * useful for log statements that are only included in modified Tor
- * versions. Therefore, the event type may be an instance of a self-defined
- * class that implements <code>EventType</code>.
- *
- * @param patternString
- * The pattern string that will be used for parsing Tor log
- * statements; the syntax corresponds to java.util.regex.Pattern.
- * @param eventType
- * The event type of the event that will be fired when a log
- * statement was parsed that includes the given pattern.
- */
- public abstract void registerEventTypePattern(String patternString,
- EventType eventType);
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/EventType.java b/src/de/uniba/wiai/lspi/puppetor/EventType.java
deleted file mode 100644
index 5d57da9..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/EventType.java
+++ /dev/null
@@ -1,26 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-import java.io.Serializable;
-
-/**
- * The super interface of possible event types that are fired on a state change
- * of an asynchronous system component, e.g. a Tor process or a client/server
- * application running as thread in the background.
- *
- * @author kloesing
- */
-public interface EventType extends Serializable {
-
- /**
- * Returns a string representation of the event type name for display
- * purposes.
- *
- * @return String representation of the event type name.
- */
- public abstract String getTypeName();
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/HiddenService.java b/src/de/uniba/wiai/lspi/puppetor/HiddenService.java
deleted file mode 100644
index 5978b41..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/HiddenService.java
+++ /dev/null
@@ -1,61 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-/**
- * A <code>HiddenService</code> instance contains all configurations of a
- * hidden service that is registered at a node.
- *
- * @author kloesing
- */
-public interface HiddenService {
-
- /**
- * Determines the onion address for a previously added hidden service with
- * name <code>serviceName</code>. Requires that the node has been
- * started, i.e. is in state <code>NodeState.RUNNING</code>, and is
- * configured to provide this hidden service.
- *
- * @return The onion address string consisting of 16 base32 chars plus
- * ".onion" for hidden service versions 0 and 1 or 16 base32 chars
- * plus "." plus 24 base32 chars plus ".onion" for hidden service
- * version 2.
- * @throws IllegalArgumentException
- * Thrown if <code>null</code> or a zero-length string is
- * passed as parameter.
- * @throws IllegalStateException
- * Thrown if the node at which this hidden service is configured
- * is not in state <code>NodeState.RUNNING</code>.
- * @throws PuppeTorException
- * Thrown if the onion address of this hidden service could not
- * be read, which is also the case when the node's configuration
- * has not been written and the node has not been HUP'ed after
- * configuring the hidden service.
- */
- public abstract String determineOnionAddress() throws PuppeTorException;
-
- /**
- * Returns the name of the hidden service.
- *
- * @return The name of the hidden service.
- */
- public String getServiceName();
-
- /**
- * Returns the port on which the service listens for requests.
- *
- * @return The port on which the service listens for requests.
- */
- public int getServicePort();
-
- /**
- * Returns the virtual port that this hidden service runs on as it is
- * announced to clients.
- *
- * @return The virtual port of this hidden service.
- */
- public int getVirtualPort();
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/HiddenServiceEventType.java b/src/de/uniba/wiai/lspi/puppetor/HiddenServiceEventType.java
deleted file mode 100644
index 3ef03fe..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/HiddenServiceEventType.java
+++ /dev/null
@@ -1,215 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-/**
- * Event types that can be fired by all Tor processes performing hidden-service
- * operations.
- */
-@SuppressWarnings("serial")
-public class HiddenServiceEventType implements EventType {
-
- /**
- * String identifying the type of the event type.
- */
- String typeString;
-
- /**
- * Creates a new event type with the given type string.
- *
- * @param typeString
- * String identifying the type of the event type.
- */
- public HiddenServiceEventType(final String typeString) {
- this.typeString = typeString;
- }
-
- public String getTypeName() {
- return typeString;
- }
-
- /**
- * Alice has received an onion request; this event is parsed from a log
- * statement in connection_ap_handshake_rewrite_and_attach().
- */
- public static final HiddenServiceEventType ALICE_ONION_REQUEST_RECEIVED =
- new HiddenServiceEventType("ALICE_ONION_REQUEST_RECEIVED");
-
- /**
- * Alice sends a fetch request for a hidden service descriptor to a
- * directory server; this event is parsed from a log statement in
- * rend_client_refetch_renddesc().
- */
- public static final HiddenServiceEventType ALICE_SENDING_FETCH_DESC =
- new HiddenServiceEventType("ALICE_SENDING_FETCH_DESC");
-
- /**
- * Alice receives a reply to a previous fetch request for a hidden service
- * descriptors from a directory server; this event is parsed from a log
- * statement in connection_dir_client_reached_eof().
- */
- public static final HiddenServiceEventType ALICE_DESC_FETCHED_RECEIVED =
- new HiddenServiceEventType("ALICE_DESC_FETCHED_RECEIVED");
-
- /**
- * Alice has built a circuit to a rendezvous point and sends an
- * ESTABLISH_RENDEZVOUS cell; this event is parsed from a log statement in
- * rend_client_send_establish_rendezvous().
- */
- public static final HiddenServiceEventType ALICE_BUILT_REND_CIRC_SENDING_ESTABLISH_RENDEZVOUS =
- new HiddenServiceEventType(
- "ALICE_BUILT_REND_CIRC_SENDING_ESTABLISH_RENDEZVOUS");
-
- /**
- * Alice receives a RENDEZVOUS_ESTABLISHED cell from a rendezvous point;
- * this event is parsed from a log statement in
- * rend_client_rendezvous_acked().
- */
- public static final HiddenServiceEventType ALICE_RENDEZVOUS_ESTABLISHED_RECEIVED =
- new HiddenServiceEventType("ALICE_RENDEZVOUS_ESTABLISHED_RECEIVED");
-
- /**
- * Alice has built a circuit to an introduction point (which does not
- * automatically lead to sending an INTRODUCE1 cell, because the rendezvous
- * circuit might not be ready); this event is parsed from a log statement in
- * rend_client_introcirc_has_opened().
- */
- public static final HiddenServiceEventType ALICE_BUILT_INTRO_CIRC =
- new HiddenServiceEventType("ALICE_BUILT_INTRO_CIRC");
-
- /**
- * Alice sends an INTRODUCE1 cell to an introduction point; this event is
- * parsed from a log statement in rend_client_send_introduction().
- */
- public static final HiddenServiceEventType ALICE_SENDING_INTRODUCE1 =
- new HiddenServiceEventType("ALICE_SENDING_INTRODUCE1");
-
- /**
- * Alice has received an INTRODUCE_ACK cell as an acknowledgement to a
- * previously sent INTRODUCE1 cell; this event is parsed from a log
- * statement in rend_client_introduction_acked().
- */
- public static final HiddenServiceEventType ALICE_INTRODUCE_ACK_RECEIVED =
- new HiddenServiceEventType("ALICE_INTRODUCE_ACK_RECEIVED");
-
- /**
- * Alice has received a RENDEZVOUS2 cell and can now open an application
- * connection to the client; this event is parsed from a log statement in
- * rend_client_receive_rendezvous().
- */
- public static final HiddenServiceEventType ALICE_RENDEZVOUS2_RECEIVED_APP_CONN_OPENED =
- new HiddenServiceEventType(
- "ALICE_RENDEZVOUS2_RECEIVED_APP_CONN_OPENED");
-
- /**
- * Bob has built a circuit to an introduction point and sends an
- * ESTABLISH_INTRO cell; this event is parsed from a log statement in
- * rend_service_intro_has_opened().
- */
- public static final HiddenServiceEventType BOB_BUILT_INTRO_CIRC_SENDING_ESTABLISH_INTRO =
- new HiddenServiceEventType(
- "BOB_BUILT_INTRO_CIRC_SENDING_ESTABLISH_INTRO");
-
- /**
- * Bob has received an INTRO_ESTABLISHED cell, i.e. a node has confirmed to
- * work as introduction point; this event is parsed from a log statement in
- * rend_service_intro_established().
- */
- public static final HiddenServiceEventType BOB_INTRO_ESTABLISHED_RECEIVED =
- new HiddenServiceEventType("BOB_INTRO_ESTABLISHED_RECEIVED");
-
- /**
- * Bob posts a hidden service descriptor to the directory servers; this
- * event is parsed from a log statement in upload_service_descriptor().
- */
- public static final HiddenServiceEventType BOB_SENDING_PUBLISH_DESC =
- new HiddenServiceEventType("BOB_SENDING_PUBLISH_DESC");
-
- /**
- * Bob received a response from a directory server to a previous publish
- * request; this event is parsed from a log statement in
- * connection_dir_client_reached_eof().
- */
- public static final HiddenServiceEventType BOB_DESC_PUBLISHED_RECEIVED =
- new HiddenServiceEventType("BOB_DESC_PUBLISHED_RECEIVED");
-
- /**
- * Bob has received an INTRODUCE2 cell, i.e. a node wants to establish a
- * connection, and will now try to establish a circuit to the client's
- * rendezvous point; this event is parsed from a log statement in
- * rend_service_introduce().
- */
- public static final HiddenServiceEventType BOB_INTRODUCE2_RECEIVED =
- new HiddenServiceEventType("BOB_INTRODUCE2_RECEIVED");
-
- /**
- * Bob has built a circuit to a rendezvous point and sends a RENDEZVOUS1
- * cell; this event is parsed from a log statement in
- * rend_service_rendezvous_has_opened().
- */
- public static final HiddenServiceEventType BOB_BUILT_REND_CIRC_SENDING_RENDEZVOUS1 =
- new HiddenServiceEventType(
- "BOB_BUILT_REND_CIRC_SENDING_RENDEZVOUS1");
-
- /**
- * Bob opens a connection to the actual hidden server; this event is parsed
- * from a log statement in connection_exit_begin_conn().
- */
- public static final HiddenServiceEventType BOB_APP_CONN_OPENED =
- new HiddenServiceEventType("BOB_APP_CONN_OPENED");
-
- /**
- * The directory server has received a descriptor post request; this event
- * is parsed from a log statement in directory_handle_command_post().
- */
- public static final HiddenServiceEventType DIR_PUBLISH_DESC_RECEIVED =
- new HiddenServiceEventType("DIR_PUBLISH_DESC_RECEIVED");
-
- /**
- * The directory server has received a descriptor fetch request; this event
- * is parsed from a log statement in directory_handle_command_get().
- */
- public static final HiddenServiceEventType DIR_FETCH_DESC_RECEIVED =
- new HiddenServiceEventType("DIR_FETCH_DESC_RECEIVED");
-
- /**
- * The node received an ESTABLISH_INTRO cell, i.e. was requested to work as
- * introduction point, and replies with an INTRO_ESTABLISHED cell; this
- * event is parsed from a log statement in rend_mid_establish_intro().
- */
- public static final HiddenServiceEventType IPO_RECEIVED_ESTABLISH_INTRO_SENDING_INTRO_ESTABLISHED =
- new HiddenServiceEventType(
- "IPO_RECEIVED_ESTABLISH_INTRO_SENDING_INTRO_ESTABLISHED");
-
- /**
- * The introduction point received an INTRODUCE1 cell and reacts by sending
- * an INTRODUCE2 cell to Bob and an INTRODUCE_ACK cell to Alice; this event
- * is parsed from a log statement in rend_mid_introduce().
- */
- public static final HiddenServiceEventType IPO_RECEIVED_INTRODUCE1_SENDING_INTRODUCE2_AND_INTRODUCE_ACK =
- new HiddenServiceEventType(
- "IPO_RECEIVED_INTRODUCE1_SENDING_INTRODUCE2_AND_INTRODUCE_ACK");
-
- /**
- * The node received an ESTABLISH_RENDEZVOUS cell, i.e. was requested to
- * work as rendezvous point, and replies with a RENDEZVOUS_ESTABLISHED cell;
- * this event is parsed from a log statement in
- * rend_mid_establish_rendezvous().
- */
- public static final HiddenServiceEventType RPO_RECEIVED_ESTABLISH_RENDEZVOUS_SENDING_RENDEZVOUS_ESTABLISHED =
- new HiddenServiceEventType(
- "RPO_RECEIVED_ESTABLISH_RENDEZVOUS_SENDING_RENDEZVOUS_ESTABLISHED");
-
- /**
- * The rendezvous point received a RENDEZVOUS1 cell and reacts by sending a
- * RENDEZVOUS2 cell to Alice; this event is parsed from a log statement in
- * rend_mid_rendezvous().
- */
- public static final HiddenServiceEventType RPO_RECEIVING_RENDEZVOUS1_SENDING_RENDEZVOUS2 =
- new HiddenServiceEventType(
- "RPO_RECEIVING_RENDEZVOUS1_SENDING_RENDEZVOUS2");
-
-}
\ No newline at end of file
diff --git a/src/de/uniba/wiai/lspi/puppetor/Network.java b/src/de/uniba/wiai/lspi/puppetor/Network.java
deleted file mode 100644
index 5ebd3c9..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/Network.java
+++ /dev/null
@@ -1,657 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-import java.io.File;
-import java.util.List;
-import java.util.Map;
-
-/**
- * A Network instance constitutes the central object of any test run and is
- * aware of the node configuration. It creates all nodes for this configuration
- * and is able to perform common operations on these nodes. Apart from the
- * factory methods, all other operations could also be performed manually by an
- * application using the appropriate interfaces.
- *
- * @author kloesing
- */
-public interface Network {
-
- /**
- * <p>
- * Configures this network as private Tor network by exchanging directory
- * strings and router fingerprints between the nodes of this network.
- * Afterwards, the nodes will be able to run a private Tor network,
- * separated from public directory servers and onion routers.
- * </p>
- *
- * <p>
- * The configuration is done in two steps:
- * <ol>
- * <li>Directory strings of directory nodes are added to the configurations
- * of all nodes in the other network.</li>
- * <li>Router fingerprints of all router and directory nodes are added to
- * the <code>approved-routers</code> files of the directory nodes.</li>
- * </ol>
- * </p>
- *
- * <p>
- * This operation may be invoked in any state of the contained nodes.
- * However, a network that does not have directory nodes of its own but
- * relies on directory nodes of a merged network <b>should not be started
- * before being configured as private network!</b> Otherwise it would
- * connect to the public Tor network before being merged with the other
- * private Tor network. However, it may also be invoked at a later time,
- * e.g. to admit new nodes.
- * </p>
- *
- * <p>
- * This operation does not write any configurations to disk and neither
- * starts a nodes nor sends HUP signals to running nodes. These operations
- * are left to the application, so that they have more control over the
- * network behavior.
- * </p>
- *
- * <p>
- * Applications need to ensure that there are enough directory nodes (2) and
- * router nodes (3) in the network to allow normal operation.
- * </p>
- *
- * @throws PuppeTorException
- * Thrown if an I/O problem occurs while determining the nodes'
- * fingerprints.
- */
- public abstract void configureAsPrivateNetwork() throws PuppeTorException;
-
- /**
- * XXX document properly -SH
- */
- public void configureAsPartOfPrivateNetwork(
- List<String> authorizedDirectoriesFingerprints);
-
-
- /**
- * Creates a new client application, but does not yet perform a request.
- *
- * @param clientApplicationName
- * The name for this client application, which is used for
- * logging purposes and as event source. May neither be
- * <code>null</code> or a zero-length string. The name needs to
- * be unique in this network.
- * @param targetAddress
- * The target for requests sent by this client application. Can
- * be a publicly available URL or an onion address. May neither
- * be <code>null</code> or a zero-length string.
- * @param targetPort
- * The TCP port for requests sent by this client application. If
- * the target address is an onion address, this port is the
- * virtual port that the hidden service has announced. May not be
- * negative or greater than 65535.
- * @param socksPort
- * The TCP port on which a local Tor process is waiting for
- * incoming SOCKS requests. May not be negative or greater than
- * 65535.
- * @return Reference to the created client application.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given for either of the
- * parameters.
- */
- public abstract ClientApplication createClient(
- String clientApplicationName, String targetAddress, int targetPort,
- int socksPort);
-
- /**
- * Creates a new directory node with automatically assigned ports and adds
- * it to the network, but does not yet write its configuration to disk or
- * start the corresponding Tor process.
- *
- * @param nodeName
- * The name for this node, which is used as name for the working
- * directory, for logging purposes, as node nickname, and as
- * event source. May neither be <code>null</code> or have zero
- * or more than 19 alpha-numeric characters. The node name needs
- * to be unique in this network.
- * @return Reference to the created directory node.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given as node name.
- */
- public abstract DirectoryNode createDirectory(String nodeName);
-
- /**
- * Creates a new directory node with automatically assigned ports that will
- * listen on the given IP address and adds it to the network, but does not
- * yet write its configuration to disk or start the corresponding Tor
- * process.
- *
- * @param nodeName
- * The name for this node, which is used as name for the working
- * directory, for logging purposes, as node nickname, and as
- * event source. May neither be <code>null</code> or have zero
- * or more than 19 alpha-numeric characters. The node name needs
- * to be unique in this network.
- * @param serverIpAddress
- * The IP address on which the node will listen. Must be a valid
- * IP v4 address in dotted decimal notation. May not be
- * <code>null</code>.
- * @return Reference to the created directory node.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given as node name.
- */
- public abstract DirectoryNode createDirectory(String nodeName,
- String serverIpAddress);
-
- /**
- * Creates a new directory node and adds it to the network, but does not yet
- * write its configuration to disk or start the corresponding Tor process.
- *
- * @param nodeName
- * The name for this node, which is used as name for the working
- * directory, for logging purposes, as node nickname, and as
- * event source. May neither be <code>null</code> or have zero
- * or more than 19 alpha-numeric characters. The node name needs
- * to be unique in this network.
- * @param controlPort
- * The TCP port on which the corresponding Tor process will wait
- * for a controller. May not be negative or greater than 65535.
- * @param socksPort
- * The TCP port on which the corresponding Tor process will wait
- * for incoming SOCKS requests. May not be negative or greater
- * than 65535.
- * @param orPort
- * The TCP port on which the corresponding Tor process will wait
- * for incoming requests from other onion routers. May not be
- * negative or greater than 65535.
- * @param dirPort
- * The TCP port on which the corresponding Tor process will wait
- * for incoming directory requests. May not be negative or
- * greater than 65535.
- * @return Reference to the created directory node.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given for either of the
- * parameters.
- */
- public abstract DirectoryNode createDirectory(String nodeName,
- int controlPort, int socksPort, int orPort, int dirPort);
-
- /**
- * Creates a new directory node that will listen on the given IP address and
- * adds it to the network, but does not yet write its configuration to disk
- * or start the corresponding Tor process.
- *
- * @param nodeName
- * The name for this node, which is used as name for the working
- * directory, for logging purposes, as node nickname, and as
- * event source. May neither be <code>null</code> or have zero
- * or more than 19 alpha-numeric characters. The node name needs
- * to be unique in this network.
- * @param controlPort
- * The TCP port on which the corresponding Tor process will wait
- * for a controller. May not be negative or greater than 65535.
- * @param socksPort
- * The TCP port on which the corresponding Tor process will wait
- * for incoming SOCKS requests. May not be negative or greater
- * than 65535.
- * @param orPort
- * The TCP port on which the corresponding Tor process will wait
- * for incoming requests from other onion routers. May not be
- * negative or greater than 65535.
- * @param dirPort
- * The TCP port on which the corresponding Tor process will wait
- * for incoming directory requests. May not be negative or
- * greater than 65535.
- * @param serverIpAddress
- * The IP address on which the node will listen. Must be a valid
- * IP v4 address in dotted decimal notation. May not be
- * <code>null</code>.
- * @return Reference to the created directory node.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given for either of the
- * parameters.
- */
- public abstract DirectoryNode createDirectory(String nodeName,
- int controlPort, int socksPort, int orPort, int dirPort,
- String serverIpAddress);
-
- /**
- * Creates a new <code>ProxyNode</code> with automatically assigned ports
- * and adds it to the network, but does not yet write its configuration to
- * disk or start the corresponding Tor process.
- *
- * @param nodeName
- * The name for this node, which is used as name for the working
- * directory, for logging purposes, and as event source. May
- * neither be <code>null</code> or have zero or more than 19
- * alpha-numeric characters. The node name needs to be unique in
- * this network.
- * @return Reference to the created proxy node.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given as node name.
- */
- public abstract ProxyNode createProxy(String nodeName);
-
- /**
- * Creates a new <code>ProxyNode</code> and adds it to the network, but
- * does not yet write its configuration to disk or start the corresponding
- * Tor process.
- *
- * @param nodeName
- * The name for this node, which is used as name for the working
- * directory, for logging purposes, and as event source. May
- * neither be <code>null</code> or have zero or more than 19
- * alpha-numeric characters. The node name needs to be unique in
- * this network.
- * @param controlPort
- * The TCP port on which the corresponding Tor process will wait
- * for a controller. May not be negative or greater than 65535.
- * @param socksPort
- * The TCP port on which the corresponding Tor process will wait
- * for incoming SOCKS requests. May not be negative or greater
- * than 65535.
- * @return Reference to the created proxy node.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given for either of the
- * parameters.
- */
- public abstract ProxyNode createProxy(String nodeName, int controlPort,
- int socksPort);
-
- /**
- * Creates a new <code>RouterNode</code> with automatically assigned ports
- * and adds it to the network, but does not yet write its configuration to
- * disk or start the corresponding Tor process.
- *
- * @param nodeName
- * The name for this node, which is used as name for the working
- * directory, for logging purposes, as node nickname, and as
- * event source. May neither be <code>null</code> or have zero
- * or more than 19 alpha-numeric characters. The node name needs
- * to be unique in this network.
- * @return Reference to the created router node.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given as node name.
- */
- public abstract RouterNode createRouter(String nodeName);
-
- /**
- * Creates a new <code>RouterNode</code> and adds it to the network, but
- * does not yet write its configuration to disk or start the corresponding
- * Tor process.
- *
- * @param nodeName
- * The name for this node, which is used as name for the working
- * directory, for logging purposes, as node nickname, and as
- * event source. May neither be <code>null</code> or have zero
- * or more than 19 alpha-numeric characters. The node name needs
- * to be unique in this network.
- * @param controlPort
- * The TCP port on which the corresponding Tor process will wait
- * for a controller. May not be negative or greater than 65535.
- * @param socksPort
- * The TCP port on which the corresponding Tor process will wait
- * for incoming SOCKS requests. May not be negative or greater
- * than 65535.
- * @param orPort
- * The TCP port on which the corresponding Tor process will wait
- * for incoming requests from other onion routers. May not be
- * negative or greater than 65535.
- * @param dirPort
- * The TCP port on which the corresponding Tor process will wait
- * for incoming directory requests which in fact are requests for
- * the mirrored directory. May not be negative or greater than
- * 65535.
- * @return Reference to the created router node.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given for either of the
- * parameters.
- */
- public abstract RouterNode createRouter(String nodeName, int controlPort,
- int socksPort, int orPort, int dirPort);
-
- /**
- * Creates a new <code>RouterNode</code> with automatically assigned ports
- * that will listen on the given IP address and adds it to the network, but
- * does not yet write its configuration to disk or start the corresponding
- * Tor process.
- *
- * @param nodeName
- * The name for this node, which is used as name for the working
- * directory, for logging purposes, as node nickname, and as
- * event source. May neither be <code>null</code> or have zero
- * or more than 19 alpha-numeric characters. The node name needs
- * to be unique in this network.
- * @param serverIpAddress
- * The IP address on which the node will listen. Must be a valid
- * IP v4 address in dotted decimal notation. May not be
- * <code>null</code>.
- * @return Reference to the created router node.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given as node name.
- */
- public abstract RouterNode createRouter(String nodeName,
- String serverIpAddress);
-
- /**
- * Creates a new <code>RouterNode</code> that will listen on the given IP
- * address and adds it to the network, but does not yet write its
- * configuration to disk or start the corresponding Tor process.
- *
- * @param nodeName
- * The name for this node, which is used as name for the working
- * directory, for logging purposes, as node nickname, and as
- * event source. May neither be <code>null</code> or have zero
- * or more than 19 alpha-numeric characters. The node name needs
- * to be unique in this network.
- * @param controlPort
- * The TCP port on which the corresponding Tor process will wait
- * for a controller. May not be negative or greater than 65535.
- * @param socksPort
- * The TCP port on which the corresponding Tor process will wait
- * for incoming SOCKS requests. May not be negative or greater
- * than 65535.
- * @param orPort
- * The TCP port on which the corresponding Tor process will wait
- * for incoming requests from other onion routers. May not be
- * negative or greater than 65535.
- * @param dirPort
- * The TCP port on which the corresponding Tor process will wait
- * for incoming directory requests which in fact are requests for
- * the mirrored directory. May not be negative or greater than
- * 65535.
- * @param serverIpAddress
- * The IP address on which the node will listen. Must be a valid
- * IP v4 address in dotted decimal notation. May not be
- * <code>null</code>.
- * @return Reference to the created router node.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given for either of the
- * parameters.
- */
- public abstract RouterNode createRouter(String nodeName, int controlPort,
- int socksPort, int orPort, int dirPort, String serverIpAddress);
-
- /**
- * Creates a new <code>ServerApplication</code> with automatically
- * assigned ports, but does not start listening for incoming requests.
- *
- * @param serverApplicationName
- * The name for this server application, which is used for
- * logging purposes and as event source. May neither be
- * <code>null</code> or a zero-length string. The name needs to
- * be unique in this network.
- * @return Reference to the created server application.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given as server application
- * name.
- */
- public abstract ServerApplication createServer(String serverApplicationName);
-
- /**
- * Creates a new <code>ServerApplication</code>, but does not start
- * listening for incoming requests.
- *
- * @param serverApplicationName
- * The name for this server application, which is used for
- * logging purposes and as event source. May neither be
- * <code>null</code> or a zero-length string. The name needs to
- * be unique in this network.
- * @param serverPort
- * The TCP port on which the server will wait for incoming
- * requests. May not be negative or greater than 65535.
- * @return Reference to the created server application.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given for either of the
- * parameters.
- */
- public abstract ServerApplication createServer(
- String serverApplicationName, int serverPort);
-
- /**
- * Returns a reference on the (single) event manager for this network.
- *
- * @return Reference on the (single) event manager for this network.
- */
- public abstract EventManager getEventManager();
-
- /**
- * Returns (a copy of) the map containing the names of all directory nodes
- * as keys and the corresponding directory nodes as values.
- *
- * @return Map containing all directory nodes.
- */
- public abstract Map<String, DirectoryNode> getAllDirectoryNodes();
-
- /**
- * Returns (a copy of) the map containing the names of all router nodes
- * (only those that are not acting as directory nodes at the same time) as
- * keys and the corresponding router nodes as values.
- *
- * @return Map containing all router nodes.
- */
- public abstract Map<String, RouterNode> getAllRouterNodes();
-
- /**
- * Returns (a copy of) the map containing the names of all proxy nodes (only
- * those that are not acting as router or directory nodes at the same time)
- * as keys and the corresponding proxy nodes as values.
- *
- * @return Map containing all proxy nodes.
- */
- public abstract Map<String, ProxyNode> getAllProxyNodes();
-
- /**
- * Returns (a copy of) the map containing the names of all nodes as keys and
- * the corresponding proxy nodes as values.
- *
- * @return Map containing all nodes.
- */
- public abstract Map<String, ProxyNode> getAllNodes();
-
- /**
- * Returns the node with name <code>nodeName</code> or <code>null</code>
- * if no such node exists.
- *
- * @param nodeName
- * The node name to look up.
- * @return The node with name <code>nodeName</code>.
- */
- public abstract ProxyNode getNode(String nodeName);
-
- /**
- * <p>
- * Sends a HUP signal to all nodes in the network in regular intervals and
- * blocks the invoking thread until all nodes have reported to have
- * successfully opened a circuit.
- * </p>
- *
- * <p>
- * First, the method waits for <code>hupInterval</code> milliseconds for
- * the nodes to have successfully opened a circuit. If they do not succeed
- * within this time, a HUP signal is sent to all nodes and the method waits
- * for another <code>hupInterval</code> milliseconds. In total, the method
- * sends at most <code>tries</code> HUP signals before giving up and
- * returning with <code>false</code>. Thus, the maximum waiting time is
- * <code>(tries + 1)</code> times <code>hupInterval</code>. As soon as
- * all nodes have successfully opened circuits, the method returns with
- * <code>true</code>. This operation can only be invoked, if all nodes in
- * the network are in state <code>NodeState.RUNNING</code>.
- * </p>
- *
- * @param tries
- * The maximum number of HUP signals that are sent to the Tor
- * processes. Negative values are not allowed. A value of zero
- * means to wait only for the given time of
- * <code>hupInterval</code> milliseconds without sending a HUP
- * signal. Typical values depend on the network being a public or
- * private Tor network and range about 3 to 5 tries.
- * @param hupInterval
- * The time in milliseconds that the method will wait between
- * sending HUP signals. Negative values are not allowed.
- * Typically, values should not be smaller than 5 seconds to
- * permit Tor to stabilize.
- * @throws IllegalStateException
- * Thrown if at least one node is not in state
- * <code>NodeState.RUNNING</code>.
- * @throws IllegalArgumentException
- * Thrown if a negative value is given for either
- * <code>tries</code> or <code>hupInterval</code>.
- * @throws PuppeTorException
- * Thrown if an I/O problem occurs while sending HUP signals.
- * @return <code>true</code> if all nodes have reported to have
- * successfully opened a circuit, <code>false</code> otherwise.
- */
- public abstract boolean hupUntilUp(int tries, long hupInterval)
- throws PuppeTorException;
-
- /**
- * Sends a HUP signal to all nodes in the network once. This operation can
- * only be invoked, if all nodes in the network are in state
- * <code>NodeState.RUNNING</code>.
- *
- * @throws IllegalStateException
- * Thrown if at least one node is not in state
- * <code>NodeState.RUNNING</code>.
- * @throws PuppeTorException
- * Thrown if an I/O problem occurs while sending HUP signals.
- */
- public abstract void hupAllNodes() throws PuppeTorException;
-
- /**
- * Sends a HUP signal to all directory nodes in the network once. This
- * operation can only be invoked, if all directory nodes in the network are
- * in state <code>NodeState.RUNNING</code>.
- *
- * @throws IllegalStateException
- * Thrown if at least one directory node is not in state
- * <code>NodeState.RUNNING</code>.
- * @throws PuppeTorException
- * Thrown if an I/O problem occurs while sending HUP signals.
- */
- public abstract void hupAllDirectories() throws PuppeTorException;
-
- /**
- * Attempts to shut down all running nodes. The method blocks until all
- * shutdown requests have been sent and either returns, or throws the first
- * exception that has been observed when shutting down nodes. The method can
- * be assumed to return very quickly. If there are no running nodes in this
- * network, this operation has no effect.
- *
- * @throws PuppeTorException
- * Thrown if an I/O problem occurs while shutting down the
- * nodes.
- */
- public abstract void shutdownNodes() throws PuppeTorException;
-
- /**
- * Attempts to start all nodes within a given timeout of
- * <code>maximumTimeToWaitInMillis</code> milliseconds. The method returns
- * as soon as all nodes have started and opened their control port so that
- * we can connect to them. It returns a boolean that states whether the
- * operation was either successful or has timed out. This operation can only
- * be invoked, if all nodes in the network have written their configuration,
- * i.e. are not in state <code>NodeState.CONFIGURING</code> anymore.
- *
- * @param maximumTimeToWaitInMillis
- * The maximum time to wait in milliseconds. A positive value or
- * zero restricts waiting to this time. Negative values are not
- * allowed. Typical values are in the range of a few seconds.
- * @return <code>true</code> if all nodes could be started successfully,
- * <code>false</code> if a timeout has occured.
- * @throws IllegalStateException
- * Thrown if at least one node in the network is still in state
- * <code>NodeState.CONFIGURING</code>.
- * @throws IllegalArgumentException
- * Thrown if a negative value is given for
- * <code>maximumTimeToWaitInMillis</code>.
- * @throws PuppeTorException
- * Thrown if an I/O problem occurs while starting the nodes.
- */
- public abstract boolean startNodes(long maximumTimeToWaitInMillis)
- throws PuppeTorException;
-
- /**
- * Writes the configurations for all nodes in the network to disk, including
- * <code>torrc</code> and <code>approved-routers</code> files. This
- * method is assumed to return very quickly. In case of a private network,
- * <code>configureAsPrivateNetwork</code> should be invoked in advance to
- * this method!
- *
- * @throws PuppeTorException
- * Thrown if an I/O problem occurs while writing to the nodes'
- * working directories.
- */
- public abstract void writeConfigurations() throws PuppeTorException;
-
- /**
- * Returns the working directory of this network configuration which is in
- * <code>test-env/networkName/</code>.
- *
- * @return Working directory of this network.
- */
- public abstract File getWorkingDirectory();
-
- /**
- * Returns all configuration strings of the template of a node class that
- * will be added to future instances of this node class and its subclasses.
- * Note that the result only contains those configuration strings that are
- * added by this node class to possible superclasses and that parameterized
- * configuration strings, e.g. port configurations, are not included.
- *
- * @param nodeClass
- * The class which will be configured with the returned template
- * configuration; may not be <code>null</code>.
- * @return The template configuration for the given node class.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given for the parameter.
- */
- public abstract List<String> getTemplateConfiguration(
- Class<? extends ProxyNode> nodeClass);
-
- /**
- * Adds a configuration string to the template of a node class, so that it
- * will be added to future instances of this node class and its subclasses.
- *
- * @param nodeClass
- * The class of nodes of which future instances will have the
- * given configuration string; may not be <code>null</code>.
- * @param templateConfigurationString
- * The configuration string to add; may neither be
- * <code>null</code> nor a zero-length string, and must consist
- * of configuration key and value.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given for either of the
- * parameters.
- */
- public abstract void addTemplateConfiguration(
- Class<? extends ProxyNode> nodeClass,
- String templateConfigurationString);
-
- /**
- * Removes a configuration string from the template of a node class, so that
- * it will not be added to future instances of this node class and its
- * subclasses.
- *
- * @param nodeClass
- * The class of nodes of which future instances will have the
- * given configuration string; may not be <code>null</code>.
- * @param templateConfigurationKey
- * The configuration key to remove; may neither be
- * <code>null</code> nor a zero-length key.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given for either of the
- * parameters.
- */
- public abstract void removeTemplateConfiguration(
- Class<? extends ProxyNode> nodeClass,
- String templateConfigurationKey);
-
- /**
- * Returns the name of this network.
- *
- * @return The name of this network.
- */
- public abstract String getNetworkName();
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/NetworkFactory.java b/src/de/uniba/wiai/lspi/puppetor/NetworkFactory.java
deleted file mode 100644
index f7b9837..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/NetworkFactory.java
+++ /dev/null
@@ -1,82 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-import java.util.concurrent.ConcurrentHashMap;
-import java.util.concurrent.ConcurrentMap;
-
-import de.uniba.wiai.lspi.puppetor.impl.NetworkImpl;
-
-/**
- * The <code>NetworkFactory</code> is a concrete factory that can create
- * <code>Network</code> instances.
- *
- * TODO At the moment, this class uses the concrete class NetworkImpl to
- * implement its only factory method. If we want to make this a real abstract
- * factory, we need to replace the concrete constructor by reading the class
- * name of the class implementing Network from a property file and invoking its
- * constructor using reflection. Currently, this is the only place where we
- * reference a class from the impl package.
- *
- * @author kloesing
- */
-public abstract class NetworkFactory {
-
- final private static ConcurrentMap<String, Network> networks =
- new ConcurrentHashMap<String, Network>();
-
- /**
- * Creates a new network that is required for a test run. The new network is
- * initially unpopulated and creates its own working directory at
- * test-env/randomTestID/. The network automatically assigns port numbers to
- * newly created nodes starting at <code>7000</code>.
- *
- * @param networkName
- * Name of this network configuration.
- * @return A new network instance.
- */
- public static Network createNetwork(final String networkName) {
- final Network network = new NetworkImpl(networkName);
- networks.put(networkName, network);
- return network;
- }
-
- /**
- * Creates a new network that is required for a test run. The new network is
- * initially unpopulated and creates its own working directory at
- * test-env/randomTestID/. The network automatically assigns port numbers to
- * newly created nodes starting at <code>startPort</code>.
- *
- * @param networkName
- * Name of this network configuration.
- * @param startPort
- * The initial value for automatically assigned port numbers of
- * nodes created by this <code>Network</code>; must be a value
- * between <code>1024</code> and <code>65535</code>.
- * Applications need to ensure that there are enough ports left
- * to the maximum number port <code>65535</code> for all
- * created nodes.
- * @return A new network instance.
- */
- public static Network createNetwork(final String networkName,
- final int startPort) {
- final Network network = new NetworkImpl(networkName, startPort);
- networks.put(networkName, network);
- return network;
- }
-
- /**
- *
- */
- public static Network getNetworkByName(final String networkName) {
- final Network network = networks.get(networkName);
- if (network == null) {
- throw new IllegalStateException("BUG: The network with name "
- + networkName + " must exist and be non-null!");
- }
- return network;
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/NodeEventType.java b/src/de/uniba/wiai/lspi/puppetor/NodeEventType.java
deleted file mode 100644
index e598a3c..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/NodeEventType.java
+++ /dev/null
@@ -1,60 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-/**
- * Event types that can be fired by all Tor processes.
- */
-@SuppressWarnings("serial")
-public class NodeEventType implements EventType {
-
- /**
- * String identifying the type of the event type.
- */
- String typeString;
-
- /**
- * Creates a new event type with the given type string.
- *
- * @param typeString
- * String identifying the type of the event type.
- */
- public NodeEventType(final String typeString) {
- this.typeString = typeString;
- }
-
- public String getTypeName() {
- return typeString;
- }
-
- /**
- * The node was started and we managed to connect to its control port; this
- * event is fired internally and not parsed from a log statement from Tor.
- */
- public static final NodeEventType NODE_STARTED =
- new NodeEventType("NODE_STARTED");
-
- /**
- * The node has opened its control port; this event is parsed from a log
- * statement in connection_create_listener().
- */
- public static final NodeEventType NODE_CONTROL_PORT_OPENED =
- new NodeEventType("NODE_CONTROL_PORT_OPENED");
-
- /**
- * The node which has successfully opened a circuit; this event is parsed
- * from a log statement in circuit_send_next_onion_skin().
- */
- public static final NodeEventType NODE_CIRCUIT_OPENED =
- new NodeEventType("NODE_CIRCUIT_OPENED");
-
- /**
- * The node was stopped; this event is fired internally and not parsed from
- * a log statement from Tor.
- */
- public static final NodeEventType NODE_STOPPED =
- new NodeEventType("NODE_STOPPED");
-}
\ No newline at end of file
diff --git a/src/de/uniba/wiai/lspi/puppetor/NodeState.java b/src/de/uniba/wiai/lspi/puppetor/NodeState.java
deleted file mode 100644
index 1891031..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/NodeState.java
+++ /dev/null
@@ -1,44 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-/**
- * The <code>NodeState</code> constitutes the state of a single Tor node. In
- * contrast to <code>EventType</code> the node states depend only on the
- * methods that have been invoked on these objects, and not on asynchronous
- * state changes. Most operations of <code>ProxyNode</code> and its subclasses
- * require a certain <code>NodeState</code> as precondition and may ensure
- * another <code>NodeState</code> as postcondition. There is a prescribed
- * order of states.
- *
- * @author kloesing
- */
-public enum NodeState {
-
- /**
- * The configuration of this node has not been written to disk. This is the
- * initial state of a <code>ProxyNode</code> or one of its subclasses.
- */
- CONFIGURING,
-
- /**
- * The configuration of this node has been written to disk, but the Tor
- * process has not been started, yet. This state could be useful to review
- * the configuration that has been written to disk.
- */
- CONFIGURATION_WRITTEN,
-
- /**
- * The node has been started and is running.
- */
- RUNNING,
-
- /**
- * The node had been started and shut down. It cannot be started at a later
- * time anymore.
- */
- SHUT_DOWN
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/ProxyNode.java b/src/de/uniba/wiai/lspi/puppetor/ProxyNode.java
deleted file mode 100644
index 3d46567..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/ProxyNode.java
+++ /dev/null
@@ -1,269 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-import java.util.List;
-
-/**
- * <p>
- * A <code>ProxyNode</code> represents a Tor process that is configured as
- * onion proxy, i.e. to relay traffic from a local application to the Tor
- * network and vice versa, and does not route traffic on behalf of remote
- * applications. It is the superclass for other node types that extend the
- * configuration of a </code>ProxyNode</code>.
- * </p>
- *
- * <p>
- * <b>Pay extra attention when using in private network!</b> Using proxy nodes
- * in private networks in the same way as router nodes will fail! Tor has two
- * different strategies for downloading network status documents: Directory
- * caches (router nodes) download these documents after every HUP signal and
- * then accept all contained router entries. But directory clients (proxy nodes)
- * only download network status documents, if the most recent download lies at
- * least 30 minutes in the past, and then accept only those of the contained
- * router entries that are at least 10 minutes old. However, when starting all
- * nodes of a private network at once, directories cannot contain 10 minutes old
- * router descriptors. You have at least the following options to cope with this
- * problem:
- * </p>
- *
- * <ul>
- * <li>Use router nodes instead of proxy nodes,</li>
- * <li>start proxy nodes with a delay of at least 10 minutes to be sure that
- * the router descriptors stored at directory authorities will be accepted by
- * directory clients, or</li>
- * <li>change the constants <code>ESTIMATED_PROPAGATION_TIME</code> and
- * <code>NETWORKSTATUS_CLIENT_DL_INTERVAL</code> in Tor to values smaller than
- * your overall HUP time for starting the network.</li>
- * </ul>
- *
- * @author kloesing
- */
-public interface ProxyNode {
-
- /**
- * Adds the entries for a hidden service to the configuration of this node.
- *
- * @param serviceName
- * Name of the hidden service that will be used as name for the
- * hidden service directory. May neither be <code>null</code>
- * or a zero-length string.
- * @param servicePort
- * The TCP port on which the service will be available for
- * requests. This can, but need not be different from the virtual
- * port that is announced to clients. May not be negative or
- * greater than 65535.
- * @param virtualPort
- * The virtual TCP port that this hidden service runs on as it is
- * announced to clients. May not be negative or greater than
- * 65535.
- * @return <code>HiddenService</code> object containing the configuration
- * of the created hidden service.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given for either of the
- * parameters.
- */
- public abstract HiddenService addHiddenService(String serviceName,
- int servicePort, int virtualPort);
-
- /**
- * Adds the entries for a hidden service with virtual port 80 to the
- * configuration of this node.
- *
- * @param serviceName
- * Name of the hidden service that will be used as name for the
- * hidden service directory. May neither be <code>null</code>
- * or a zero-length string.
- * @param servicePort
- * The TCP port on which the service will be available for
- * requests. This can, but need not be different from the virtual
- * port that is announced to clients. May not be negative or
- * greater than 65535.
- * @return <code>HiddenService</code> object containing the configuration
- * of the created hidden service.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given for either of the
- * parameters.
- */
- public abstract HiddenService addHiddenService(String serviceName,
- int servicePort);
-
- /**
- * Adds the entries for a hidden service with an automatically assigned
- * service port and virtual port 80 to the configuration of this node.
- *
- * service port automatically assigned virtual port 80
- *
- * @param serviceName
- * Name of the hidden service that will be used as name for the
- * hidden service directory. May neither be <code>null</code>
- * or a zero-length string.
- * @return <code>HiddenService</code> object containing the configuration
- * of the created hidden service.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given for the parameter.
- */
- public abstract HiddenService addHiddenService(String serviceName);
-
- /**
- * Adds the given configuration string, consisting of "<configuration key>
- * <configuration value>", to the configuration of this node.
- *
- * @param configurationString
- * The configuration string to be added.
- * @throws IllegalArgumentException
- * Thrown if the given configuration string is either
- * <code>null</code>, a zero-length string, or does not
- * consist of configuration key and value.
- */
- public abstract void addConfiguration(String configurationString);
-
- /**
- * Adds the given configuration strings, each consisting of "<configuration
- * key> <configuration value>", to the configuration of this node.
- *
- * @param configurationStrings
- * A list of the configuration strings to be added.
- * @throws IllegalArgumentException
- * Thrown if the given list is <code>null</code>, or any of
- * the contained strings is either <code>null</code>, a
- * zero-length string, or does not consist of configuration key
- * and value.
- */
- public abstract void addConfigurations(List<String> configurationStrings);
-
- /**
- * Replaces the first configuration string, consisting of "<configuration
- * key> <configuration value>", that contains the same configuration key as
- * <code>configurationString</code> by this new configuration string; if
- * multiple occurrences of the given configuration key are found, only the
- * first occurrence is replaced; if no configuration can be found, the
- * configuration string is appended.
- *
- * @param configurationString
- * The replacing configuration string.
- * @throws IllegalArgumentException
- * Thrown if the given configuration string is either
- * <code>null</code>, a zero-length string, or does not
- * consist of configuration key and value.
- */
- public abstract void replaceConfiguration(String configurationString);
-
- /**
- * Removes all configuration strings containing the given configuration key
- * in "<configuration key> <configuration value>", regardless of their
- * configuration value.
- *
- * @param configurationKey
- * The configuration key to remove.
- * @throws IllegalArgumentException
- * Thrown if the given configuration key is either
- * <code>null</code> or a zero-length key.
- */
- public abstract void removeConfiguration(String configurationKey);
-
- /**
- * Returns the name of this node.
- *
- * @return The name of this node.
- */
- public abstract String getNodeName();
-
- /**
- * Returns the state of this node.
- *
- * @return The state of this node.
- */
- public abstract NodeState getNodeState();
-
- /**
- * Sends a HUP command to the process via its control port to restart it;
- * can only be done if the node has already been started, i.e. is in state
- * <code>NodeState.RUNNING</code>!
- *
- * @throws PuppeTorException
- * Thrown if an I/O problem occurs while sending the HUP signal.
- * @throws IllegalStateException
- * Thrown if node is not in state <code>NodeState.RUNNING</code>.
- */
- public abstract void hup() throws PuppeTorException;
-
- /**
- * Shuts down the Tor process corresponding to this node immediately. This
- * is done by sending the <code>SHUTDOWN</code> signal twice, so that
- * those nodes extending <code>ProxyNode</code> which have opened their OR
- * port shutdown immediately, too.
- *
- * @throws IllegalStateException
- * Thrown if this node is not in state
- * <code>NodeState.RUNNING</code>.
- * @throws PuppeTorException
- * Thrown if an I/O problem occurs while sending the
- * <code>SHUTDOWN</code> signal.
- */
- public abstract void shutdown() throws PuppeTorException;
-
- /**
- * Starts the Tor process for this node and connects to the control port as
- * soon as it is opened. <b>In order for this method to succeed it is
- * absolutely necessary, that logging on the console is not changed in the
- * configuration of this node to a higher level than NOTICE, because the
- * output is parsed to see when the control port is opened.</b>
- *
- * @param maximumTimeToWaitInMillis
- * Maximum time in milliseconds we will wait for the Tor process
- * to be started and the control port being opened. If this value
- * is negative or zero, we will wait potentially forever.
- * @return <code>true</code> if the node could be started successfully,
- * <code>false</code> otherwise.
- * @throws IllegalStateException
- * Thrown if node is not in state
- * <code>NodeState.CONFIGURATION_WRITTEN</code>, i.e. if
- * either configuration has not been written or the process has
- * already been started.
- * @throws PuppeTorException
- * Thrown if either the process could not be started, or the
- * connection to the control port could not be established.
- */
- public abstract boolean startNode(long maximumTimeToWaitInMillis)
- throws PuppeTorException;
-
- /**
- * Writes the configuration of this node to the <code>torrc</code> file in
- * its working directory and changes the state to
- * <code>NodeState.CONFIGURATION_WRITTEN</code>, if it was in state
- * <code>NodeState.CONFIGURING</code> before.
- *
- * @throws PuppeTorException
- * Thrown if the configuration file <code>torrc</code> cannot
- * be written to disk.
- */
- public abstract void writeConfiguration() throws PuppeTorException;
-
- /**
- * Returns the SOCKS port of this node.
- *
- * @return The SOCKS port of this node.
- */
- public abstract int getSocksPort();
-
- /**
- * Returns the control port of this node.
- *
- * @return The control port of this node.
- */
- public abstract int getControlPort();
-
- /**
- * Returns (a copy of) the list of strings containing the configuration of
- * this node.
- *
- * @return (A copy of) the list of strings containing the configuration of
- * this node.
- */
- public abstract List<String> getConfiguration();
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/PuppeTorException.java b/src/de/uniba/wiai/lspi/puppetor/PuppeTorException.java
deleted file mode 100644
index 648141e..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/PuppeTorException.java
+++ /dev/null
@@ -1,65 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-/**
- * The <code>PuppeTorException</code> comprises all kinds of checked
- * exceptions that occur when interacting with the JVM-external Tor processes or
- * with the local file system. Any occurence of this exception denotes either a
- * configuration problem that can only be solved outside of the JVM, or an
- * unexpected problem. In contrast to this, all kinds of programming errors of
- * an application using this API (invoking a method with wrong parameter values,
- * in wrong state, etc.) will instead cause appropriate runtime exceptions from
- * the Java API.
- *
- * @author kloesing
- */
-@SuppressWarnings("serial")
-public class PuppeTorException extends Exception {
-
- /**
- * Creates a <code>PuppeTorException</code> without detail message or
- * cause.
- */
- public PuppeTorException() {
- super();
- }
-
- /**
- * Creates a <code>PuppeTorException</code> with the given detail
- * <code>message</code> and <code>cause</code>.
- *
- * @param message
- * The detail message of this exception.
- * @param cause
- * The cause for this exception.
- */
- public PuppeTorException(final String message, final Throwable cause) {
- super(message, cause);
- }
-
- /**
- * Creates a <code>PuppeTorException</code> with the given detail
- * <code>message</code>, but without a <code>cause</code>.
- *
- * @param message
- * The detail message of this exception.
- */
- public PuppeTorException(final String message) {
- super(message);
- }
-
- /**
- * Creates a <code>PuppeTorException</code> with the given
- * <code>cause</code>, but without a detail message.
- *
- * @param cause
- * The cause for this exception.
- */
- public PuppeTorException(final Throwable cause) {
- super(cause);
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/RouterNode.java b/src/de/uniba/wiai/lspi/puppetor/RouterNode.java
deleted file mode 100644
index 9b6bb5a..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/RouterNode.java
+++ /dev/null
@@ -1,55 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-/**
- * A <code>RouterNode</code> represents a Tor process that is configured to
- * both, relay traffic from a local application to the Tor network and to route
- * traffic on behalf of remote applications. It inherits most of its
- * configuration and behavior from its superclass <code>ProxyNode</code> and
- * adds some router-specific configurations and behavior.
- *
- * @author kloesing
- */
-public interface RouterNode extends ProxyNode {
-
- /**
- * Returns the dir port of this node.
- *
- * @return The dir port of this node.
- */
- public abstract int getDirPort();
-
- /**
- * Returns the onion port of this node.
- *
- * @return The onion port of this node.
- */
- public abstract int getOrPort();
-
- /**
- * <p>
- * Returns the fingerprint string of this node, formatted like
- * <code>nickname 0000 0000 0000 0000 0000 0000 0000 0000 0000 0000</code>.
- * </p>
- *
- * <p>
- * The fingerprint is determined by a background thread that is started as
- * soon as the node is instantiated. If this background thread has not
- * finished when this method is invoked, the invoking thread will be blocked
- * until the fingerprint is available (or determining it has failed,
- * whereupon an exception will be thrown).
- * </p>
- *
- * @return The fingerprint of this node.
- * @throws PuppeTorException
- * Thrown if either the temporary <code>torrc.temp</code>
- * configuration file cannot be written, the Tor process cannot
- * be started temporarily, or the fingerprint file cannot be
- * read.
- */
- public abstract String getFingerprint() throws PuppeTorException;
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/ServerApplication.java b/src/de/uniba/wiai/lspi/puppetor/ServerApplication.java
deleted file mode 100644
index 11cfbb5..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/ServerApplication.java
+++ /dev/null
@@ -1,58 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-/**
- * The <code>ServerApplication</code> can be used as simple HTTP server that
- * answers all <code>HTTP GET</code> requests by empty <code>HTTP OK</code>
- * replies. Therefore, a thread will be started to listen for incoming requests
- * in the background.
- *
- * @author kloesing
- */
-public interface ServerApplication {
-
- /**
- * Starts listening for incoming <code>HTTP GET</code> requests from
- * clients. Any incoming request is answered by an empty
- * <code>HTTP OK</code> reply. This method may only be invoked when the
- * server is currently not in listening state!
- *
- * @throws IllegalStateException
- * Thrown if the server is currently not in listening state.
- */
- public abstract void startListening();
-
- /**
- * Stops listening for requests. This method may only be invoked when the
- * server is currently in listening state!
- *
- * @throws IllegalStateException
- * Thrown if the server is currently in listening state.
- */
- public abstract void stopListening();
-
- /**
- * Returns whether this server is currently in listening state.
- *
- * @return The listening state of this server.
- */
- public abstract boolean isListening();
-
- /**
- * Returns the name of this server.
- *
- * @return The name of this server.
- */
- public abstract String getServerApplicationName();
-
- /**
- * Returns the port on which this server listens.
- *
- * @return The port on which this server listens.
- */
- public abstract int getServerPort();
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/ServerEventType.java b/src/de/uniba/wiai/lspi/puppetor/ServerEventType.java
deleted file mode 100644
index e2dc296..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/ServerEventType.java
+++ /dev/null
@@ -1,41 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor;
-
-/**
- * Event types that can be fired by a server application running as thread in
- * the background.
- */
-@SuppressWarnings("serial")
-public class ServerEventType implements EventType {
-
- /**
- * String identifying the type of the event type.
- */
- String typeString;
-
- /**
- * Creates a new event type with the given type string.
- *
- * @param typeString
- * String identifying the type of the event type.
- */
- public ServerEventType(final String typeString) {
- this.typeString = typeString;
- }
-
- public String getTypeName() {
- return typeString;
- }
-
- /**
- * The server application has received a request and sent a reply to it;
- * this event is fired internally and not parsed from a log statement from
- * Tor.
- */
- public static final ServerEventType SERVER_RECEIVING_REQUEST_SENDING_REPLY =
- new ServerEventType("SERVER_RECEIVING_REQUEST_SENDING_REPLY");
-}
\ No newline at end of file
diff --git a/src/de/uniba/wiai/lspi/puppetor/examples/AccessingPublicWebServerOverTor.java b/src/de/uniba/wiai/lspi/puppetor/examples/AccessingPublicWebServerOverTor.java
deleted file mode 100644
index 15d8ad7..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/examples/AccessingPublicWebServerOverTor.java
+++ /dev/null
@@ -1,119 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.examples;
-
-import de.uniba.wiai.lspi.puppetor.ClientApplication;
-import de.uniba.wiai.lspi.puppetor.ClientEventType;
-import de.uniba.wiai.lspi.puppetor.Event;
-import de.uniba.wiai.lspi.puppetor.EventListener;
-import de.uniba.wiai.lspi.puppetor.EventManager;
-import de.uniba.wiai.lspi.puppetor.Network;
-import de.uniba.wiai.lspi.puppetor.NetworkFactory;
-import de.uniba.wiai.lspi.puppetor.ProxyNode;
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-
-/**
- * Example for accessing a public web server (here: <code>www.google.com</code>)
- * over Tor to measure the access time.
- *
- * @author kloesing
- */
-public class AccessingPublicWebServerOverTor {
-
- /**
- * Sets up and runs the test.
- *
- * @param args
- * Command-line arguments (ignored).
- * @throws PuppeTorException
- * Thrown if there is a problem with the JVM-external Tor
- * processes that we cannot handle.
- */
- public static void main(final String[] args) throws PuppeTorException {
-
- // though we only need a single proxy, we always need to create a
- // network to initialize a test case.
- final Network network = NetworkFactory.createNetwork("example1");
-
- // create a single proxy node with name "proxy"
- final ProxyNode proxy = network.createProxy("proxy");
-
- // write configuration of proxy node
- network.writeConfigurations();
-
- // start proxy node and wait until it has opened a circuit with a
- // timeout of 5 seconds
- if (!network.startNodes(5000)) {
-
- // failed to start the proxy
- System.out.println("Failed to start the node!");
- return;
- }
- System.out.println("Successfully started the node!");
-
- // hup until proxy has built circuits (5 retries, 10 seconds timeout
- // each)
- if (!network.hupUntilUp(5, 10000)) {
-
- // failed to build circuits
- System.out.println("Failed to build circuits!");
- System.exit(0);
- }
- System.out.println("Successfully built circuits!");
-
- // create client application
- final ClientApplication client =
- network.createClient("client", "www.google.com", 80, proxy
- .getSocksPort());
-
- // create event listener to listen for client application events
- final EventListener clientEventListener = new EventListener() {
-
- // remember time when request was sent
- private long before;
-
- public void handleEvent(Event event) {
- if (event.getType() == ClientEventType.CLIENT_SENDING_REQUEST) {
- before = System.currentTimeMillis();
- } else if (event.getType() == ClientEventType.CLIENT_REPLY_RECEIVED) {
- System.out.println("Request took "
- + (System.currentTimeMillis() - before)
- + " milliseconds");
- }
- }
- };
-
- // obtain reference to event manager to be able to respond to events
- final EventManager manager = network.getEventManager();
-
- // register event handler for client application events
- manager.addEventListener(client.getClientApplicationName(),
- clientEventListener);
-
- // perform at most three request with a timeout of 20 seconds each
- client.startRequests(3, 20000, true);
-
- // block this thread as long as client requests are running
- manager.waitForAnyOccurence(client.getClientApplicationName(),
- ClientEventType.CLIENT_REQUESTS_PERFORMED);
-
- // wait a second before shutting down the proxy
- try {
- Thread.sleep(1000);
- } catch (final InterruptedException e) {}
-
- // shut down proxy
- network.shutdownNodes();
-
- // wait another second before exiting the application
- try {
- Thread.sleep(1000);
- } catch (final InterruptedException e) {}
-
- // Shut down the JVM
- System.out.println("Goodbye.");
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/examples/AdvertisingAndAccessingHiddenServiceOverPrivateTorNetwork.java b/src/de/uniba/wiai/lspi/puppetor/examples/AdvertisingAndAccessingHiddenServiceOverPrivateTorNetwork.java
deleted file mode 100644
index ba830d5..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/examples/AdvertisingAndAccessingHiddenServiceOverPrivateTorNetwork.java
+++ /dev/null
@@ -1,132 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.examples;
-
-import de.uniba.wiai.lspi.puppetor.ClientApplication;
-import de.uniba.wiai.lspi.puppetor.ClientEventType;
-import de.uniba.wiai.lspi.puppetor.Event;
-import de.uniba.wiai.lspi.puppetor.EventListener;
-import de.uniba.wiai.lspi.puppetor.EventManager;
-import de.uniba.wiai.lspi.puppetor.HiddenService;
-import de.uniba.wiai.lspi.puppetor.HiddenServiceEventType;
-import de.uniba.wiai.lspi.puppetor.Network;
-import de.uniba.wiai.lspi.puppetor.NetworkFactory;
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-import de.uniba.wiai.lspi.puppetor.RouterNode;
-import de.uniba.wiai.lspi.puppetor.ServerApplication;
-
-/**
- * Example for advertising and accessing a hidden service over a private Tor
- * network.
- *
- * @author kloesing
- */
-public class AdvertisingAndAccessingHiddenServiceOverPrivateTorNetwork {
-
- /**
- * Sets up and runs the test.
- *
- * @param args
- * Command-line arguments (ignored).
- * @throws PuppeTorException
- * Thrown if there is a problem with the JVM-external Tor
- * processes that we cannot handle.
- */
- public static void main(final String[] args) throws PuppeTorException {
-
- // create a network to initialize the test case
- final Network network = NetworkFactory.createNetwork("example4");
-
- // create three router nodes
- final RouterNode router1 = network.createRouter("router1");
- network.createRouter("router2");
- final RouterNode router3 = network.createRouter("router3");
-
- // create only one directory node
- network.createDirectory("dir1");
-
- // add hidden service
- final HiddenService hidServ1 = router1.addHiddenService("hidServ");
-
- // configure nodes of this network to be part of a private network
- network.configureAsPrivateNetwork();
-
- // write node configurations
- network.writeConfigurations();
-
- // start nodes and wait until they have opened a circuit with a timeout
- // of 5 seconds
- if (!network.startNodes(5000)) {
-
- // failed to start the nodes
- System.out.println("Failed to start nodes!");
- System.exit(0);
- }
- System.out.println("Successfully started nodes!");
-
- // hup until nodes have built circuits (60 retries, 10 seconds timeout
- // each)
- if (!network.hupUntilUp(60, 10000)) {
-
- // failed to build circuits
- System.out.println("Failed to build circuits!");
- System.exit(0);
- }
- System.out.println("Successfully built circuits!");
-
- // obtain reference to event manager to be able to respond to events
- final EventManager manager = network.getEventManager();
-
- // wait for 1 hour that the proxy has published its first RSD
- if (!manager.waitForAnyOccurence(router1.getNodeName(),
- HiddenServiceEventType.BOB_DESC_PUBLISHED_RECEIVED,
- 1L * 60L * 60L * 1000L)) {
- // failed to publish an RSD
- System.out.println("Failed to publish an RSD!");
- System.exit(0);
- }
- System.out.println("Successfully published an RSD!");
-
- // create server application
- final ServerApplication server =
- network.createServer("server", hidServ1.getServicePort());
-
- // create client application
- final ClientApplication client =
- network.createClient("client",
- hidServ1.determineOnionAddress(), hidServ1
- .getVirtualPort(), router3.getSocksPort());
-
- // register event listener
- final EventListener clientAndServerEventListener = new EventListener() {
- public void handleEvent(Event event) {
- System.out.println("Handling event: " + event.getMessage());
- }
- };
- manager.addEventListener(client.getClientApplicationName(),
- clientAndServerEventListener);
- manager.addEventListener(server.getServerApplicationName(),
- clientAndServerEventListener);
-
- // start server
- server.startListening();
- System.out.println("Started server");
-
- // perform at most five request with a timeout of 45 seconds each
- client.startRequests(5, 45000, true);
-
- // wait for request to be performed
- manager.waitForAnyOccurence(client.getClientApplicationName(),
- ClientEventType.CLIENT_REQUESTS_PERFORMED);
-
- // shut down nodes
- network.shutdownNodes();
-
- // Shut down the JVM
- System.out.println("Goodbye.");
- System.exit(0);
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/examples/AdvertisingAndAccessingHiddenServiceOverPublicTorNetwork.java b/src/de/uniba/wiai/lspi/puppetor/examples/AdvertisingAndAccessingHiddenServiceOverPublicTorNetwork.java
deleted file mode 100644
index 9ef6774..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/examples/AdvertisingAndAccessingHiddenServiceOverPublicTorNetwork.java
+++ /dev/null
@@ -1,149 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.examples;
-
-import de.uniba.wiai.lspi.puppetor.ClientApplication;
-import de.uniba.wiai.lspi.puppetor.ClientEventType;
-import de.uniba.wiai.lspi.puppetor.Event;
-import de.uniba.wiai.lspi.puppetor.EventListener;
-import de.uniba.wiai.lspi.puppetor.EventManager;
-import de.uniba.wiai.lspi.puppetor.HiddenService;
-import de.uniba.wiai.lspi.puppetor.HiddenServiceEventType;
-import de.uniba.wiai.lspi.puppetor.Network;
-import de.uniba.wiai.lspi.puppetor.NetworkFactory;
-import de.uniba.wiai.lspi.puppetor.ProxyNode;
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-import de.uniba.wiai.lspi.puppetor.ServerApplication;
-import de.uniba.wiai.lspi.puppetor.ServerEventType;
-
-/**
- * Example for advertising and accessing a hidden service over the public Tor
- * network.
- *
- * @author kloesing
- */
-public class AdvertisingAndAccessingHiddenServiceOverPublicTorNetwork {
-
- /**
- * Sets up and runs the test.
- *
- * @param args
- * Command-line arguments (ignored).
- * @throws PuppeTorException
- * Thrown if there is a problem with the JVM-external Tor
- * processes that we cannot handle.
- */
- public static void main(final String[] args) throws PuppeTorException {
-
- // create a network to initialize the test case
- final Network network = NetworkFactory.createNetwork("example3");
-
- // create two proxy nodes
- final ProxyNode proxy1 = network.createProxy("proxy1");
- final ProxyNode proxy2 = network.createProxy("proxy2");
-
- // add hidden service to the configuration of proxy1
- final HiddenService hidServ1 = proxy1.addHiddenService("hidServ");
-
- // write configuration of proxy nodes
- network.writeConfigurations();
-
- // start nodes and wait until they have opened a circuit with a timeout
- // of 5 seconds
- if (!network.startNodes(5000)) {
-
- // failed to start the proxy
- System.out.println("Failed to start nodes!");
- System.exit(0);
- }
- System.out.println("Successfully started nodes!");
-
- // hup until nodes have built circuits (5 retries, 10 seconds timeout
- // each)
- if (!network.hupUntilUp(5, 10000)) {
-
- // failed to build circuits
- System.out.println("Failed to build circuits!");
- System.exit(0);
- }
- System.out.println("Successfully built circuits!");
-
- // obtain reference to event manager to be able to respond to events
- final EventManager manager = network.getEventManager();
-
- // wait for 3 minutes that the proxy has published its first RSD
- if (!manager.waitForAnyOccurence(proxy1.getNodeName(),
- HiddenServiceEventType.BOB_DESC_PUBLISHED_RECEIVED,
- 3L * 60L * 1000L)) {
-
- // failed to publish an RSD
- System.out.println("Failed to publish an RSD!");
- System.exit(0);
- }
- System.out.println("Successfully published an RSD!");
-
- // create server application
- final ServerApplication server =
- network.createServer("server", hidServ1.getServicePort());
-
- // create client application
- final ClientApplication client =
- network.createClient("client",
- hidServ1.determineOnionAddress(), hidServ1
- .getVirtualPort(), proxy2.getSocksPort());
-
- // create event listener to listen for client and server application
- // events
- final EventListener clientAndServerEventListener = new EventListener() {
-
- private long requestReceivedAtServer;
-
- // remember time when request was sent and when it was received
- private long requestSentFromClient;
-
- public void handleEvent(Event event) {
- if (event.getType() == ClientEventType.CLIENT_SENDING_REQUEST) {
- requestSentFromClient = event.getOccurrenceTime();
- } else if (event.getType() == ServerEventType.SERVER_RECEIVING_REQUEST_SENDING_REPLY) {
- requestReceivedAtServer = event.getOccurrenceTime();
- System.out.println("Request took "
- + (requestReceivedAtServer - requestSentFromClient)
- + " milliseconds from client to server!");
- } else if (event.getType() == ClientEventType.CLIENT_REPLY_RECEIVED) {
- System.out
- .println("Request took "
- + (event.getOccurrenceTime() - requestSentFromClient)
- + " milliseconds for the round-trip and "
- + (event.getOccurrenceTime() - requestReceivedAtServer)
- + " milliseconds from server to client!");
- }
- }
- };
-
- // register event handler for client and server application events
- manager.addEventListener(client.getClientApplicationName(),
- clientAndServerEventListener);
- manager.addEventListener(server.getServerApplicationName(),
- clientAndServerEventListener);
-
- // start server
- server.startListening();
-
- // perform at most five request with a timeout of 45 seconds each
- client.startRequests(5, 45000, true);
-
- // block this thread as long as client requests are running
- manager.waitForAnyOccurence(client.getClientApplicationName(),
- ClientEventType.CLIENT_REQUESTS_PERFORMED);
-
- // shut down proxy
- network.shutdownNodes();
-
- // Shut down the JVM
- System.out.println("Goodbye.");
- System.exit(0);
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/examples/AdvertisingHiddenServiceToPublicTorNetwork.java b/src/de/uniba/wiai/lspi/puppetor/examples/AdvertisingHiddenServiceToPublicTorNetwork.java
deleted file mode 100644
index 1521487..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/examples/AdvertisingHiddenServiceToPublicTorNetwork.java
+++ /dev/null
@@ -1,110 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.examples;
-
-import de.uniba.wiai.lspi.puppetor.Event;
-import de.uniba.wiai.lspi.puppetor.EventListener;
-import de.uniba.wiai.lspi.puppetor.EventManager;
-import de.uniba.wiai.lspi.puppetor.HiddenServiceEventType;
-import de.uniba.wiai.lspi.puppetor.Network;
-import de.uniba.wiai.lspi.puppetor.NetworkFactory;
-import de.uniba.wiai.lspi.puppetor.NodeEventType;
-import de.uniba.wiai.lspi.puppetor.ProxyNode;
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-
-/**
- * Example for advertising a hidden service to the public Tor network and
- * observing the publication of rendezvous service descriptors.
- *
- * @author kloesing
- */
-public class AdvertisingHiddenServiceToPublicTorNetwork {
-
- /**
- * Sets up and runs the test.
- *
- * @param args
- * Command-line arguments (ignored).
- * @throws PuppeTorException
- * Thrown if there is a problem with the JVM-external Tor
- * processes that we cannot handle.
- */
- public static void main(final String[] args) throws PuppeTorException {
- // create a network to initialize the test case
- final Network network = NetworkFactory.createNetwork("example2");
-
- // create a single proxy node
- final ProxyNode proxy = network.createProxy("proxy");
-
- // add hidden service to the configuration
- proxy.addHiddenService("hidServ");
-
- // write configuration of proxy node
- network.writeConfigurations();
-
- // create event listener to listen for events from our proxy
- final EventListener proxyEventListener = new EventListener() {
-
- // remember time when request was sent
- private long circuitOpened = -1;
-
- public void handleEvent(Event event) {
- if (event.getType() == NodeEventType.NODE_CIRCUIT_OPENED) {
- if (circuitOpened == -1) {
- circuitOpened = System.currentTimeMillis();
- }
- } else if (event.getType() == HiddenServiceEventType.BOB_DESC_PUBLISHED_RECEIVED) {
- System.out.println("RSD published "
- + (System.currentTimeMillis() - circuitOpened)
- + " milliseconds after first circuit was opened");
- }
- }
- };
-
- // obtain reference to event manager to be able to respond to events
- final EventManager manager = network.getEventManager();
-
- // register event handler for proxy events
- manager.addEventListener(proxy.getNodeName(), proxyEventListener);
-
- // start proxy node and wait until it has opened a circuit with a
- // timeout of 5 seconds
- if (!network.startNodes(5000)) {
-
- // failed to start the proxy
- System.out.println("Failed to start the node!");
- System.exit(0);
- }
- System.out.println("Successfully started the node!");
-
- // hup until proxy has built circuits (5 retries, 10 seconds timeout
- // each)
- if (!network.hupUntilUp(5, 10000)) {
-
- // failed to build circuits
- System.out.println("Failed to build circuits!");
- System.exit(0);
- }
- System.out.println("Successfully built circuits!");
-
- // let it run for 2 minutes and observe when RSDs are published...
- System.out
- .println("Waiting for 2 minutes and observing RSD publications...");
-
- try {
- Thread.sleep(2L * 60L * 1000L);
- } catch (final InterruptedException e) {
- // do nothing
- }
-
- // shut down proxy
- network.shutdownNodes();
-
- // Shut down the JVM
- System.out.println("Goodbye.");
- System.exit(0);
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/impl/ClientApplicationImpl.java b/src/de/uniba/wiai/lspi/puppetor/impl/ClientApplicationImpl.java
deleted file mode 100644
index 9dfe014..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/impl/ClientApplicationImpl.java
+++ /dev/null
@@ -1,457 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.impl;
-
-import java.io.BufferedReader;
-import java.io.IOException;
-import java.io.InputStreamReader;
-import java.io.PrintStream;
-import java.net.InetSocketAddress;
-import java.net.Proxy;
-import java.net.Socket;
-import java.net.SocketTimeoutException;
-import java.net.Proxy.Type;
-import java.util.logging.Level;
-import java.util.logging.Logger;
-
-import de.uniba.wiai.lspi.puppetor.ClientApplication;
-import de.uniba.wiai.lspi.puppetor.ClientEventType;
-
-/**
- * Implementation of <code>ClientApplication</code>.
- *
- * @author kloesing
- */
-public class ClientApplicationImpl implements ClientApplication {
-
- /**
- * Internal thread class that is used to perform requests.
- */
- private class RequestThread extends Thread {
-
- /**
- * Flag to remember whether requests are performed at the moment (<code>true</code>),
- * or have been stopped (<code>false</code>).
- */
- private boolean connected;
-
- /**
- * Number of retries to be performed.
- */
- private final int retries;
-
- /**
- * Flag that determines whether requests shall be stopped after the
- * first successful reply (<code>true</code>), or not (<code>false</code>).
- */
- private final boolean stopOnSuccess;
-
- /**
- * Timeout in milliseconds for each retry.
- */
- private final long timeoutForEachRetry;
-
- /**
- * Creates a new thread, but does not start performing requests, yet.
- *
- * @param retries
- * Number of retries to be performed.
- * @param timeoutForEachRetry
- * Timeout in milliseconds for each retry.
- * @param stopOnSuccess
- * Flag that determines whether requests shall be stopped
- * after the first successful reply (<code>true</code>),
- * or not (<code>false</code>).
- */
- RequestThread(final int retries, final long timeoutForEachRetry,
- final boolean stopOnSuccess) {
-
- // log entering
- logger
- .entering(this.getClass().getName(), "RequestThread",
- new Object[] { retries, timeoutForEachRetry,
- stopOnSuccess });
-
- // check parameters
- if (retries < 0 || timeoutForEachRetry < 0) {
- final IllegalArgumentException e =
- new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "RequestThread", e);
- throw e;
- }
-
- // remember parameters
- this.retries = retries;
- this.timeoutForEachRetry = timeoutForEachRetry;
- this.stopOnSuccess = stopOnSuccess;
-
- // start connected
- connected = true;
-
- // log exiting
- logger.exiting(this.getClass().getName(), "RequestThread");
- }
-
- /**
- * Perform one or more requests.
- */
- @Override
- public void run() {
-
- // log entering
- logger.entering(this.getClass().getName(), "run");
-
- try {
-
- // set Tor as proxy
- final InetSocketAddress isa =
- new InetSocketAddress("127.0.0.1", socksPort);
- final Proxy p = new Proxy(Type.SOCKS, isa);
-
- // create target address for socket -- don't resolve the target
- // name to an IP address!
- final InetSocketAddress hs =
- InetSocketAddress.createUnresolved(targetName,
- targetPort);
-
- // start retry loop
- for (int i = 0; connected && i < retries; i++) {
-
- // log this try
- logger.log(Level.FINE, "Trying to perform request");
-
- // remember when we started
- final long timeBeforeConnectionAttempt =
- System.currentTimeMillis();
-
- // send event to event manager
- eventManager.observeInternalEvent(
- timeBeforeConnectionAttempt,
- getClientApplicationName(),
- ClientEventType.CLIENT_SENDING_REQUEST,
- "Sending request.");
-
- Socket s = null;
- try {
-
- // create new socket using Tor as proxy
- s = new Socket(p);
-
- // try to connect to remote server
- s.connect(hs, (int) timeoutForEachRetry);
-
- // open output stream to write request
- final PrintStream out =
- new PrintStream(s.getOutputStream());
- out.print("GET / HTTP/1.0\r\n\r\n");
-
- // open input stream to read reply
- final BufferedReader in =
- new BufferedReader(new InputStreamReader(s
- .getInputStream()));
-
- // only read the first char in the response; this method
- // blocks until there is a response
- in.read();
-
- // send event to event manager
- eventManager.observeInternalEvent(System
- .currentTimeMillis(),
- getClientApplicationName(),
- ClientEventType.CLIENT_REPLY_RECEIVED,
- "Received response.");
-
- // if we should stop on success, stop further connection
- // attempts
- if (stopOnSuccess) {
- connected = false;
- }
-
- // clean up socket
- in.close();
- out.close();
- s.close();
-
- } catch (final SocketTimeoutException e) {
-
- // log warning
- logger.log(Level.WARNING,
- "Connection to remote server timed out!", e);
-
- // send event to event manager
- eventManager.observeInternalEvent(System
- .currentTimeMillis(),
- getClientApplicationName(),
- ClientEventType.CLIENT_GAVE_UP_REQUEST,
- "Giving up request.");
-
- // try again immediately, if there are retries left
-
- } catch (final IOException e) {
-
- // log warning
- logger.log(Level.WARNING,
- "Connection to remote server could not be "
- + "established!", e);
-
- // send event to event manager
- eventManager.observeInternalEvent(System
- .currentTimeMillis(),
- getClientApplicationName(),
- ClientEventType.CLIENT_GAVE_UP_REQUEST,
- "Giving up request.");
-
- // wait for the rest of the timeout
- final long timeOfTimeoutLeft =
- timeBeforeConnectionAttempt
- + timeoutForEachRetry
- - System.currentTimeMillis();
- if (timeOfTimeoutLeft > 0) {
- try {
- Thread.sleep(timeOfTimeoutLeft);
- } catch (final InterruptedException ex) {
- // do nothing
- }
- }
-
- } finally {
- if (s != null) {
- // close connection
- try {
-
- // try to close socket
- logger.log(Level.FINER,
- "Trying to close socket.");
- s.close();
- logger.log(Level.FINE, "Socket closed.");
-
- } catch (final IOException e) {
-
- // log warning
- logger.log(Level.WARNING,
- "Exception when trying to close socket!",
- e);
- }
- }
- }
- }
-
- } finally {
-
- // we are done here
- logger.log(Level.FINE, "Requests performed!");
-
- // send event to event manager
- eventManager.observeInternalEvent(System.currentTimeMillis(),
- getClientApplicationName(),
- ClientEventType.CLIENT_REQUESTS_PERFORMED,
- "Requests performed.");
-
- // log exiting
- logger.exiting(this.getClass().getName(), "run");
- }
- }
-
- /**
- * Immediately stops this and all possibly subsequent requests.
- */
- public void stopRequest() {
-
- // log entering
- logger.entering(this.getClass().getName(), "stopRequest");
-
- // change connected state to false and interrupt thread
- connected = false;
- interrupt();
-
- // log exiting
- logger.exiting(this.getClass().getName(), "stopRequest");
- }
- }
-
- /**
- * Name of this client application that is used as logger name of this node.
- */
- private final String clientApplicationName;
-
- /**
- * Thread that performs the requests in the background.
- */
- private RequestThread clientThread;
-
- /**
- * Event manager that handles all events concerning this client application.
- */
- private final EventManagerImpl eventManager;
-
- /**
- * Logger for this client which is called "client." plus the name of this
- * client application.
- */
- private final Logger logger;
-
- /**
- * SOCKS port of the local Tor node to which requests are sent.
- */
- private final int socksPort;
-
- /**
- * Target name for the requests sent by this client; can be a publicly
- * available URL or an onion address.
- */
- private final String targetName;
-
- /**
- * Target port for the requests sent by this client; can be either a server
- * port or a virtual port of a hidden service.
- */
- private final int targetPort;
-
- /**
- * Creates a new HTTP client within this JVM, but does not start sending
- * requests.
- *
- * @param network
- * Network to which this HTTP client belongs; at the moment this
- * is only used to determine the event manager instance.
- * @param clientApplicationName
- * Name of this client that is used as part of the logger name.
- * @param targetName
- * Target name for requests; can be either a server name/address
- * or an onion address.
- * @param targetPort
- * Target port for requests; can be either a server port or a
- * virtual port of a hidden service.
- * @param socksPort
- * SOCKS port of the local Tor node.
- * @throws IllegalArgumentException
- * If at least one of the parameters is <code>null</code> or
- * has an invalid value.
- */
- ClientApplicationImpl(final NetworkImpl network,
- final String clientApplicationName, final String targetName,
- final int targetPort, final int socksPort) {
-
- // check if clientApplicationName can be used as logger name
- if (clientApplicationName == null
- || clientApplicationName.length() == 0) {
- throw new IllegalArgumentException(
- "Invalid clientApplicationName: " + clientApplicationName);
- }
-
- // create logger
- logger = Logger.getLogger("client." + clientApplicationName);
-
- // log entering
- logger.entering(this.getClass().getName(), "ClientApplicationImpl",
- new Object[] { network, clientApplicationName, targetName,
- targetPort, socksPort });
-
- // check parameters
- if (network == null || targetName == null || targetName.length() == 0
- || targetPort < 0 || targetPort > 65535 || socksPort < 0
- || socksPort > 65535) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "ClientApplicationImpl",
- e);
- throw e;
- }
-
- // remember parameters
- this.clientApplicationName = clientApplicationName;
- this.targetName = targetName;
- this.targetPort = targetPort;
- this.socksPort = socksPort;
-
- // obtain and store reference on event manager
- eventManager = network.getEventManagerImpl();
-
- // log exiting
- logger.exiting(this.getClass().getName(), "ClientApplicationImpl");
- }
-
- public synchronized void startRequests(final int retries,
- final long timeoutForEachRetry, final boolean stopOnSuccess) {
-
- // log entering
- logger.entering(this.getClass().getName(), "performRequest",
- new Object[] { retries, timeoutForEachRetry, stopOnSuccess });
-
- // check parameters
- if (retries <= 0 || timeoutForEachRetry < 0) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "performRequest", e);
- throw e;
- }
-
- // check if we already have started a request (TODO change this to allow
- // multiple requests in parallel? would be possible)
- if (clientThread != null) {
- final IllegalStateException e =
- new IllegalStateException(
- "Another request has already been started!");
- logger.throwing(this.getClass().getName(), "performRequest", e);
- throw e;
- }
-
- // create a thread that performs requests in the background
- clientThread =
- new RequestThread(retries, timeoutForEachRetry, stopOnSuccess);
- clientThread.setName("Request Thread");
- clientThread.setDaemon(true);
- clientThread.start();
-
- // log exiting
- logger.exiting(this.getClass().getName(), "performRequest");
- }
-
- public synchronized void stopRequest() {
-
- // log entering
- logger.entering(this.getClass().getName(), "stopRequest");
-
- // check if a request is running
- if (clientThread == null) {
- final IllegalStateException e =
- new IllegalStateException("Cannot stop "
- + "request, because no request has been started!");
- logger.throwing(this.getClass().getName(), "stopRequest", e);
- throw e;
- }
-
- // log this event
- logger.log(Level.FINE, "Shutting down client");
-
- // interrupt thread
- clientThread.stopRequest();
-
- // log exiting
- logger.exiting(this.getClass().getName(), "stopRequest");
- }
-
- @Override
- public String toString() {
- return this.getClass().getSimpleName() + ": clientApplicationName=\""
- + clientApplicationName + "\", targetAddress=\"" + targetName
- + "\", targetPort=" + targetPort + ", socksPort=" + socksPort;
- }
-
- public String getClientApplicationName() {
- return clientApplicationName;
- }
-
- public int getSocksPort() {
- return socksPort;
- }
-
- public String getTargetName() {
- return targetName;
- }
-
- public int getTargetPort() {
- return targetPort;
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/impl/DirectoryNodeImpl.java b/src/de/uniba/wiai/lspi/puppetor/impl/DirectoryNodeImpl.java
deleted file mode 100644
index 2dd82ea..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/impl/DirectoryNodeImpl.java
+++ /dev/null
@@ -1,431 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.impl;
-
-import java.io.BufferedReader;
-import java.io.BufferedWriter;
-import java.io.File;
-import java.io.FileReader;
-import java.io.FileWriter;
-import java.io.IOException;
-import java.io.InputStream;
-import java.io.InputStreamReader;
-import java.io.OutputStreamWriter;
-import java.util.ArrayList;
-import java.util.List;
-import java.util.Set;
-import java.util.SortedSet;
-import java.util.TreeSet;
-import java.util.logging.Level;
-
-import de.uniba.wiai.lspi.puppetor.DirectoryNode;
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-
-/**
- * Implementation of <code>DirectoryNode</code>.
- *
- * @author kloesing
- */
-public class DirectoryNodeImpl extends RouterNodeImpl implements DirectoryNode {
-
- /**
- * Executable file for generating v3 directory authority certificates.
- *
- * TODO make this configurable!
- */
- protected static final File torGencertExecutable = new File("tor-gencert");
-
- /**
- * Internal thread class that is used to generate v3 directory authority
- * certificates in parallel, which can take a few seconds.
- */
- private class GenerateCertificateThread extends Thread {
-
- @Override
- public void run() {
-
- // log entering
- logger.entering(this.getClass().getName(), "run");
-
- // run tor-gencert
- final ProcessBuilder processBuilder =
- new ProcessBuilder(torGencertExecutable.getPath(),
- "--create-identity-key"// );
- , "--passphrase-fd", "0");
- final File workingDirectory =
- new File(workingDir.getAbsolutePath() + File.separator
- + "keys" + File.separator);
-
- // create working directory
- workingDirectory.mkdirs();
-
- processBuilder.directory(workingDirectory);
- processBuilder.redirectErrorStream(true);
- Process tmpProcess = null;
- try {
- tmpProcess = processBuilder.start();
- } catch (final IOException e) {
- final PuppeTorException ex =
- new PuppeTorException(
- "Could not start tor-gencert process for generating a "
- + "v3 directory authority certificate!",
- e);
- logger.log(Level.WARNING, "Could not start tor-gencert "
- + "process for generating a v3 directory authority "
- + "certificate!", ex);
- setCaughtException(ex);
- return;
- }
-
- final BufferedWriter writer =
- new BufferedWriter(new OutputStreamWriter(tmpProcess
- .getOutputStream()));
- try {
- writer.write("somepassword\n");
- writer.close();
- } catch (final IOException e1) {
- System.out.println("Exception at write! " + e1.getMessage());
- e1.printStackTrace();
- }
- // final InputStream read = tmpProcess.getErrorStream();
-
- final InputStream stderr = tmpProcess.getInputStream();
- final InputStreamReader isr = new InputStreamReader(stderr);
- final BufferedReader br = new BufferedReader(isr);
- String line = null;
- try {
- while ((line = br.readLine()) != null) {
- ;
- }
- } catch (final IOException e1) {
- e1.printStackTrace();
- }
-
- // wait for process to terminate
- int exitValue = 0;
- try {
- exitValue = tmpProcess.waitFor();
- } catch (final InterruptedException e) {
- final PuppeTorException ex =
- new PuppeTorException(
- "Interrupted while waiting for tor-gencert process to exit!",
- e);
- logger.log(Level.WARNING,
- "tor-gencert process was interrupted!", ex);
- setCaughtException(ex);
- return;
- }
-
- if (exitValue != 0) {
- final PuppeTorException ex =
- new PuppeTorException(
- "Could not start tor-gencert process! tor-gencert exited with "
- + "exit value " + exitValue + "!");
- logger.log(Level.WARNING,
- "Could not start tor-gencert process!", ex);
- setCaughtException(ex);
- return;
- }
-
- // read fingerprint from file
- final File authorityCertificateFile =
- new File(workingDirectory.getAbsolutePath()
- + File.separator + "authority_certificate");
- String identity;
- try {
- final BufferedReader br2 =
- new BufferedReader(new FileReader(
- authorityCertificateFile));
- while ((line = br2.readLine()) != null
- && !line.startsWith("fingerprint ")) {
- ;
- }
- if (line == null) {
- final PuppeTorException ex =
- new PuppeTorException(
- "Could not find fingerprint line in file "
- + "authority_certificate!");
- logger.log(Level.WARNING,
- "Could not find fingerprint line in file "
- + "authority_certificate!", ex);
- setCaughtException(ex);
- return;
- }
- identity = line.substring(line.indexOf(" ") + 1);
- br2.close();
- } catch (final IOException e) {
- final PuppeTorException ex =
- new PuppeTorException(
- "Could not read fingerprint from file!", e);
- logger.log(Level.WARNING, "Could not read fingerprint file!",
- ex);
- setCaughtException(ex);
- return;
- }
-
- setV3Identity(identity);
-
- // log exiting
- logger.exiting(this.getClass().getName(), "run");
- }
- }
-
- /**
- * Set of routers that are approved by this directory node.
- */
- private final SortedSet<String> approvedRouters;
-
- /**
- * Creates a <code>DirectoryNodeImpl</code> and adds it to the given
- * <code>network</code>, but does not yet write its configuration to disk
- * or start the corresponding Tor process.
- *
- * @param network
- * Network configuration to which this node belongs.
- * @param nodeName
- * The name of the new node which may only consist of between 1
- * and 19 alpha-numeric characters.
- * @param controlPort
- * Port on which the Tor node will be listening for us as its
- * controller. May not be negative or greater than 65535.
- * @param socksPort
- * Port on which the Tor node will be listening for SOCKS
- * connection requests. May not be negative or greater than
- * 65535.
- * @param orPort
- * Port on which the Tor node will be listening for onion
- * requests by other Tor nodes. May not be negative or greater
- * than 65535.
- * @param dirPort
- * Port on which the Tor node will be listening for directory
- * requests from other Tor nodes. May not be negative or greater
- * than 65535.
- * @param serverIpAddress
- * The IP address on which the node will listen. Must be a valid
- * IP v4 address in dotted decimal notation. May not be
- * <code>null</code>.
- * @throws IllegalArgumentException
- * If at least one of the parameters is <code>null</code> or
- * has an invalid value.
- * @throws PuppeTorException
- * Thrown if an I/O problem occurs while writing the temporary
- * <code>approved-routers</code> file.
- */
- DirectoryNodeImpl(final NetworkImpl network, final String nodeName,
- final int controlPort, final int socksPort, final int orPort,
- final int dirPort, final String serverIpAddress) {
- // create superclass instance; parameter checking is done in super
- // constructor
- super(network, nodeName, controlPort, socksPort, orPort, dirPort,
- serverIpAddress);
-
- // log entering
- logger.entering(this.getClass().getName(), "DirectoryNodeImpl",
- new Object[] { network, nodeName, controlPort, socksPort,
- orPort, dirPort });
-
- // initialize attribute
- approvedRouters = new TreeSet<String>();
-
- // extend configuration by template configuration of directory nodes
- configuration.addAll(templateConfiguration);
-
- // log exiting
- logger.exiting(this.getClass().getName(), "DirectoryNodeImpl");
- }
-
- /**
- * Invoked by the certificate generating thread: sets the generated v3
- * identity string.
- *
- * @param v3Identity
- * The generated v3 identity string.
- */
- private synchronized void setV3Identity(final String v3Identity) {
- // log entering
- logger.entering(this.getClass().getName(), "setV3Identity", v3Identity);
-
- // remember fingerprint and notify all waiting threads
- this.v3Identity = v3Identity;
- notifyAll();
-
- // log exiting
- logger.exiting(this.getClass().getName(), "setV3Identity");
- }
-
- /**
- * The generated v3 identity string.
- */
- private String v3Identity;
-
- public synchronized String getV3Identity() throws PuppeTorException {
-
- // log entering
- logger.entering(this.getClass().getName(), "getV3Identity");
-
- // wait until either the v3 identity has been determined or an exception
- // was caught
- while (v3Identity == null && caughtException == null) {
-
- try {
- wait(500);
- } catch (final InterruptedException e) {
- // do nothing
- }
- }
-
- if (caughtException != null) {
- logger.throwing(this.getClass().getName(), "getV3Identity",
- caughtException);
- throw caughtException;
- }
-
- // log exiting
- logger.exiting(this.getClass().getName(), "getV3Identity", v3Identity);
- return v3Identity;
- }
-
- public synchronized String getDirServerString() throws PuppeTorException {
-
- // log entering
- logger.entering(this.getClass().getName(), "getDirServerString");
-
- // determine fingerprint
- String fingerprint = getFingerprint();
-
- // cut off router nickname
- fingerprint = fingerprint.substring(fingerprint.indexOf(" ") + 1);
-
- // determine v3 identity
- final String determinedV3Identity = getV3Identity();
-
- // put everything together
- final String dirServerString =
- "DirServer " + nodeName + " v3ident=" + determinedV3Identity
- + " orport=" + orPort + " " + serverIpAddress + ":"
- + dirPort + " " + fingerprint;
-
- // log exiting and return dir server string
- logger.exiting(this.getClass().getName(), "getDirServerString",
- dirServerString);
- return dirServerString;
- }
-
- public void addApprovedRouters(final Set<String> routers) {
-
- // log entering
- logger.entering(this.getClass().getName(), "addApprovedRouters",
- routers);
-
- // check parameter
- if (routers == null) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "addApprovedRouters", e);
- throw e;
- }
-
- // add the given approved router strings to the sorted set of already
- // known strings (if any)
- approvedRouters.addAll(routers);
-
- // log exiting
- logger.exiting(this.getClass().getName(), "addApprovedRouters");
- }
-
- @Override
- protected synchronized void determineFingerprint() {
-
- // log entering
- logger.entering(this.getClass().getName(), "determineFingerprint");
-
- // start a thread to generate the directory's certificate
- final GenerateCertificateThread certificateThread =
- new GenerateCertificateThread();
- certificateThread.setName(nodeName + " Certificate Generator");
- certificateThread.start();
-
- // wait (non-blocking) for the v3 identity string
- try {
- getV3Identity();
- } catch (final PuppeTorException e1) {
- final PuppeTorException ex =
- new PuppeTorException("Could not read v3 identity string!",
- e1);
- caughtException = ex;
- return;
- }
-
- // create an empty approved-routers file to make Tor happy
- try {
- new File(workingDir.getAbsolutePath() + File.separator
- + "approved-routers").createNewFile();
- } catch (final IOException e) {
- final PuppeTorException ex =
- new PuppeTorException(
- "Could not write empty approved-routers file!", e);
- caughtException = ex;
- return;
- }
-
- // invoke overwritten method
- super.determineFingerprint();
- }
-
- @Override
- public synchronized void writeConfiguration() throws PuppeTorException {
-
- // log entering
- logger.entering(this.getClass().getName(), "writeConfiguration");
-
- // write approved-routers file
- try {
- final File approvedRoutersFile =
- new File(workingDir.getAbsolutePath() + File.separator
- + "approved-routers");
- final BufferedWriter bw =
- new BufferedWriter(new FileWriter(approvedRoutersFile));
- for (final String approvedRouter : approvedRouters) {
- bw.write(approvedRouter + "\n");
- }
- bw.close();
- } catch (final IOException e) {
- final PuppeTorException ex = new PuppeTorException(e);
- logger
- .throwing(this.getClass().getName(), "writeConfiguration",
- ex);
- throw ex;
- }
-
- // invoke overridden method
- super.writeConfiguration();
-
- // log exiting
- logger.exiting(this.getClass().getName(), "writeConfiguration");
- }
-
- /**
- * Template configuration of directory nodes.
- */
- static List<String> templateConfiguration;
-
- static {
- templateConfiguration = new ArrayList<String>();
-
- // configure this node as an authoritative directory
- templateConfiguration.add("AuthoritativeDirectory 1");
- templateConfiguration.add("V2AuthoritativeDirectory 1");
- templateConfiguration.add("V3AuthoritativeDirectory 1");
- templateConfiguration.add("DirAllowPrivateAddresses 1");
- templateConfiguration.add("MinUptimeHidServDirectoryV2 0 minutes");
-
- // TODO This is now contained in proposal 135.
- // templateConfiguration.add("AuthDirMaxServersPerAddr 0");
- // templateConfiguration.add("AuthDirMaxServersPerAuthAddr 0");
- // templateConfiguration.add("V3AuthVotingInterval 5 minutes");
- // templateConfiguration.add("V3AuthVoteDelay 20 seconds");
- // templateConfiguration.add("V3AuthDistDelay 20 seconds");
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/impl/EventImpl.java b/src/de/uniba/wiai/lspi/puppetor/impl/EventImpl.java
deleted file mode 100644
index 52384fd..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/impl/EventImpl.java
+++ /dev/null
@@ -1,122 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.impl;
-
-import java.util.Date;
-
-import de.uniba.wiai.lspi.puppetor.Event;
-import de.uniba.wiai.lspi.puppetor.EventType;
-
-/**
- * Implementation of <code>Event</code>.
- *
- * @author kloesing
- */
-@SuppressWarnings("serial")
-public class EventImpl implements Event {
-
- /**
- * The source of this event.
- */
- private final String source;
-
- /**
- * The type of this event.
- */
- private EventType type;
-
- /**
- * Either the log message that led to firing this event, or an internal
- * message.
- */
- private final String message;
-
- /**
- * The occurrence time of the event or of the corresponding log statement.
- */
- private long occurrenceTime;
-
- /**
- * Creates a new <code>EventImpl</code>.
- *
- * @param occurrenceTime
- * The occurrence time of the event or of the corresponding log
- * statement.
- * @param source
- * The source of this event.
- * @param type
- * The type of this event.
- * @param message
- * Either the log message that led to firing this event, or an
- * internal message.
- */
- EventImpl(final long occurrenceTime, final String source,
- final EventType type, final String message) {
- this.occurrenceTime = occurrenceTime;
- this.source = source;
- this.type = type;
- this.message = message;
- }
-
- /**
- * Creates a new <code>EventImpl</code>.
- *
- * @param source
- * The source of this event.
- * @param message
- * Either the log message that led to firing this event, or an
- * internal message.
- */
- EventImpl(final String source, final String message) {
- this.source = source;
- this.message = message;
- }
-
- public String getSource() {
- return source;
- }
-
- public EventType getType() {
- return type;
- }
-
- /**
- * Sets the event type to the given type.
- *
- * @param type
- * The type of this event.
- */
- void setType(final EventType type) {
- this.type = type;
- }
-
- public String getMessage() {
- return message;
- }
-
- public long getOccurrenceTime() {
- return occurrenceTime;
- }
-
- @Override
- public String toString() {
- return this.getClass().getSimpleName() + ": occurenceTime="
- + new Date(occurrenceTime) + ", source=\"" + source
- + "\", type=" + type.getTypeName() + ", message=\"" + message
- + "\"";
- }
-
- /**
- * Sets the occurrence time to the given time.
- *
- * @param occurrenceTime
- * The occurrence time of the event or of the corresponding log
- * statement.
- */
- void setOccurenceTime(final long occurrenceTime) {
- this.occurrenceTime = occurrenceTime;
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/impl/EventManagerImpl.java b/src/de/uniba/wiai/lspi/puppetor/impl/EventManagerImpl.java
deleted file mode 100644
index 788d61f..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/impl/EventManagerImpl.java
+++ /dev/null
@@ -1,729 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.impl;
-
-import java.text.ParsePosition;
-import java.text.SimpleDateFormat;
-import java.util.ArrayList;
-import java.util.Calendar;
-import java.util.Date;
-import java.util.HashMap;
-import java.util.HashSet;
-import java.util.LinkedList;
-import java.util.List;
-import java.util.Locale;
-import java.util.Map;
-import java.util.Set;
-import java.util.Map.Entry;
-import java.util.logging.Level;
-import java.util.logging.Logger;
-import java.util.regex.Matcher;
-import java.util.regex.Pattern;
-
-import de.uniba.wiai.lspi.puppetor.Event;
-import de.uniba.wiai.lspi.puppetor.EventListener;
-import de.uniba.wiai.lspi.puppetor.EventManager;
-import de.uniba.wiai.lspi.puppetor.EventType;
-import de.uniba.wiai.lspi.puppetor.HiddenServiceEventType;
-import de.uniba.wiai.lspi.puppetor.NodeEventType;
-
-/**
- * Implementation of <code>EventManager</code>.
- *
- * @author kloesing
- */
-public class EventManagerImpl implements EventManager {
-
- /**
- * Registered event handlers for specific sources.
- */
- private final Map<String, Set<EventListener>> eventHandlers;
-
- /**
- * Registered event handlers for all sources.
- */
- private final Set<EventListener> eventHandlersForAllSources;
-
- /**
- * Logger for this event manager which is called "event." plus the name of
- * the network.
- */
- private final Logger logger;
-
- /**
- * Events observed so far.
- */
- private final Map<String, List<Event>> observedEvents;
-
- /**
- * Set of all registered event sources. This is required to ensure that
- * requests for events from a given source specify valid event sources.
- */
- private final Set<String> eventSources;
-
- /**
- * Creates a new <code>EventManagerImpl</code> for the network with name
- * <code>networkName</code> and initializes it.
- *
- * @param networkName
- * Name of this event manager that is used as part of the logger
- * name.
- * @throws IllegalArgumentException
- * Thrown if the given <code>networkName</code> is either
- * <code>null</code> or a zero-length string.
- */
- EventManagerImpl(final String networkName) {
-
- // check if networkName can be used as logger name
- if (networkName == null || networkName.length() == 0) {
- throw new IllegalArgumentException("Invalid networkName: "
- + networkName);
- }
-
- // create logger
- logger = Logger.getLogger("event." + networkName);
-
- // log entering
- logger.entering(this.getClass().getName(), "EventManagerImpl",
- networkName);
-
- // create data structures
- observedEvents = new HashMap<String, List<Event>>();
- eventHandlers = new HashMap<String, Set<EventListener>>();
- eventHandlersForAllSources = new HashSet<EventListener>();
- eventSources = new HashSet<String>();
-
- // start thread to parse events
- final Thread eventParseThread = new Thread() {
- @Override
- public void run() {
- while (true) {
- parseNextEvent();
- }
- }
- };
- eventParseThread.setDaemon(true);
- eventParseThread.start();
-
- // initialize event type patterns
- initializeEventTypePatterns();
-
- // log exiting
- logger.exiting(this.getClass().getName(), "EventManagerImpl");
- }
-
- public synchronized List<Event> addEventListener(final String source,
- final EventListener listener) {
-
- // log entering
- logger.entering(this.getClass().getName(), "addEventListener",
- new Object[] { source, listener });
-
- // check parameters
- if (source == null || listener == null
- || !eventSources.contains(source)) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "addEventListener", e);
- throw e;
- }
-
- // if necessary, create new event listener set for source
- if (!eventHandlers.containsKey(source)) {
- eventHandlers.put(source, new HashSet<EventListener>());
- }
-
- // add listener
- eventHandlers.get(source).add(listener);
-
- // log change
- logger.log(Level.FINE, "Added event listener for source " + source);
-
- // log exiting and return
- final List<Event> result = getEventHistory(source);
- logger.exiting(this.getClass().getName(), "addEventListener", result);
- return result;
- }
-
- public synchronized void addEventListener(final EventListener listener) {
-
- // log entering
- logger.entering(this.getClass().getName(), "addEventListener",
- new Object[] { listener });
-
- // check parameters
- if (listener == null) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "addEventListener", e);
- throw e;
- }
-
- // add listener
- eventHandlersForAllSources.add(listener);
-
- // log change
- logger.log(Level.FINE, "Added event listener for all sources.");
-
- // log exiting and return
- logger.exiting(this.getClass().getName(), "addEventListener");
- return;
- }
-
- public synchronized List<Event> getEventHistory(final String source) {
-
- // log entering
- logger.entering(this.getClass().getName(), "getEventHistory", source);
-
- // check parameter
- if (source == null || !eventSources.contains(source)) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "getEventHistory", e);
- throw e;
- }
-
- // prepare result
- final List<Event> result = new ArrayList<Event>();
-
- // did we already observe events for this source?
- if (observedEvents.containsKey(source)) {
- // yes, add all events to result list
- result.addAll(observedEvents.get(source));
- }
-
- // log exiting and return result
- logger.exiting(this.getClass().getName(), "getEventHistory", result);
- return result;
- }
-
- public synchronized boolean hasEventOccured(final String source,
- final EventType type) {
-
- // log entering
- logger.entering(this.getClass().getName(), "hasEventOccured",
- new Object[] { source, type });
-
- // check parameters
- if (source == null || type == null || !eventSources.contains(source)) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "hasEventOccured", e);
- throw e;
- }
-
- // determine result
- boolean result = false;
- if (observedEvents.containsKey(source)) {
- for (final Event event : observedEvents.get(source)) {
- if (event.getType() == type) {
- result = true;
- break;
- }
- }
- }
-
- // log exiting and return result
- logger.exiting(this.getClass().getName(), "hasEventOccured", result);
- return result;
- }
-
- /**
- * An ordered list of all log statements that are still unparsed.
- */
- private final List<EventImpl> unparsedLogStatements =
- new LinkedList<EventImpl>();
-
- /**
- * Stores the occurrence of an unparsed Tor log events that might result in
- * an event. All such unparsed events are later parsed by a background
- * thread in invocation order. Then, the occurrence time and the event type
- * will be parsed from the log message; if the log message does not contain
- * anything of interest, the event will be discarded.
- *
- * @param source
- * The event source.
- * @param logMessage
- * The log message.
- * @throws IllegalArgumentException
- * Thrown if the source is unknown.
- */
- synchronized void observeUnparsedEvent(final String source,
- final String logMessage) {
- unparsedLogStatements.add(new EventImpl(source, logMessage));
- notifyAll();
- }
-
- /**
- * Add an internal event to the event queue.
- *
- * @param occurrenceTime
- * The occurrence time of the event.
- * @param source
- * The event source.
- * @param type
- * The event type.
- * @param message
- * The event message.
- * @throws IllegalArgumentException
- * Thrown if the source is unknown.
- */
- public synchronized void observeInternalEvent(final long occurrenceTime,
- final String source, final EventType type, final String message) {
-
- if (!eventSources.contains(source)) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "observeInternalEvent",
- e);
- throw e;
- }
-
- unparsedLogStatements.add(new EventImpl(occurrenceTime, source, type,
- message));
- notifyAll();
- }
-
- /**
- * Parses a log statement coming from Tor and decides whether it is
- * interesting for us.
- */
- void parseNextEvent() {
-
- // wait for the next event in the queue
- EventImpl event = null;
- synchronized (this) {
- while (unparsedLogStatements.isEmpty()) {
- try {
- wait();
- } catch (final InterruptedException e) {}
- }
- event = unparsedLogStatements.remove(0);
- }
-
- // does the event contain a known source? if not, discard it
- if (!eventSources.contains(event.getSource())) {
- logger.log(Level.WARNING,
- "Unknown event source while parsing an event: "
- + event.getSource());
- return;
- }
-
- // does the event require parsing? if not, process immediately
- if (event.getType() != null) {
- observeEvent(event);
- } else {
- final String line = event.getMessage();
-
- /*
- * the logging output of Tor does not contain a year component; put
- * in the year at the time of parsing (which happens approx. 10 ms
- * after the logging took place) in the good hope that this tool is
- * not run at midnight on New Year's Eve...
- */
- final Calendar c = Calendar.getInstance();
- final int currentYear = c.get(Calendar.YEAR);
-
- // try to apply one of the event type patterns
- for (final Entry<Pattern, EventType> entry : eventTypePatterns
- .entrySet()) {
- final Matcher matcher = entry.getKey().matcher(line);
- if (matcher.find()) {
- final SimpleDateFormat sdf =
- new SimpleDateFormat("MMM dd HH:mm:ss.SSS",
- Locale.US);
- final Date logTime = sdf.parse(line, new ParsePosition(0));
- c.setTimeInMillis(logTime.getTime());
- c.set(Calendar.YEAR, currentYear);
- final long t = c.getTimeInMillis();
- event.setOccurenceTime(t);
- event.setType(entry.getValue());
- observeEvent(event);
- break;
- }
- }
- }
- }
-
- /**
- * Map of all patterns, that should be included when parsing log statements
- * coming from Tor, and the respective event types of the events that should
- * be fired.
- */
- Map<Pattern, EventType> eventTypePatterns;
-
- public synchronized void registerEventTypePattern(
- final String patternString, final EventType eventType) {
- eventTypePatterns.put(Pattern.compile(patternString), eventType);
- }
-
- /**
- * Initializes the parsing engine with the standard log message patterns
- * that should be included in current Tor. Any further patterns need to be
- * added by the test application manually.
- */
- private void initializeEventTypePatterns() {
- eventTypePatterns = new HashMap<Pattern, EventType>();
- registerEventTypePattern("Opening Control listener on .*",
- NodeEventType.NODE_CONTROL_PORT_OPENED);
- registerEventTypePattern("Tor has successfully opened a circuit. "
- + "Looks like client functionality is working.",
- NodeEventType.NODE_CIRCUIT_OPENED);
- registerEventTypePattern(
- "Established circuit .* as introduction "
- + "point for service .*",
- HiddenServiceEventType.BOB_BUILT_INTRO_CIRC_SENDING_ESTABLISH_INTRO);
- registerEventTypePattern("Received INTRO_ESTABLISHED cell on "
- + "circuit .* for service .*",
- HiddenServiceEventType.BOB_INTRO_ESTABLISHED_RECEIVED);
- registerEventTypePattern("Sending publish request for hidden "
- + "service .*", HiddenServiceEventType.BOB_SENDING_PUBLISH_DESC);
- registerEventTypePattern("Uploaded rendezvous descriptor",
- HiddenServiceEventType.BOB_DESC_PUBLISHED_RECEIVED);
- registerEventTypePattern("Received INTRODUCE2 cell for service .* "
- + "on circ .*", HiddenServiceEventType.BOB_INTRODUCE2_RECEIVED);
- registerEventTypePattern("Done building circuit .* to rendezvous "
- + "with cookie .* for service .*",
- HiddenServiceEventType.BOB_BUILT_REND_CIRC_SENDING_RENDEZVOUS1);
- registerEventTypePattern("begin is for rendezvous",
- HiddenServiceEventType.BOB_APP_CONN_OPENED);
- registerEventTypePattern("Got a hidden service request for ID '.*'",
- HiddenServiceEventType.ALICE_ONION_REQUEST_RECEIVED);
- registerEventTypePattern("Fetching rendezvous descriptor for "
- + "service .*", HiddenServiceEventType.ALICE_SENDING_FETCH_DESC);
- registerEventTypePattern("Received rendezvous descriptor",
- HiddenServiceEventType.ALICE_DESC_FETCHED_RECEIVED);
- registerEventTypePattern(
- "Sending an ESTABLISH_RENDEZVOUS cell",
- HiddenServiceEventType.ALICE_BUILT_REND_CIRC_SENDING_ESTABLISH_RENDEZVOUS);
- registerEventTypePattern("Got rendezvous ack. This circuit is now "
- + "ready for rendezvous",
- HiddenServiceEventType.ALICE_RENDEZVOUS_ESTABLISHED_RECEIVED);
- registerEventTypePattern("introcirc is open",
- HiddenServiceEventType.ALICE_BUILT_INTRO_CIRC);
- registerEventTypePattern("Sending an INTRODUCE1 cell",
- HiddenServiceEventType.ALICE_SENDING_INTRODUCE1);
- registerEventTypePattern("Received ack. Telling rend circ",
- HiddenServiceEventType.ALICE_INTRODUCE_ACK_RECEIVED);
- registerEventTypePattern(
- "Got RENDEZVOUS2 cell from hidden service",
- HiddenServiceEventType.ALICE_RENDEZVOUS2_RECEIVED_APP_CONN_OPENED);
- registerEventTypePattern("Handling rendezvous descriptor post",
- HiddenServiceEventType.DIR_PUBLISH_DESC_RECEIVED);
- registerEventTypePattern("Handling rendezvous descriptor get",
- HiddenServiceEventType.DIR_FETCH_DESC_RECEIVED);
- registerEventTypePattern(
- "Received an ESTABLISH_INTRO request on circuit .*",
- HiddenServiceEventType.IPO_RECEIVED_ESTABLISH_INTRO_SENDING_INTRO_ESTABLISHED);
- registerEventTypePattern(
- "Received an INTRODUCE1 request on circuit .*",
- HiddenServiceEventType.IPO_RECEIVED_INTRODUCE1_SENDING_INTRODUCE2_AND_INTRODUCE_ACK);
- registerEventTypePattern(
- "Received an ESTABLISH_RENDEZVOUS request on circuit .*",
- HiddenServiceEventType.RPO_RECEIVED_ESTABLISH_RENDEZVOUS_SENDING_RENDEZVOUS_ESTABLISHED);
- registerEventTypePattern(
- "Got request for rendezvous from circuit .* to cookie .*",
- HiddenServiceEventType.RPO_RECEIVING_RENDEZVOUS1_SENDING_RENDEZVOUS2);
- }
-
- /**
- * Stores the given <code>event</code> from <code>source</code> to the
- * event history and propagates its occurrence to all registered event
- * handlers.
- *
- * @param event
- * The observed event.
- */
- private synchronized void observeEvent(final Event event) {
-
- // log entering
- logger.entering(this.getClass().getName(), "observeEvent", event);
-
- final String source = event.getSource();
- logger.log(Level.FINE, "Observed event " + event + " from source "
- + source + "!");
-
- // remember observed event
- if (!observedEvents.containsKey(event.getSource())) {
- observedEvents.put(source, new ArrayList<Event>());
- }
- observedEvents.get(source).add(event);
-
- // notify waiting threads
- notifyAll();
-
- // inform event listeners
- if (eventHandlers.containsKey(source)) {
-
- // make a copy of the event handler set, because some event handlers
- // might want to remove themselves from this set while handling the
- // event
- final Set<EventListener> copyOfEventHandlers =
- new HashSet<EventListener>(eventHandlers.get(source));
-
- for (final EventListener eventHandler : copyOfEventHandlers) {
-
- logger.log(Level.FINE, "Informing event listener "
- + eventHandler + " about recently observed event "
- + event + " from source " + source + "!");
- eventHandler.handleEvent(event);
-
- }
- }
-
- // make a copy of the event handler set for all sources, because some
- // event handlers might want to remove themselves from this set while
- // handling the event
- final Set<EventListener> copyOfEventHandlersForAllSources =
- new HashSet<EventListener>(eventHandlersForAllSources);
-
- for (final EventListener eventHandler : copyOfEventHandlersForAllSources) {
-
- logger.log(Level.FINE, "Informing event listener " + eventHandler
- + " about recently observed event " + event
- + " from source " + source + "!");
- eventHandler.handleEvent(event);
- }
-
- // log exiting
- logger.exiting(this.getClass().getName(), "observeEvent");
- }
-
- public synchronized void removeEventListener(
- final EventListener eventListener) {
-
- // log entering
- logger.entering(this.getClass().getName(), "removeEventListener",
- eventListener);
-
- // check parameters
- if (eventListener == null) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger
- .throwing(this.getClass().getName(), "removeEventListener",
- e);
- throw e;
- }
-
- // don't know to which source this listener has been added (may to more
- // than one), so remove it from all possible sets
- for (final Set<EventListener> set : eventHandlers.values()) {
- if (set.remove(eventListener)) {
- logger.log(Level.FINE, "Removed event listener!");
- }
- }
-
- // remove from event listeners for all sources
- if (eventHandlersForAllSources.remove(eventListener)) {
- logger.log(Level.FINE, "Removed event listener!");
- }
-
- // log exiting
- logger.exiting(this.getClass().getName(), "removeEventListener");
- }
-
- public synchronized void waitForAnyOccurence(final String source,
- final EventType event) {
-
- // log entering
- logger.entering(this.getClass().getName(), "waitForAnyOccurence",
- new Object[] { source, event });
-
- // check parameters
- if (source == null || event == null || !eventSources.contains(source)) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger
- .throwing(this.getClass().getName(), "waitForAnyOccurence",
- e);
- throw e;
- }
-
- // invoke overloaded method with maximumTimeToWaitInMillis of -1L which
- // means to wait forever
- waitForAnyOccurence(source, event, -1L);
-
- // log exiting
- logger.exiting(this.getClass().getName(), "waitForAnyOccurence");
-
- }
-
- public synchronized boolean waitForAnyOccurence(final String source,
- final EventType event, final long maximumTimeToWaitInMillis) {
-
- // log entering
- logger.entering(this.getClass().getName(), "waitForAnyOccurence",
- new Object[] { source, event, maximumTimeToWaitInMillis });
-
- // check parameters
- if (source == null || event == null || !eventSources.contains(source)) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger
- .throwing(this.getClass().getName(), "waitForAnyOccurence",
- e);
- throw e;
- }
-
- // check if we have already observed the event
- if (hasEventOccured(source, event)) {
- logger.log(Level.FINE, "Waiting for any occurence of event "
- + event + " returned immediately!");
- logger.exiting(this.getClass().getName(), "waitForAnyOccurence",
- true);
- return true;
- }
-
- // invoke method that waits for next occurence of the event
- final boolean result =
- waitForNextOccurence(source, event, maximumTimeToWaitInMillis);
-
- // log exiting and return result
- logger
- .exiting(this.getClass().getName(), "waitForAnyOccurence",
- result);
- return result;
- }
-
- public synchronized void waitForNextOccurence(final String source,
- final EventType event) {
-
- // log entering
- logger.entering(this.getClass().getName(), "waitForNextOccurence",
- new Object[] { source, event });
-
- // check parameters
- if (source == null || event == null || !eventSources.contains(source)) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "waitForNextOccurence",
- e);
- throw e;
- }
-
- // invoke overloaded method with maximumTimeToWaitInMillis of -1L which
- // means to wait forever
- waitForNextOccurence(source, event, -1L);
-
- // log exiting
- logger.exiting(this.getClass().getName(), "waitForNextOccurence");
- }
-
- public synchronized boolean waitForNextOccurence(final String source,
- final EventType type, final long maximumTimeToWaitInMillis) {
-
- // log entering
- logger.entering(this.getClass().getName(), "waitForNextOccurence",
- new Object[] { source, type, maximumTimeToWaitInMillis });
-
- // check parameters
- if (source == null || type == null || !eventSources.contains(source)) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "waitForNextOccurence",
- e);
- throw e;
- }
-
- // distinguish between negative waiting time (wait forever) and zero or
- // positive waiting time
- if (maximumTimeToWaitInMillis < 0) {
-
- // wait forever
- while (!hasEventOccured(source, type)) {
-
- logger.log(Level.FINEST,
- "We will wait infinetely for the next occurence of "
- + "event type " + type + " from source "
- + source + "...");
- try {
- wait();
- } catch (final InterruptedException e) {
- // don't handle
- }
-
- logger.log(Level.FINEST,
- "We have been notified about an observed event while "
- + "waiting for events of type " + type
- + " from source " + source
- + "; need to check whether the observed event "
- + "is what we are looking for...");
- }
-
- logger.log(Level.FINE, "Waiting for occurence of event type "
- + type + " succeeded!");
-
- // log exiting and return result
- logger.exiting(this.getClass().getName(), "waitForNextOccurence",
- true);
- return true;
-
- }
-
- // wait for the given time at most
- final long endOfTime =
- System.currentTimeMillis() + maximumTimeToWaitInMillis;
- long timeLeft = 0;
- while (!hasEventOccured(source, type)
- && (timeLeft = endOfTime - System.currentTimeMillis()) > 0) {
-
- logger.log(Level.FINEST, "We will wait for " + timeLeft
- + " milliseconds for the next occurence of event type "
- + type + " from source " + source + "...");
-
- try {
- wait(timeLeft);
- } catch (final InterruptedException e) {
- // don't handle
- }
-
- logger.log(Level.FINEST,
- "We have been notified about an observed event while "
- + "waiting for events of type " + type
- + " from source " + source
- + "; need to check whether the observed event "
- + "is what we are looking for...");
- }
-
- // determine result
- final boolean result = hasEventOccured(source, type);
-
- logger.log(Level.FINE, "Waiting for next occurence of event type "
- + type + " from source " + source
- + (result ? " succeeded!" : " did not succeed!"));
-
- // log exiting and return result
- logger.exiting(this.getClass().getName(), "waitForNextOccurence",
- result);
- return result;
- }
-
- /**
- * Adds the given <code>source</code> as possible event source.
- *
- * @param source
- * The name of the node, client, or server to add.
- * @throws IllegalArgumentException
- * Thrown if there is already an event source with this name.
- */
- void addEventSource(final String source) {
-
- // log entering
- logger.entering(this.getClass().getName(), "addEventSource", source);
-
- // check if source name is unique in this network
- if (eventSources.contains(source)) {
- final IllegalArgumentException e =
- new IllegalArgumentException(
- "There is already an event source with name "
- + source + " in this network!");
- logger.throwing(this.getClass().getName(), "addEventSource", e);
- throw e;
- }
-
- // add event source name
- eventSources.add(source);
-
- // log exiting
- logger.exiting(this.getClass().getName(), "addEventSource");
- }
-
- @Override
- public String toString() {
- return this.getClass().getSimpleName();
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/impl/HiddenServiceImpl.java b/src/de/uniba/wiai/lspi/puppetor/impl/HiddenServiceImpl.java
deleted file mode 100644
index 61c0763..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/impl/HiddenServiceImpl.java
+++ /dev/null
@@ -1,166 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.impl;
-
-import java.io.BufferedReader;
-import java.io.File;
-import java.io.FileReader;
-import java.io.IOException;
-import java.util.logging.Level;
-import java.util.logging.Logger;
-
-import de.uniba.wiai.lspi.puppetor.HiddenService;
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-
-/**
- * Implementation of <code>HiddenService</code>.
- *
- * @author kloesing
- */
-public class HiddenServiceImpl implements HiddenService {
-
- /**
- * Logger for this node which is called "hidserv." plus the name of this
- * hidden service.
- */
- protected Logger logger;
-
- /**
- * The node at which this hidden service is configured.
- */
- private final ProxyNodeImpl node;
-
- /**
- * Name of the hidden service that will be used as name for the hidden
- * service directory.
- */
- private final String serviceName;
-
- /**
- * The TCP port on which the service will be available for requests.
- */
- private final int servicePort;
-
- /**
- * The virtual TCP port that this hidden service runs on as it is announced
- * to clients.
- */
- private final int virtualPort;
-
- /**
- * Adds the entries for a hidden service to the configuration of this node.
- *
- * Creates a new <code>HiddenServiceImpl</code>.
- *
- * @param node
- * The node at which this hidden service is configured.
- * @param serviceName
- * Name of the hidden service that will be used as name for the
- * hidden service directory. May neither be <code>null</code>
- * or a zero-length string.
- * @param servicePort
- * The TCP port on which the service will be available for
- * requests. This can, but need not be different from the virtual
- * port that is announced to clients. May not be negative or
- * greater than 65535.
- * @param virtualPort
- * The virtual TCP port that this hidden service runs on as it is
- * announced to clients. May not be negative or greater than
- * 65535.
- * @throws IllegalArgumentException
- * Thrown if an invalid value is given for either of the
- * parameters.
- */
- HiddenServiceImpl(final ProxyNodeImpl node, final String serviceName,
- final int servicePort, final int virtualPort) {
-
- // check if networkName can be used as logger name
- if (serviceName == null || serviceName.length() == 0) {
- throw new IllegalArgumentException("Invalid serviceName: "
- + serviceName);
- }
-
- // create logger
- logger = Logger.getLogger("hidserv." + serviceName);
-
- // log entering
- logger.entering(this.getClass().getName(), "HiddenServiceImpl",
- new Object[] { node, serviceName, servicePort, virtualPort });
-
- // check parameters
- if (servicePort < 0 || servicePort > 65535 || virtualPort < 0
- || virtualPort > 65535) {
- logger.log(Level.SEVERE,
- "Illegal argument when adding hidden service!");
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "addHiddenService", e);
- throw e;
- }
-
- // store parameter values
- this.node = node;
- this.serviceName = serviceName;
- this.servicePort = servicePort;
- this.virtualPort = virtualPort;
-
- // log exiting
- logger.exiting(this.getClass().getName(), "HiddenServiceImpl");
- }
-
- public String determineOnionAddress() throws PuppeTorException {
-
- // log entering
- logger.entering(this.getClass().getName(), "getOnionAddress");
-
- // check if hidden service directory exists
- final File hiddenServiceFile =
- new File(node.getWorkingDir().getAbsolutePath()
- + File.separator + serviceName + File.separator
- + "hostname");
- if (!hiddenServiceFile.exists()) {
- logger.log(Level.SEVERE,
- "Hidden service directory or hostname file does not exist: "
- + hiddenServiceFile.getAbsolutePath());
-
- final PuppeTorException e =
- new PuppeTorException(
- "Hidden service directory or hostname file does not exist: "
- + hiddenServiceFile.getAbsolutePath());
- logger.throwing(this.getClass().getName(), "getOnionAddress", e);
- throw e;
- }
-
- // read hostname from file
- String address = null;
- try {
- final BufferedReader br =
- new BufferedReader(new FileReader(hiddenServiceFile));
- address = br.readLine();
- br.close();
- } catch (final IOException e) {
- final PuppeTorException ex =
- new PuppeTorException("Could not read hostname file!", e);
- logger.throwing(this.getClass().getName(), "getOnionAddress", ex);
- throw ex;
- }
-
- // log exiting and return address
- logger.exiting(this.getClass().getName(), "getOnionAddress", address);
- return address;
- }
-
- public String getServiceName() {
- return serviceName;
- }
-
- public int getServicePort() {
- return servicePort;
- }
-
- public int getVirtualPort() {
- return virtualPort;
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/impl/NetworkImpl.java b/src/de/uniba/wiai/lspi/puppetor/impl/NetworkImpl.java
deleted file mode 100644
index 5321abd..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/impl/NetworkImpl.java
+++ /dev/null
@@ -1,1082 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.impl;
-
-import java.io.File;
-import java.util.ArrayList;
-import java.util.HashMap;
-import java.util.HashSet;
-import java.util.List;
-import java.util.Map;
-import java.util.Set;
-import java.util.concurrent.ConcurrentHashMap;
-import java.util.logging.Level;
-import java.util.logging.Logger;
-
-import de.uniba.wiai.lspi.puppetor.ClientApplication;
-import de.uniba.wiai.lspi.puppetor.DirectoryNode;
-import de.uniba.wiai.lspi.puppetor.Event;
-import de.uniba.wiai.lspi.puppetor.EventListener;
-import de.uniba.wiai.lspi.puppetor.EventManager;
-import de.uniba.wiai.lspi.puppetor.Network;
-import de.uniba.wiai.lspi.puppetor.NodeEventType;
-import de.uniba.wiai.lspi.puppetor.NodeState;
-import de.uniba.wiai.lspi.puppetor.ProxyNode;
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-import de.uniba.wiai.lspi.puppetor.RouterNode;
-import de.uniba.wiai.lspi.puppetor.ServerApplication;
-
-/**
- * Implementation of <code>Network</code>.
- *
- * @author kloesing
- */
-public class NetworkImpl implements Network {
-
- /**
- * Internal thread class that is used to start Tor processes in parallel.
- */
- private class NodeStarter extends Thread {
-
- /**
- * The exception, if one is caught while trying to start the node.
- */
- Exception caughtException;
-
- /**
- * The maximum time to wait for the Tor process to start in
- * milliseconds.
- */
- private final long maximumTimeToWaitInMillis;
-
- /**
- * The node for which the Tor process shall be started.
- */
- private final ProxyNode node;
-
- /**
- * Flag that denotes whether starting the Tor process was successful.
- */
- boolean success = false;
-
- /**
- * Creates a new <code>NodeStarter</code> for node <code>node</code>
- * that will wait for <code>maximumTimeToWaitInMillis</code>
- * milliseconds to start a Tor process, but that is not started, yet.
- *
- * @param node
- * The node for which the Tor process shall be started.
- * @param maximumTimeToWaitInMillis
- * The maximum time to wait for the Tor process to start in
- * milliseconds.
- */
- NodeStarter(final ProxyNode node, final long maximumTimeToWaitInMillis) {
-
- // log entering
- logger.entering(this.getClass().getName(), "NodeStarter",
- new Object[] { node, maximumTimeToWaitInMillis });
-
- // store parameters
- this.node = node;
- this.maximumTimeToWaitInMillis = maximumTimeToWaitInMillis;
-
- // log exiting
- logger.exiting(this.getClass().getName(), "NodeStarter");
- }
-
- @Override
- public void run() {
-
- // log entering
- logger.entering(this.getClass().getName(), "run");
-
- try {
- // try to start node
- success = node.startNode(maximumTimeToWaitInMillis);
- } catch (final PuppeTorException e) {
- logger.log(Level.SEVERE,
- "Caught an exception while starting node "
- + node.toString() + "!");
- caughtException = e;
- }
-
- // log exiting
- logger.exiting(this.getClass().getName(), "run");
- }
- }
-
- /**
- * Event manager to which all events concerning this network are notified.
- */
- private final EventManagerImpl eventManager;
-
- /**
- * Logger for this network which is called "network." plus the name of this
- * network.
- */
- private final Logger logger;
-
- /**
- * Contains the name of this network configuration which is the String
- * conversion of System.currentTimeMillis() of the network creation time.
- */
- private final String networkName;
-
- /**
- * All nodes contained in this network.
- */
- private final Map<String, ProxyNode> nodes;
-
- /**
- * Directory that contains status information of all nodes contained in this
- * network.
- */
- private final File workingDir;
-
- /**
- * The counter for automatically assigned port numbers created by this
- * <code>Network</code>.
- */
- private int portCounter = 7000;
-
- /**
- * Creates an initially unpopulated Tor network and creates a new working
- * directory for it at test-env/randomTestID/.
- *
- * @param networkName
- * Name of this network configuration. May neither be
- * <code>null</code> or a zero-length string.
- * @param startPort
- * The initial value for automatically assigned port numbers of
- * nodes created by this <code>Network</code>; must be a value
- * between 1024 and 65535.
- * @throws IllegalArgumentException
- * Thrown if the given <code>networkName</code> is either
- * <code>null</code> or a zero-length string, or if an illegal
- * number is given for <code>startPort</code>.
- */
- public NetworkImpl(final String networkName, final int startPort) {
- // initialize using overloaded constructor
- this(networkName);
-
- // check if start port is valid
- if (startPort < 1024 || startPort > 65535) {
- throw new IllegalArgumentException("Invalid startPort: "
- + startPort);
- }
-
- // remember parameter
- portCounter = startPort;
-
- // log exiting
- logger.exiting(this.getClass().getName(), "NetworkImpl");
- }
-
- /**
- * Creates an initially unpopulated Tor network and creates a new working
- * directory for it at test-env/randomTestID/.
- *
- * @param networkName
- * Name of this network configuration. May neither be
- * <code>null</code> or a zero-length string.
- * @throws IllegalArgumentException
- * Thrown if the given <code>networkName</code> is either
- * <code>null</code> or a zero-length string.
- */
- public NetworkImpl(final String networkName) {
-
- // check if networkName can be used as logger name
- if (networkName == null || networkName.length() == 0) {
- throw new IllegalArgumentException("Invalid networkName: "
- + networkName);
- }
-
- // create logger
- logger = Logger.getLogger("network." + networkName);
-
- // log entering
- logger.entering(this.getClass().getName(), "NetworkImpl", networkName);
-
- // TODO is this necessary?!
- logger.setLevel(Level.ALL);
-
- // remember parameter
- this.networkName = networkName;
-
- // create working directory
- workingDir = new File("test-env/" + System.currentTimeMillis());
- workingDir.mkdirs();
- logger.log(Level.FINE, "Created working directory \""
- + workingDir.getAbsolutePath() + "\"");
-
- // TODO if we want to log to file, set this... somehow...
- // this.logFile = new File(this.workingDir.getAbsolutePath()
- // + "\\events.log");
-
- // initialize data structures
- nodes = new ConcurrentHashMap<String, ProxyNode>();
-
- // create event manager
- eventManager = new EventManagerImpl(this.networkName);
-
- // log exiting
- logger.exiting(this.getClass().getName(), "NetworkImpl");
- }
-
- /**
- * Returns whether all nodes in this network are up, or not.
- *
- * @return <code>true</code> if all nodes are up, <code>false</code> if
- * at least one node is not up.
- */
- private boolean allNodesUp() {
-
- // log entering
- logger.entering(this.getClass().getName(), "allNodesUp");
-
- // fail on first node that is not up
- for (final ProxyNode node : nodes.values()) {
- if (!eventManager.hasEventOccured(node.getNodeName(),
- NodeEventType.NODE_CIRCUIT_OPENED)) {
-
- // log exiting and return false
- logger.exiting(this.getClass().getName(), "allNodesUp");
- return false;
- }
- }
-
- // log exiting and return true
- logger.exiting(this.getClass().getName(), "allNodesUp");
- return true;
- }
-
- public void configureAsPrivateNetwork() throws PuppeTorException {
- // log entering
- logger.entering(this.getClass().getName(), "configureAsPrivateNetwork");
-
- // read DirServer strings for all directories
- final List<String> authorizedDirectoriesFingerprints =
- new ArrayList<String>();
- for (final ProxyNode node : nodes.values()) {
- if (node instanceof DirectoryNode) {
- final DirectoryNode dirNode = (DirectoryNode) node;
- authorizedDirectoriesFingerprints.add(dirNode
- .getDirServerString());
- }
- }
-
- this.configureAsPartOfPrivateNetwork(authorizedDirectoriesFingerprints);
-
- // read fingerprints for all directories and routers
- final HashSet<String> approvedRoutersFingerprints =
- new HashSet<String>();
- for (final ProxyNode node : nodes.values()) {
- if (node instanceof RouterNode) {
- final RouterNode routerOrDirNode = (RouterNode) node;
- approvedRoutersFingerprints.add(routerOrDirNode
- .getFingerprint());
- }
- }
-
- // write fingerprints for all directories and routers to the
- // approved-routers file
- for (final ProxyNode node : nodes.values()) {
- if (node instanceof DirectoryNode) {
- final DirectoryNode dirNode = (DirectoryNode) node;
- dirNode.addApprovedRouters(approvedRoutersFingerprints);
- }
- }
-
- // log exiting
- logger.exiting(this.getClass().getName(), "configureAsPrivateNetwork");
- }
-
- public void configureAsPartOfPrivateNetwork(
- List<String> authorizedDirectoriesFingerprints) {
- //throws PuppeTorException {
- // log entering
- logger.entering(this.getClass().getName(), "configureAsPartOfPrivateNetwork");
-
- for (final ProxyNode node : nodes.values()) {
- if (node.getNodeState() == NodeState.CONFIGURING ||
- node.getNodeState() == NodeState.CONFIGURATION_WRITTEN) {
- // add to configuration
- node.addConfiguration("TestingTorNetwork 1");
- }
- }
-
- // configure nodes
- for (final ProxyNode node : nodes.values()) {
- if (node.getNodeState() == NodeState.CONFIGURING ||
- node.getNodeState() == NodeState.CONFIGURATION_WRITTEN) {
- // add to configuration
- node.addConfigurations(authorizedDirectoriesFingerprints);
- }
- }
-
-/* // read fingerprints for all directories and routers
- final HashSet<String> approvedRoutersFingerprints =
- new HashSet<String>();
- for (final ProxyNode node : nodes.values()) {
- if (node instanceof RouterNode) {
- final RouterNode routerOrDirNode = (RouterNode) node;
- approvedRoutersFingerprints.add(routerOrDirNode
- .getFingerprint());
- }
- }
-
- // write fingerprints for all directories and routers to the
- // approved-routers file
- for (final ProxyNode node : nodes.values()) {
- if (node instanceof DirectoryNode) {
- final DirectoryNode dirNode = (DirectoryNode) node;
- dirNode.addApprovedRouters(approvedRoutersFingerprints);
- }
- }
-*/
- // log exiting
- logger.exiting(this.getClass().getName(), "configureAsPartOfPrivateNetwork");
- }
-
- public ClientApplication createClient(final String clientApplicationName,
- final String targetAddress, final int targetPort,
- final int socksPort) {
- // log entering
- logger.entering(this.getClass().getName(), "createClient",
- new Object[] { clientApplicationName, targetAddress,
- targetPort, socksPort });
-
- // create client; parameter checking is done in constructor
- final ClientApplicationImpl client =
- new ClientApplicationImpl(this, clientApplicationName,
- targetAddress, targetPort, socksPort);
-
- // add name to event manager as event source
- eventManager.addEventSource(clientApplicationName);
-
- // log exiting and return client
- logger.exiting(this.getClass().getName(), "createClient", client);
- return client;
- }
-
- public DirectoryNode createDirectory(final String nodeName,
- final int controlPort, final int socksPort, final int orPort,
- final int dirPort, final String serverIpAddress) {
- // log entering
- logger.entering(this.getClass().getName(), "createDirectory",
- new Object[] { nodeName, controlPort, socksPort, orPort,
- dirPort, serverIpAddress });
-
- // create directory node; parameter checking is done in constructor
- final DirectoryNode dir =
- new DirectoryNodeImpl(this, nodeName, controlPort, socksPort,
- orPort, dirPort, serverIpAddress);
-
- // add new directory node to nodes collection
- nodes.put(nodeName, dir);
-
- // add name to event manager as event source
- eventManager.addEventSource(nodeName);
-
- // log exiting and return directory node
- logger.exiting(this.getClass().getName(), "createDirectory", dir);
- return dir;
- }
-
- public DirectoryNode createDirectory(final String nodeName,
- final int controlPort, final int socksPort, final int orPort,
- final int dirPort) {
-
- // log entering
- logger.entering(this.getClass().getName(), "createDirectory",
- new Object[] { nodeName, controlPort, socksPort, orPort,
- dirPort });
-
- // invoke overloaded method
- final DirectoryNode dir =
- this.createDirectory(nodeName, controlPort, socksPort, orPort,
- dirPort, "127.0.0.1");
-
- // log exiting and return directory node
- logger.exiting(this.getClass().getName(), "createDirectory", dir);
- return dir;
- }
-
- public DirectoryNode createDirectory(final String nodeName,
- final String serverIpAddress) {
-
- // log entering
- logger.entering(this.getClass().getName(), "createDirectory",
- new Object[] { nodeName, serverIpAddress });
-
- // invoke overloaded method
- final DirectoryNode dir =
- this.createDirectory(nodeName, portCounter++, portCounter++,
- portCounter++, portCounter++, serverIpAddress);
-
- // log exiting and return directory node
- logger.exiting(this.getClass().getName(), "createDirectory", dir);
- return dir;
- }
-
- public DirectoryNode createDirectory(final String nodeName) {
-
- // log entering
- logger.entering(this.getClass().getName(), "createDirectory", nodeName);
-
- // invoke overloaded method
- final DirectoryNode dir =
- this.createDirectory(nodeName, portCounter++, portCounter++,
- portCounter++, portCounter++, "127.0.0.1");
-
- // log exiting and return directory node
- logger.exiting(this.getClass().getName(), "createDirectory", dir);
- return dir;
- }
-
- public ProxyNode createProxy(final String nodeName, final int controlPort,
- final int socksPort) {
-
- // log entering
- logger.entering(this.getClass().getName(), "createProxy", new Object[] {
- nodeName, controlPort, socksPort });
-
- // create proxy node; parameter checking is done in constructor
- final ProxyNode proxy =
- new ProxyNodeImpl(this, nodeName, controlPort, socksPort);
-
- // add new proxy node to nodes collection
- nodes.put(nodeName, proxy);
-
- // add name to event manager as event source
- eventManager.addEventSource(nodeName);
-
- // log exiting and return proxy node
- logger.exiting(this.getClass().getName(), "createProxy", proxy);
- return proxy;
- }
-
- public ProxyNode createProxy(final String nodeName) {
-
- // log entering
- logger.entering(this.getClass().getName(), "createProxy", nodeName);
-
- // invoke overloaded method
- final ProxyNode proxy =
- this.createProxy(nodeName, portCounter++, portCounter++);
-
- // log exiting and return proxy node
- logger.exiting(this.getClass().getName(), "createProxy", proxy);
- return proxy;
- }
-
- public RouterNode createRouter(final String nodeName,
- final int controlPort, final int socksPort, final int orPort,
- final int dirPort, final String serverIpAddress) {
-
- // log entering
- logger.entering(this.getClass().getName(), "createRouter",
- new Object[] { nodeName, controlPort, socksPort, orPort,
- dirPort, serverIpAddress });
-
- // create router node; parameter checking is done in constructor
- final RouterNode router =
- new RouterNodeImpl(this, nodeName, controlPort, socksPort,
- orPort, dirPort, serverIpAddress);
-
- // add new router node to nodes collection
- nodes.put(nodeName, router);
-
- // add name to event manager as event source
- eventManager.addEventSource(nodeName);
-
- // log exiting and return router node
- logger.exiting(this.getClass().getName(), "createRouter", router);
- return router;
- }
-
- public RouterNode createRouter(final String nodeName,
- final int controlPort, final int socksPort, final int orPort,
- final int dirPort) {
-
- // log entering
- logger.entering(this.getClass().getName(), "createRouter",
- new Object[] { nodeName, controlPort, socksPort, orPort,
- dirPort });
-
- // invoke overloaded method
- final DirectoryNode dir =
- this.createDirectory(nodeName, controlPort, socksPort, orPort,
- dirPort, "127.0.0.1");
-
- // log exiting and return directory node
- logger.exiting(this.getClass().getName(), "createRouter", dir);
- return dir;
- }
-
- public RouterNode createRouter(final String nodeName,
- final String serverIpAddress) {
-
- // log entering
- logger.entering(this.getClass().getName(), "createRouter",
- new Object[] { nodeName, serverIpAddress });
-
- // invoke overloaded method
- final RouterNode dir =
- this.createRouter(nodeName, portCounter++, portCounter++,
- portCounter++, portCounter++, serverIpAddress);
-
- // log exiting and return directory node
- logger.exiting(this.getClass().getName(), "createRouter", dir);
- return dir;
- }
-
- public RouterNode createRouter(final String nodeName) {
-
- // log entering
- logger.entering(this.getClass().getName(), "createRouter", nodeName);
-
- // invoke overloaded method
- final RouterNode router =
- this.createRouter(nodeName, portCounter++, portCounter++,
- portCounter++, portCounter++, "127.0.0.1");
-
- // log exiting and return router node
- logger.exiting(this.getClass().getName(), "createRouter", router);
- return router;
- }
-
- public ServerApplication createServer(final String serverApplicationName,
- final int serverPort) {
-
- // log entering
- logger.entering(this.getClass().getName(), "createServer",
- new Object[] { serverApplicationName, serverPort });
-
- // create server; parameter checking is done in constructor
- final ServerApplicationImpl server =
- new ServerApplicationImpl(this, serverApplicationName,
- serverPort);
-
- // add name to event manager as event source
- eventManager.addEventSource(serverApplicationName);
-
- // log exiting and return server
- logger.exiting(this.getClass().getName(), "createServer", server);
- return server;
- }
-
- public ServerApplication createServer(final String serverApplicationName) {
-
- // log entering
- logger.entering(this.getClass().getName(), "createServer",
- serverApplicationName);
-
- // invoke overloaded method
- final ServerApplication server =
- this.createServer(serverApplicationName, portCounter++);
-
- // log exiting and return server
- logger.exiting(this.getClass().getName(), "createServer", server);
- return server;
- }
-
- public EventManager getEventManager() {
- return eventManager;
- }
-
- /**
- * Returns the implementation instance of the event manager of this network.
- *
- * @return The implementation instance of the event manager of this network.
- */
- public EventManagerImpl getEventManagerImpl() {
- return eventManager;
- }
-
- public File getWorkingDirectory() {
- return workingDir;
- }
-
- public ProxyNode getNode(final String nodeName) {
- return nodes.get(nodeName);
- }
-
- public Map<String, ProxyNode> getAllProxyNodes() {
- final Map<String, ProxyNode> result = new HashMap<String, ProxyNode>();
- for (final String nodeName : nodes.keySet()) {
- final ProxyNode node = nodes.get(nodeName);
- if (!(node instanceof RouterNode)) {
- result.put(nodeName, node);
- }
- }
- return result;
- }
-
- public Map<String, RouterNode> getAllRouterNodes() {
- final Map<String, RouterNode> result =
- new HashMap<String, RouterNode>();
- for (final String nodeName : nodes.keySet()) {
- final ProxyNode node = nodes.get(nodeName);
- if (node instanceof RouterNode && !(node instanceof DirectoryNode)) {
- result.put(nodeName, (RouterNode) node);
- }
- }
- return result;
- }
-
- public Map<String, DirectoryNode> getAllDirectoryNodes() {
- final Map<String, DirectoryNode> result =
- new HashMap<String, DirectoryNode>();
- for (final String nodeName : nodes.keySet()) {
- final ProxyNode node = nodes.get(nodeName);
- if (node instanceof DirectoryNode) {
- result.put(nodeName, (DirectoryNode) node);
- }
- }
- return result;
- }
-
- public Map<String, ProxyNode> getAllNodes() {
- return new HashMap<String, ProxyNode>(nodes);
- }
-
- public boolean hupUntilUp(final int tries, final long hupInterval)
- throws PuppeTorException {
-
- // log entering
- logger.entering(this.getClass().getName(), "hupUntilUp", new Object[] {
- tries, hupInterval });
-
- // check if all nodes are running
- for (final ProxyNode node : nodes.values()) {
- if (node.getNodeState() != NodeState.RUNNING) {
- final IllegalStateException e =
- new IllegalStateException(
- "All nodes must be running before sending them HUP "
- + "commands!");
- logger.throwing(this.getClass().getName(), "hupUntilUp", e);
- throw e;
- }
- }
-
- // check if nodes are already up; if so, return immediately
- if (allNodesUp()) {
-
- // log exiting and return true
- logger.exiting(this.getClass().getName(), "hupUntilUp", true);
- return true;
- }
-
- final Object hupLock = new Object();
- // create and register a new event handler for each node
- for (final ProxyNode node : nodes.values()) {
- eventManager.addEventListener(node.getNodeName(),
- new EventListener() {
- public void handleEvent(final Event event) {
- if (event.getType() == NodeEventType.NODE_CIRCUIT_OPENED) {
- synchronized(hupLock) {
- hupLock.notify();
- }
- eventManager.removeEventListener(this);
- }
- }
- });
- }
-
- // walk through wait-check-hup loop until there are no tries left
- for (int i = 0; i < tries; i++) {
-
- // determine how long to try waiting for the hangup
- final long endOfSleeping = System.currentTimeMillis() + hupInterval;
-
- // unless all nodes have reported to be up, wait for the given
- // maximum time
- while (System.currentTimeMillis() < endOfSleeping) {
-
- synchronized(hupLock) {
- // check if nodes are up now
- if (allNodesUp()) {
-
- // log exiting and return true
- logger.exiting(this.getClass().getName(), "hupUntilUp",
- true);
- return true;
- }
- // sleep
- try {
- hupLock.wait(hupInterval);
- } catch (final InterruptedException e) {
- // do nothing about it
- }
- }
- }
-
- logger.log(Level.FINE, "Sending HUP to nodes");
- // send a HUP signal to all nodes
- for (final ProxyNode node : nodes.values()) {
- logger
- .log(Level.FINE, "Sending HUP to node "
- + node.toString());
- node.hup();
- }
-
- // continue in loop
- }
-
- // no retries left and not all nodes are up; log exiting and return
- // failure
- logger.exiting(this.getClass().getName(), "hupUntilUp", false);
- return false;
- }
-
- public void hupAllNodes() throws PuppeTorException {
-
- // log entering
- logger.entering(this.getClass().getName(), "hupAllNodes");
-
- // check if all nodes are running
- for (final ProxyNode node : nodes.values()) {
- if (node.getNodeState() != NodeState.RUNNING) {
- final IllegalStateException e =
- new IllegalStateException(
- "All nodes must be running before sending them HUP "
- + "commands!");
- logger.throwing(this.getClass().getName(), "hupAllNodes", e);
- throw e;
- }
- }
-
- // send a HUP signal to all nodes
- for (final ProxyNode node : nodes.values()) {
- logger.log(Level.FINE, "Sending HUP to node " + node.toString());
- node.hup();
- }
-
- // no retries left and not all nodes are up; log exiting and return
- // failure
- logger.exiting(this.getClass().getName(), "hupAllNodes");
- }
-
- public void hupAllDirectories() throws PuppeTorException {
-
- // log entering
- logger.entering(this.getClass().getName(), "hupAllDirectories");
-
- // check if all directory nodes are running
- for (final ProxyNode node : nodes.values()) {
- if (node instanceof DirectoryNode
- && node.getNodeState() != NodeState.RUNNING) {
- final IllegalStateException e =
- new IllegalStateException(
- "All directory nodes must be running before sending "
- + "them HUP commands!");
- logger.throwing(this.getClass().getName(), "hupAllDirectories",
- e);
- throw e;
- }
- }
-
- // send a HUP signal to all nodes
- for (final ProxyNode node : nodes.values()) {
- if (node instanceof DirectoryNode) {
- logger
- .log(Level.FINE, "Sending HUP to node "
- + node.toString());
- node.hup();
- }
- }
-
- // no retries left and not all nodes are up; log exiting and return
- // failure
- logger.exiting(this.getClass().getName(), "hupAllDirectories");
- }
-
- public void shutdownNodes() throws PuppeTorException {
-
- // log entering
- logger.entering(this.getClass().getName(), "shutdownNodes");
-
- // iteratively shut down all running nodes; if an exception is caught,
- // continue shutting down the other nodes and throw the first exception
- // subsequently
- PuppeTorException firstCaughtException = null;
- for (final ProxyNode node : nodes.values()) {
- if (node.getNodeState() == NodeState.RUNNING) {
- try {
- node.shutdown();
- } catch (final PuppeTorException e) {
- if (firstCaughtException == null) {
- firstCaughtException = e;
- }
- }
- }
- }
-
- // if an exception was caught during shutting down nodes, throw the
- // first caught exception
- if (firstCaughtException != null) {
- logger.throwing(this.getClass().getName(), "shutdownNodes",
- firstCaughtException);
- throw firstCaughtException;
- }
-
- // log exiting
- logger.exiting(this.getClass().getName(), "shutdownNodes");
- }
-
- public boolean startNodes(final long maximumTimeToWaitInMillis)
- throws PuppeTorException {
-
- // log entering
- logger.entering(this.getClass().getName(), "startNodes",
- maximumTimeToWaitInMillis);
-
- // check node states
- for (final ProxyNode node : nodes.values()) {
- if (node.getNodeState() != NodeState.CONFIGURATION_WRITTEN) {
- final IllegalStateException e =
- new IllegalStateException(
- "All configurations must be written before starting "
- + "nodes!");
- logger.throwing(this.getClass().getName(), "startNodes", e);
- throw e;
- }
- }
-
- // check parameter
- if (maximumTimeToWaitInMillis < 0) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "startNodes", e);
- throw e;
- }
-
- // remember time when we begin starting the nodes
- final long before = System.currentTimeMillis();
-
- // start nodes in parallel
- final Set<NodeStarter> allNodeStarters = new HashSet<NodeStarter>();
- for (final ProxyNode node : nodes.values()) {
- final NodeStarter nodeStarter =
- new NodeStarter(node, maximumTimeToWaitInMillis);
- allNodeStarters.add(nodeStarter);
- nodeStarter.start();
- }
-
- // wait for all node starts to complete
- for (final NodeStarter nodeStarter : allNodeStarters) {
-
- // join node starts one after the other
- try {
- nodeStarter.join();
- } catch (final InterruptedException e) {
- // this happens?! we have some kind of problem here!
- logger.log(Level.WARNING,
- "Interrupt while joining node starter!");
-
- // log exiting and return false
- logger.exiting(this.getClass().getName(), "startNodes", false);
- return false;
- }
-
- // if any thread has caught an exception, throw that exception now
- final Exception caughtException = nodeStarter.caughtException;
- if (caughtException != null) {
- final PuppeTorException ex =
- new PuppeTorException("Exception while starting node "
- + nodeStarter.node.getNodeName(),
- caughtException);
- logger.throwing(this.getClass().getName(), "startNodes", ex);
- throw ex;
- }
-
- // if node start did not succeed in the given time, fail
- if (!nodeStarter.success) {
- logger.log(Level.WARNING,
- "Starting nodes was not successful in "
- + maximumTimeToWaitInMillis / 1000
- + " seconds.", networkName);
-
- // log exiting and return false
- logger.exiting(this.getClass().getName(), "startNodes", false);
- return false;
- }
- }
-
- // determine how long we took to start all nodes
- final long after = System.currentTimeMillis();
- logger.log(Level.FINE, "Starting nodes was successful and took "
- + (after - before) / 1000 + " seconds.", networkName);
-
- // log exiting and return true
- logger.exiting(this.getClass().getName(), "startNodes", true);
- return true;
- }
-
- @Override
- public String toString() {
- return this.getClass().getSimpleName() + ": networkName=\""
- + networkName;
- }
-
- public String getNetworkName() {
- return networkName;
- }
-
- public void writeConfigurations() throws PuppeTorException {
-
- // log entering
- logger.entering(this.getClass().getName(), "writeConfigurations");
-
- // write configurations for all nodes
- for (final ProxyNode node : nodes.values()) {
- node.writeConfiguration();
- }
-
- // log exiting
- logger.exiting(this.getClass().getName(), "writeConfigurations");
- }
-
- public void addTemplateConfiguration(
- final Class<? extends ProxyNode> nodeClass,
- final String templateConfigurationString) {
-
- // log entering
- logger.entering(this.getClass().getName(), "addTemplateConfiguration",
- new Object[] { nodeClass, templateConfigurationString });
-
- // check parameters
- if (nodeClass == null || templateConfigurationString == null
- || templateConfigurationString.length() < 1
- || !templateConfigurationString.contains(" ")) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(),
- "addTemplateConfiguration", e);
- throw e;
- }
-
- // add template string to appropriate template configuration
- if (nodeClass == ProxyNode.class) {
- ProxyNodeImpl.templateConfiguration
- .add(templateConfigurationString);
- } else if (nodeClass == RouterNode.class) {
- RouterNodeImpl.templateConfiguration
- .add(templateConfigurationString);
- } else if (nodeClass == DirectoryNode.class) {
- DirectoryNodeImpl.templateConfiguration
- .add(templateConfigurationString);
- } else {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(),
- "addTemplateConfiguration", e);
- throw e;
- }
-
- // log exiting
- logger.exiting(this.getClass().getName(), "addTemplateConfiguration");
- }
-
- public List<String> getTemplateConfiguration(
- final Class<? extends ProxyNode> nodeClass) {
-
- // log entering
- logger.entering(this.getClass().getName(), "getTemplateConfiguration",
- nodeClass);
-
- // check parameter
- if (nodeClass == null) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(),
- "getTemplateConfiguration", e);
- throw e;
- }
-
- // obtain reference on appropriate template configuration
- List<String> result = null;
- if (nodeClass == ProxyNode.class) {
- result = new ArrayList<String>(ProxyNodeImpl.templateConfiguration);
- } else if (nodeClass == RouterNode.class) {
- result =
- new ArrayList<String>(RouterNodeImpl.templateConfiguration);
- } else if (nodeClass == DirectoryNode.class) {
- result =
- new ArrayList<String>(
- DirectoryNodeImpl.templateConfiguration);
- } else {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(),
- "getTemplateConfiguration", e);
- throw e;
- }
-
- // log exiting and return result
- logger.exiting(this.getClass().getName(), "getTemplateConfiguration",
- result);
- return result;
- }
-
- public void removeTemplateConfiguration(
- final Class<? extends ProxyNode> nodeClass,
- final String templateConfigurationKey) {
-
- // log entering
- logger.entering(this.getClass().getName(),
- "removeTemplateConfiguration", new Object[] { nodeClass,
- templateConfigurationKey });
-
- // check parameters
- if (nodeClass == null || templateConfigurationKey == null
- || templateConfigurationKey.length() < 1) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(),
- "removeTemplateConfiguration", e);
- throw e;
- }
-
- // obtain reference on appropriate template configuration
- List<String> templateConfig = null;
- if (nodeClass == ProxyNode.class) {
- templateConfig = ProxyNodeImpl.templateConfiguration;
- } else if (nodeClass == RouterNode.class) {
- templateConfig = RouterNodeImpl.templateConfiguration;
- } else if (nodeClass == DirectoryNode.class) {
- templateConfig = DirectoryNodeImpl.templateConfiguration;
- } else {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(),
- "removeTemplateConfiguration", e);
- throw e;
- }
-
- // iterate over existing template configuration strings and remove all
- // configuration strings that have the given configuration key
- final List<String> configurationStringsToRemove =
- new ArrayList<String>();
- for (final String currentConfigurationString : templateConfig) {
- final String currentConfigurationKey =
- currentConfigurationString.substring(0,
- currentConfigurationString.indexOf(" "));
- if (currentConfigurationKey.equals(templateConfigurationKey)) {
- configurationStringsToRemove.add(currentConfigurationString);
- }
- }
- templateConfig.removeAll(configurationStringsToRemove);
-
- // log exiting
- logger
- .exiting(this.getClass().getName(),
- "removeTemplateConfiguration");
- }
-
- /**
- * Returns the current port number and increments it afterwards.
- *
- * @return The current port number.
- */
- int getNextPortNumber() {
- return portCounter++;
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/impl/ProxyNodeImpl.java b/src/de/uniba/wiai/lspi/puppetor/impl/ProxyNodeImpl.java
deleted file mode 100644
index b4417fa..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/impl/ProxyNodeImpl.java
+++ /dev/null
@@ -1,734 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.impl;
-
-import java.io.BufferedReader;
-import java.io.BufferedWriter;
-import java.io.File;
-import java.io.FileWriter;
-import java.io.IOException;
-import java.io.InputStreamReader;
-import java.net.Socket;
-import java.util.ArrayList;
-import java.util.Iterator;
-import java.util.List;
-import java.util.concurrent.atomic.AtomicInteger;
-import java.util.logging.Level;
-import java.util.logging.Logger;
-
-import net.freehaven.tor.control.TorControlConnection;
-import de.uniba.wiai.lspi.puppetor.HiddenService;
-import de.uniba.wiai.lspi.puppetor.NodeEventType;
-import de.uniba.wiai.lspi.puppetor.NodeState;
-import de.uniba.wiai.lspi.puppetor.ProxyNode;
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-
-/**
- * Implementation of <code>ProxyNode</code>.
- *
- * @author kloesing
- */
-public class ProxyNodeImpl implements ProxyNode {
-
- /**
- * Executable file containing Tor.
- *
- * TODO make this configurable!
- */
- protected static final File torExecutable = new File("tor");
-
- /**
- * The <code>torrc</code> configuration file of this Tor node.
- */
- protected File configFile;
-
- /**
- * Collects all configuration strings for this node during the configuration
- * phase in the order they are added.
- */
- protected List<String> configuration;
-
- /**
- * Connection via Tor controller.
- */
- protected TorControlConnection conn;
-
- /**
- * Port on which the Tor node will be listening for us as its controller.
- */
- protected int controlPort;
-
- /**
- * Event manager to which all events concerning this node are notified.
- */
- private final EventManagerImpl eventManager;
-
- /**
- * Logger for this node which is called "node." plus the name of this node.
- */
- protected Logger logger;
-
- /**
- * Network to which this node belongs.
- */
- protected NetworkImpl network;
-
- /**
- * Name of this node that is used as part of the working directory, as
- * logger name of this node, and as event source.
- */
- protected String nodeName;
-
- /**
- * The state of this node.
- */
- protected NodeState nodeState = NodeState.CONFIGURING;
-
- /**
- * Port on which the Tor node will be listening for SOCKS connection
- * requests.
- */
- protected int socksPort;
-
- /**
- * The running Tor process that belongs to this node.
- */
- protected Process torProcess;
-
- /**
- * Directory in which all information concerning this node is stored.
- */
- protected File workingDir;
-
- /**
- * Returns this node's working directory.
- *
- * @return This node's working directory.
- */
- File getWorkingDir() {
- return workingDir;
- }
-
- /**
- * Creates a new <code>ProxyNodeImpl</code> and adds it to the given
- * <code>network</code>, but does not yet write its configuration to disk
- * or start the corresponding Tor process.
- *
- * @param network
- * Network configuration to which this node belongs.
- * @param nodeName
- * The name of the new node which may only consist of between 1
- * and 19 alpha-numeric characters.
- * @param controlPort
- * Port on which the Tor node will be listening for us as its
- * controller. May not be negative or greater than 65535.
- * @param socksPort
- * Port on which the Tor node will be listening for SOCKS
- * connection requests. May not be negative or greater than
- * 65535.
- * @throws IllegalArgumentException
- * If at least one of the parameters is <code>null</code> or
- * has an invalid value.
- */
- ProxyNodeImpl(final NetworkImpl network, final String nodeName,
- final int controlPort, final int socksPort) {
-
- // make sure that nodeName is a valid logger name
- if (nodeName == null || nodeName.length() < 1 || nodeName.length() > 19
- || !nodeName.matches("[a-zA-Z0-9]*")) {
- final String reason =
- "\"" + nodeName + "\" is not a valid node name!";
- final IllegalArgumentException e =
- new IllegalArgumentException(reason);
- throw e;
- }
-
- // create logger
- logger = Logger.getLogger(nodeName + "." + this.getClass().getName());
-
- logger.setLevel(Level.ALL);
-
- // log entering
- logger.entering(this.getClass().getName(), "ProxyNodeImpl",
- new Object[] { network, nodeName, controlPort, socksPort });
-
- // check remaining parameters
- if (network == null || controlPort < 0 || controlPort > 65535
- || socksPort < 0 || socksPort > 65535) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "ProxyNodeImpl", e);
- throw e;
- }
-
- // store parameter values
- this.network = network;
- this.nodeName = nodeName;
- this.controlPort = controlPort;
- this.socksPort = socksPort;
-
- // obtain reference on event manager from network
- eventManager = network.getEventManagerImpl();
-
- // create working directory
- workingDir =
- new File(this.network.getWorkingDirectory().getAbsolutePath()
- + File.separator + nodeName + File.separator);
- workingDir.mkdirs();
- logger.log(Level.FINE, "Created working directory \""
- + workingDir.getAbsolutePath() + "\"");
-
- // create reference on config file
- configFile =
- new File(workingDir.getAbsolutePath() + File.separator
- + "torrc");
-
- // initialize configuration
- configuration = new ArrayList<String>(templateConfiguration);
- configuration.add("ControlPort " + controlPort);
- configuration.add("SocksPort " + socksPort);
-
- // initialize state
- nodeState = NodeState.CONFIGURING;
-
- // log exiting
- logger.exiting(this.getClass().getName(), "ProxyNodeImpl");
- }
-
- public void addConfiguration(final String configurationString) {
-
- // log entering
- logger.entering(this.getClass().getName(), "addConfiguration",
- configurationString);
-
- // check parameter
- if (configurationString == null || configurationString.length() < 1
- || !configurationString.contains(" ")) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "addConfiguration", e);
- throw e;
- }
-
- // add configuration string
- configuration.add(configurationString);
-
- // log exiting
- logger.exiting(this.getClass().getName(), "addConfiguration");
- }
-
- public void addConfigurations(final List<String> configurationStrings) {
-
- // log entering
- logger.entering(this.getClass().getName(), "addConfigurations",
- configurationStrings);
-
- // check parameter
- if (configurationStrings == null) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "addConfigurations", e);
- throw e;
- }
-
- // add configuration strings one by one
- for (final String conf : configurationStrings) {
- addConfiguration(conf);
- }
-
- // log exiting
- logger.exiting(this.getClass().getName(), "addConfigurations");
- }
-
- public void replaceConfiguration(final String configurationString) {
-
- // log entering
- logger.entering(this.getClass().getName(), "replaceConfiguration",
- configurationString);
-
- // check parameter
- if (configurationString == null || configurationString.length() < 1
- || !configurationString.contains(" ")) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "replaceConfiguration",
- e);
- throw e;
- }
-
- // extract configuration key
- final String configurationKey =
- configurationString.substring(0, configurationString
- .indexOf(" "));
-
- // iterate over existing configuration strings and replace the first
- // occurrence of configuration key with new configuration string
- final Iterator<String> it = configuration.listIterator();
- boolean replaced = false;
- for (int counter = 0; !replaced && it.hasNext(); counter++) {
- final String currentConfigurationString = it.next();
- final String currentConfigurationKey =
- currentConfigurationString.substring(0,
- currentConfigurationString.indexOf(" "));
- if (currentConfigurationKey.equals(configurationKey)) {
- configuration.set(counter, configurationString);
- replaced = true;
- }
- }
-
- // if no such configuration key was found, append the configuration
- // string
- if (!replaced) {
- configuration.add(configurationString);
- }
-
- // log exiting
- logger.exiting(this.getClass().getName(), "replaceConfiguration");
- }
-
- public void removeConfiguration(final String configurationKey) {
-
- // log entering
- logger.entering(this.getClass().getName(), "deleteConfiguration",
- configurationKey);
-
- // check parameter
- if (configurationKey == null || configurationKey.length() < 1) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger
- .throwing(this.getClass().getName(), "deleteConfiguration",
- e);
- throw e;
- }
-
- // iterate over existing configuration strings and remove all
- // configuration strings that have the given configuration key
- final List<String> configurationStringsToRemove =
- new ArrayList<String>();
- for (final String currentConfigurationString : configuration) {
- final String currentConfigurationKey =
- currentConfigurationString.substring(0,
- currentConfigurationString.indexOf(" "));
- if (currentConfigurationKey.equals(configurationKey)) {
- configurationStringsToRemove.add(currentConfigurationString);
- }
- }
- configuration.removeAll(configurationStringsToRemove);
-
- // log exiting
- logger.exiting(this.getClass().getName(), "deleteConfiguration");
- }
-
- public synchronized HiddenService addHiddenService(
- final String serviceName, final int servicePort,
- final int virtualPort) {
-
- // log entering
- logger.entering(this.getClass().getName(), "addHiddenService",
- new Object[] { serviceName, servicePort, virtualPort });
-
- // create hidden service object; parameter checking is done in
- // constructor
- final HiddenService result =
- new HiddenServiceImpl(this, serviceName, servicePort,
- virtualPort);
-
- // add hidden service using Tor controller
- configuration.add("HiddenServiceDir " + workingDir.getAbsolutePath()
- + File.separator + serviceName + "\nHiddenServicePort "
- + virtualPort + " 127.0.0.1:" + servicePort);
-
- // log exiting and return hidden service object
- logger.exiting(this.getClass().getName(), "addHiddenService", result);
- return result;
- }
-
- public synchronized HiddenService addHiddenService(
- final String serviceName, final int servicePort) {
-
- // log entering
- logger.entering(this.getClass().getName(), "addHiddenService",
- new Object[] { serviceName, servicePort });
-
- // invoke overloaded method
- final HiddenService result =
- this.addHiddenService(serviceName, servicePort, 80);
-
- // log exiting and return hidden service
- logger.exiting(this.getClass().getName(), "addHiddenService", result);
- return result;
- }
-
- public synchronized HiddenService addHiddenService(final String serviceName) {
-
- // log entering
- logger.entering(this.getClass().getName(), "addHiddenService",
- serviceName);
-
- // invoke overloaded method
- final HiddenService result =
- this.addHiddenService(serviceName, network.getNextPortNumber(),
- 80);
-
- // log exiting and return hidden service
- logger.exiting(this.getClass().getName(), "addHiddenService", result);
- return result;
- }
-
- public String getNodeName() {
- return nodeName;
- }
-
- public synchronized NodeState getNodeState() {
- return nodeState;
- }
-
- public synchronized void hup() throws PuppeTorException {
-
- // log entering
- logger.entering(this.getClass().getName(), "hup");
-
- // check state
- if (nodeState != NodeState.RUNNING || conn == null) {
- final IllegalStateException e =
- new IllegalStateException(
- "Cannot hup a process when it's not running or there is "
- + "no connection to its control port!");
- logger.throwing(this.getClass().getName(), "hup", e);
- throw e;
- }
-
- // send HUP signal to Tor process
- try {
- conn.signal("HUP");
- } catch (final IOException e) {
- final PuppeTorException ex =
- new PuppeTorException(
- "Could not send the command HUP to the Tor process!",
- e);
- logger.throwing(this.getClass().getName(), "hup", ex);
- throw ex;
- }
- // log exiting
- logger.exiting(this.getClass().getName(), "hup");
- }
-
- public synchronized void shutdown() throws PuppeTorException {
-
- // log entering
- logger.entering(this.getClass().getName(), "shutdown");
-
- // check state
- if (nodeState != NodeState.RUNNING) {
- final IllegalStateException e = new IllegalStateException();
- logger.throwing(this.getClass().getName(), "shutdown", e);
- throw e;
- }
-
- // we cannot simply kill the Tor process, because we have established a
- // controller connection to it which would interpret a closed socket as
- // failure and throw a RuntimeException
- try {
- conn.shutdownTor("SHUTDOWN");
- } catch (final IOException e) {
- final PuppeTorException ex =
- new PuppeTorException(
- "Could not send shutdown command to Tor process!",
- e);
- logger.throwing(this.getClass().getName(), "shutdown", ex);
- throw ex;
- }
-
- // change state
- nodeState = NodeState.SHUT_DOWN;
-
- // fire event
- eventManager.observeInternalEvent(System.currentTimeMillis(),
- getNodeName(), NodeEventType.NODE_STOPPED, "Node stopped.");
-
- // log exiting
- logger.exiting(this.getClass().getName(), "shutdown");
- }
-
- /**
- * Helper thread that waits for a given time for a given process to
- * potentially terminate in order to find out if there are problems. If
- * either the process terminates cleanly within this timeout, or does not
- * terminate, the exit value will be 0; otherwise it will contain the exit
- * code of the terminated process. This functionality is added, because it
- * is not provided by Process. XXX Some stuff in here looks still dodgy.
- * What happens if we get an InterruptedException in run and thus don't set
- * exitValue?-SH
- */
- private static class ProcessWaiter extends Thread {
-
- /** The process to wait for. */
- private final Process process;
-
- /** The exit value or 0 if the process is still running. */
- private final AtomicInteger exitValue;
-
- /**
- * Creates a new <code>ProcessWaiter</code> for process
- * <code>process</code>, but does not start it, yet.
- *
- * @param process
- * The process to wait for.
- */
- ProcessWaiter(final Process process) {
- this.process = process;
- exitValue = new AtomicInteger(0);
- }
-
- @Override
- public void run() {
- try {
- exitValue.set(process.waitFor());
- } catch (final InterruptedException e) {}
- }
-
- /**
- * Causes the current thread to wait until the process has terminated or
- * the <code>timeoutInMillis</code> has expired. This method returns
- * immediately if the subprocess has already terminated.
- *
- * @param timeoutInMillis
- * The maximum time to wait for the process to terminate.
- * @return The exit value of the terminated process or 0 if the process
- * is still running.
- */
- public int waitFor(final long timeoutInMillis) {
- try {
- sleep(timeoutInMillis);
- } catch (final InterruptedException e) {}
- interrupt();
- return exitValue.get();
- }
- }
-
- public synchronized boolean startNode(final long maximumTimeToWaitInMillis)
- throws PuppeTorException {
-
- // log entering
- logger.entering(this.getClass().getName(), "startNode",
- maximumTimeToWaitInMillis);
-
- // check state
- if (nodeState != NodeState.CONFIGURATION_WRITTEN) {
- final String reason =
- "Node is not in state "
- + "NodeState.CONFIGURATION_WRITTEN!";
- final IllegalStateException e = new IllegalStateException(reason);
- logger.throwing(this.getClass().getName(), "startNode", e);
- throw e;
- }
-
- // start process
- final ProcessBuilder processBuilder =
- new ProcessBuilder(torExecutable.getPath(), "-f", "torrc");
- processBuilder.directory(workingDir);
- processBuilder.redirectErrorStream(true);
- try {
- torProcess = processBuilder.start();
- logger.log(Level.FINE, "Started Tor process successfully!");
- } catch (final IOException e) {
- final String reason = "Could not start Tor process!";
- final PuppeTorException ex = new PuppeTorException(reason, e);
- logger.throwing(this.getClass().getName(), "startNode", ex);
- throw ex;
- }
-
- // start thread to parse output
- final BufferedReader br =
- new BufferedReader(new InputStreamReader(torProcess
- .getInputStream()));
- final Thread outputThread = new Thread() {
- @Override
- public void run() {
-
- // log entering
- logger.entering(this.getClass().getName(), "run");
-
- // read output from Tor to parse it
- String line = null;
- try {
- while ((line = br.readLine()) != null) {
- eventManager.observeUnparsedEvent(ProxyNodeImpl.this
- .getNodeName(), line);
- }
- } catch (IOException e) {
-
- // only print out a warning for this exception if this node
- // is running; otherwise, silently ignore it...
- if (getNodeState() == NodeState.RUNNING) {
- String reason =
- "IOException when reading output from Tor "
- + "process "
- + ProxyNodeImpl.this.getNodeName()
- + "!";
- logger.log(Level.WARNING, reason, e);
- }
- }
-
- // log exiting
- logger.exiting(this.getClass().getName(), "run");
- }
- };
- outputThread.setDaemon(true);
- outputThread.setName(nodeName + " Output Parser");
- outputThread.start();
- logger.log(Level.FINE, "Started thread to parse output!");
-
- // add shutdown hook that kills the process on JVM exit
- final Process p = torProcess;
- Runtime.getRuntime().addShutdownHook(new Thread() {
- @Override
- public void run() {
-
- // log entering
- logger.entering(this.getClass().getName(), "run");
-
- // destroy Tor process
- p.destroy();
-
- // log exiting
- logger.exiting(this.getClass().getName(), "run");
- }
- });
- logger.log(Level.FINER,
- "Started shutdown hook that will destroy the Tor process on "
- + "JVM exit!");
-
- // wait to see if the process is started or exited immediately; wait for
- // one second to be sure that Tor terminates if there is an error,
- // especially if the computer is very busy and many nodes are created
- final ProcessWaiter waiter = new ProcessWaiter(torProcess);
- waiter.start();
- final int exitValue = waiter.waitFor(1000);
- if (exitValue != 0) {
- // Tor did not manage to start correctly
- logger.log(Level.WARNING, "Could not start Tor process! Tor "
- + "exited with exit value " + exitValue
- + "! Please go check the config options in " + configFile
- + " manually!");
-
- // log exiting
- logger.exiting(this.getClass().getName(), "startNode", false);
- return false;
- }
-
- // wait for Tor to open the control port
- logger.log(Level.FINER, "Waiting for Tor to open its control port...");
- if (!eventManager.waitForAnyOccurence(nodeName,
- NodeEventType.NODE_CONTROL_PORT_OPENED,
- maximumTimeToWaitInMillis)) {
-
- // Tor did not open its control port
- logger.log(Level.WARNING, "Tor node " + nodeName
- + " did not manage to open its control port within "
- + maximumTimeToWaitInMillis + " milliseconds!");
-
- // log exiting
- logger.exiting(this.getClass().getName(), "startNode", false);
- return false;
- }
- logger.log(Level.FINE,
- "Tor has successfully opened its control port and told us "
- + "about that!");
-
- // connect to the controller
- logger.log(Level.FINER, "Connecting to control port...");
- try {
- final Socket controlSocket =
- new java.net.Socket("127.0.0.1", controlPort);
- conn = TorControlConnection.getConnection(controlSocket);
- conn.authenticate(new byte[0]);
- } catch (final IOException e) {
- final String reason =
- "Could not connect to control port " + controlPort + "!";
- final PuppeTorException ex = new PuppeTorException(reason, e);
- logger.throwing(this.getClass().getName(), "startNode", ex);
- throw ex;
- }
- logger.log(Level.FINE, "Connected to control port successfully!");
-
- // set state to RUNNING
- nodeState = NodeState.RUNNING;
-
- // fire event
- eventManager.observeInternalEvent(System.currentTimeMillis(),
- getNodeName(), NodeEventType.NODE_STARTED, "Node started.");
-
- // log exiting and return with success
- logger.exiting(this.getClass().getName(), "startNode", true);
- return true;
- }
-
- @Override
- public String toString() {
- return this.getClass().getSimpleName() + ": nodeName=\"" + nodeName
- + "\", controlPort=" + controlPort + ", socksPort=" + socksPort;
- }
-
- public synchronized void writeConfiguration() throws PuppeTorException {
-
- // log entering
- logger.entering(this.getClass().getName(), "writeConfiguration");
-
- // write config file
- try {
- final BufferedWriter bw =
- new BufferedWriter(new FileWriter(configFile));
- for (final String c : configuration) {
- bw.write(c + "\n");
- }
- bw.close();
- } catch (final IOException e) {
- final PuppeTorException ex =
- new PuppeTorException(
- "Could not write configuration file!", e);
- logger.throwing(this.getClass().getName(),
- "writeConfigurationFile", ex);
- throw ex;
- }
-
- // change state, if necessary
- if (nodeState == NodeState.CONFIGURING) {
- nodeState = NodeState.CONFIGURATION_WRITTEN;
- }
-
- // log exiting
- logger.exiting(this.getClass().getName(), "writeConfiguration");
- }
-
- public int getSocksPort() {
- return socksPort;
- }
-
- public int getControlPort() {
- return controlPort;
- }
-
- public List<String> getConfiguration() {
- return new ArrayList<String>(configuration);
- }
-
- /**
- * Template configuration of proxy nodes.
- */
- static List<String> templateConfiguration;
-
- static {
- templateConfiguration = new ArrayList<String>();
-
- templateConfiguration.add("DataDirectory .");
- templateConfiguration.add("SafeLogging 0");
- templateConfiguration.add("UseEntryGuards 0");
-
- templateConfiguration.add("Log info stdout");
- templateConfiguration.add("Log info file log");
-
- // TODO This is now contained in proposal 135.
- // templateConfiguration.add("EnforceDistinctSubnets 0");
- // templateConfiguration.add("ClientDNSRejectInternalAddresses 0");
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/impl/RouterNodeImpl.java b/src/de/uniba/wiai/lspi/puppetor/impl/RouterNodeImpl.java
deleted file mode 100644
index 166f8c6..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/impl/RouterNodeImpl.java
+++ /dev/null
@@ -1,402 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.impl;
-
-import java.io.BufferedReader;
-import java.io.BufferedWriter;
-import java.io.File;
-import java.io.FileReader;
-import java.io.FileWriter;
-import java.io.IOException;
-import java.util.ArrayList;
-import java.util.List;
-import java.util.logging.Level;
-import java.util.regex.Pattern;
-
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-import de.uniba.wiai.lspi.puppetor.RouterNode;
-
-/**
- * Implementation of <code>RouterNode</code>.
- *
- * @author kloesing
- */
-public class RouterNodeImpl extends ProxyNodeImpl implements RouterNode {
-
- /**
- * Internal thread class that is used to determine fingerprints in parallel,
- * which can take a few seconds.
- */
- private class FingerprintThread extends Thread {
-
- @Override
- public void run() {
-
- // log entering
- logger.entering(this.getClass().getName(), "run");
-
- // create file reference for temporary config file
- final File tempConfigFile =
- new File(workingDir.getAbsolutePath() + File.separator
- + "torrc.tmp");
-
- // compose a modified config file, including a DirServer option with
- // false fingerprint; this is necessary, because otherwise Tor
- // would not accept that this router node might have a private IP
- // address, but connects to the public directory servers
- final List<String> copyOfConfig =
- new ArrayList<String>(configuration);
- final String fakeDirServerString =
- "DirServer "
- + nodeName
- + " 127.0.0.1:"
- + dirPort
- + " 0000 0000 0000 0000 0000 0000 0000 0000 0000 0000";
- copyOfConfig.add(fakeDirServerString);
-
- // write config file
- try {
- final BufferedWriter bw =
- new BufferedWriter(new FileWriter(tempConfigFile));
- for (final String c : copyOfConfig) {
- bw.write(c + "\n");
- }
- bw.close();
- } catch (final IOException e) {
- final PuppeTorException ex =
- new PuppeTorException(
- "Could not write configuration file!", e);
- logger.log(Level.WARNING, "Could not start Tor process!", ex);
- setCaughtException(ex);
- return;
- }
-
- // start process with option --list-fingerprint
- final ProcessBuilder processBuilder =
- new ProcessBuilder(torExecutable.getPath(),
- "--list-fingerprint", "-f", "torrc.tmp");
- processBuilder.directory(workingDir);
- processBuilder.redirectErrorStream(true);
- Process tmpProcess = null;
- try {
- tmpProcess = processBuilder.start();
- } catch (final IOException e) {
- final PuppeTorException ex =
- new PuppeTorException(
- "Could not start Tor process temporarily with "
- + "--list-fingerprint option!", e);
- logger.log(Level.WARNING, "Could not start Tor process!", ex);
- setCaughtException(ex);
- return;
- }
-
- // wait for process to terminate
- int exitValue = 0;
- try {
- exitValue = tmpProcess.waitFor();
- } catch (final InterruptedException e) {
- final PuppeTorException ex =
- new PuppeTorException(
- "Interrupted while waiting for Tor process to exit!",
- e);
- logger.log(Level.WARNING,
- "Temporary Tor process was interrupted!", ex);
- setCaughtException(ex);
- return;
- }
-
- if (exitValue != 0) {
- final PuppeTorException ex =
- new PuppeTorException(
- "Could not start Tor process temporarily with "
- + "--list-fingerprint option! Tor exited with "
- + "exit value "
- + exitValue
- + "! Please go check the config options in "
- + tempConfigFile + " manually!");
- logger.log(Level.WARNING, "Could not start Tor process!", ex);
- setCaughtException(ex);
- return;
- }
-
- // read fingerprint from file
- final File fingerprintFile =
- new File(workingDir.getAbsolutePath() + File.separator
- + "fingerprint");
- try {
- final BufferedReader br2 =
- new BufferedReader(new FileReader(fingerprintFile));
- setFingerprint(br2.readLine());
- br2.close();
- } catch (final IOException e) {
- final PuppeTorException ex =
- new PuppeTorException(
- "Could not read fingerprint from file!", e);
- logger.log(Level.WARNING, "Could not read fingerprint file!",
- ex);
- setCaughtException(ex);
- return;
- }
-
- // delete temporary config file
- tempConfigFile.delete();
-
- // log exiting
- logger.exiting(this.getClass().getName(), "run");
- }
- }
-
- /**
- * Invoked by the fingerprint thread: sets the determined fingerprint string
- *
- * @param fingerprint
- * The determined fingerprint string.
- */
- private synchronized void setFingerprint(final String fingerprint) {
-
- // log entering
- logger.entering(this.getClass().getName(), "setFingerprint",
- fingerprint);
-
- // remember fingerprint and notify all waiting threads
- this.fingerprint = fingerprint;
- notifyAll();
-
- // log exiting
- logger.exiting(this.getClass().getName(), "setFingerprint");
- }
-
- /**
- * Invoked by the fingerprint thread: sets the exception that occurred when
- * trying to determine the fingerprint.
- *
- * @param caughtException
- * The exception that occurred when trying to determine the
- * fingerprint.
- */
- protected synchronized void setCaughtException(
- final PuppeTorException caughtException) {
-
- // log entering
- logger.entering(this.getClass().getName(), "setCaughtException",
- caughtException);
-
- // remember caught exception and notify all waiting threads
- this.caughtException = caughtException;
- notifyAll();
-
- // log exiting
- logger.exiting(this.getClass().getName(), "setCaughtException");
- }
-
- /**
- * Port on which the Tor node will be listening for directory requests from
- * other Tor nodes.
- */
- protected int dirPort;
-
- /**
- * The IP v4 address on which the node will listen in dotted decimal
- * notation.
- */
- protected String serverIpAddress;
-
- /**
- * The fingerprint of this node that is determined as hash value of its
- * onion key. It is initialized with <code>null</code> and set by the
- * fingerprint thread as soon as it is determined.
- */
- private String fingerprint;
-
- /**
- * The exception that was caught when determining the fingerprint of this
- * node, if any.
- */
- protected PuppeTorException caughtException;
-
- /**
- * The pattern for valid IP v4 addresses in dotted decimal notation.
- */
- private static final Pattern validIpAddressPattern =
- Pattern
- .compile("([0-9]|[1-9][0-9]|1[0-9][0-9]|2[0-4][0-9]|25[0-5])"
- + "(\\.([0-9]|[1-9][0-9]|1[0-9][0-9]|2[0-4][0-9]|25[0-5])){1,3}");
-
- /**
- * Port on which the Tor node will be listening for onion requests by other
- * Tor nodes.
- */
- protected int orPort;
-
- /**
- * Creates a new <code>RouterNodeImpl</code> and adds it to the given
- * <code>network</code>, but does not yet write its configuration to disk
- * or start the corresponding Tor process.
- *
- * @param network
- * Network configuration to which this node belongs.
- * @param nodeName
- * The name of the new node which may only consist of between 1
- * and 19 alpha-numeric characters.
- * @param controlPort
- * Port on which the Tor node will be listening for us as its
- * controller. May not be negative or greater than 65535.
- * @param socksPort
- * Port on which the Tor node will be listening for SOCKS
- * connection requests. May not be negative or greater than
- * 65535.
- * @param orPort
- * Port on which the Tor node will be listening for onion
- * requests by other Tor nodes. May not be negative or greater
- * than 65535.
- * @param dirPort
- * Port on which the Tor node will be listening for directory
- * requests from other Tor nodes. May not be negative or greater
- * than 65535.
- * @param serverIpAddress
- * The IP address on which the node will listen. Must be a valid
- * IP v4 address in dotted decimal notation. May not be
- * <code>null</code>.
- * @throws IllegalArgumentException
- * If at least one of the parameters is <code>null</code> or
- * has an invalid value.
- */
- RouterNodeImpl(final NetworkImpl network, final String nodeName,
- final int controlPort, final int socksPort, final int orPort,
- final int dirPort, final String serverIpAddress) {
-
- // create superclass instance; parameter checking is done in super
- // constructor
- super(network, nodeName, controlPort, socksPort);
-
- // log entering
- logger.entering(this.getClass().getName(), "RouterNodeImpl",
- new Object[] { network, nodeName, controlPort, socksPort,
- orPort, dirPort, serverIpAddress });
-
- // check parameters
- if (orPort < 0 || orPort > 65535 || dirPort < 0 || dirPort > 65535
- || serverIpAddress == null
- || !validIpAddressPattern.matcher(serverIpAddress).matches()) {
- final IllegalArgumentException e =
- new IllegalArgumentException("nodeName=" + nodeName
- + ", controlPort=" + controlPort + ", socksPort="
- + socksPort + ", orPort=" + orPort + ", dirPort="
- + dirPort + ", serverIpAddress='" + serverIpAddress
- + "'");
- logger.throwing(this.getClass().getName(), "RouterNodeImpl", e);
- throw e;
- }
-
- // remember parameters
- this.orPort = orPort;
- this.dirPort = dirPort;
- this.serverIpAddress = serverIpAddress;
-
- // extend configuration by template configuration of router nodes
- configuration.addAll(templateConfiguration);
-
- // add further configuration to make this node a router node
- configuration.add("ORPort " + orPort);
- configuration.add("Nickname " + nodeName);
-
- // all routers mirror the directory
- configuration.add("DirPort " + dirPort);
-
- // the address of this node should be manually specified and not guessed
- // by Tor
- configuration.add("Address " + serverIpAddress);
-
- // the OR port may only be contacted locally
- configuration.add("ORListenAddress " + serverIpAddress);
-
- // offer directory only locally (either by being an authority, or by
- // mirroring it)
- configuration.add("DirListenAddress " + serverIpAddress);
-
- // start a thread to determine the node's fingerprint in the background
- determineFingerprint();
-
- // log exiting
- logger.exiting(this.getClass().getName(), "RouterNodeImpl");
- }
-
- public synchronized String getFingerprint() throws PuppeTorException {
-
- // log entering
- logger.entering(this.getClass().getName(), "getFingerprint");
-
- // wait until either the fingerprint has been determined or an exception
- // was caught
- while (fingerprint == null && caughtException == null) {
- try {
- wait();
- } catch (final InterruptedException e) {
- // do nothing
- }
- }
-
- if (caughtException != null) {
- logger.throwing(this.getClass().getName(), "getFingerprint",
- caughtException);
- throw caughtException;
- }
-
- // log exiting
- logger
- .exiting(this.getClass().getName(), "getFingerprint",
- fingerprint);
- return fingerprint;
- }
-
- @Override
- public String toString() {
- return super.toString() + ", orPort=" + orPort + ", dirPort=" + dirPort;
- }
-
- public int getDirPort() {
- return dirPort;
- }
-
- public int getOrPort() {
- return orPort;
- }
-
- /**
- * Determines the fingerprint of this node by starting a background thread
- * that performs this operation.
- */
- protected synchronized void determineFingerprint() {
-
- // log entering
- logger.entering(this.getClass().getName(), "determineFingerprint");
-
- // start a thread to determine this node's fingerprint
- final FingerprintThread fingerprintThread = new FingerprintThread();
- fingerprintThread.setName(nodeName + " Fingerprint Resolver");
- fingerprintThread.start();
-
- // log exiting
- logger.exiting(this.getClass().getName(), "determineFingerprint");
- }
-
- /**
- * Template configuration of router nodes.
- */
- static List<String> templateConfiguration;
-
- static {
- templateConfiguration = new ArrayList<String>();
-
- templateConfiguration.add("ContactInfo wont@xxxxxxxxx");
- templateConfiguration.add("HidServDirectoryV2 1");
-
- // TODO This is now contained in proposal 135.
- // templateConfiguration.add("ExitPolicyRejectPrivate 0");
- // templateConfiguration.add("AssumeReachable 1");
- // templateConfiguration.add("ServerDNSAllowBrokenResolvConf 1");
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/impl/ServerApplicationImpl.java b/src/de/uniba/wiai/lspi/puppetor/impl/ServerApplicationImpl.java
deleted file mode 100644
index 5ac0ea6..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/impl/ServerApplicationImpl.java
+++ /dev/null
@@ -1,351 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.impl;
-
-import java.io.BufferedReader;
-import java.io.IOException;
-import java.io.InputStreamReader;
-import java.io.PrintStream;
-import java.net.ServerSocket;
-import java.net.Socket;
-import java.util.logging.Level;
-import java.util.logging.Logger;
-
-import de.uniba.wiai.lspi.puppetor.ServerApplication;
-import de.uniba.wiai.lspi.puppetor.ServerEventType;
-
-/**
- * Implementation of <code>ServerApplication</code>.
- *
- * @author kloesing
- */
-public class ServerApplicationImpl implements ServerApplication {
-
- /**
- * Internal thread class that is used to process an incoming request.
- */
- private class HandlerThread extends Thread {
-
- /**
- * Accepted socket on which the request came in.
- */
- private Socket handleSocket = null;
-
- /**
- * Creates a new thread to handle the request coming in on
- * <code>handleSocket</code>, but does not start handling it.
- *
- * @param handleSocket
- * Accepted socket on which the request came in.
- */
- public HandlerThread(final Socket handleSocket) {
-
- // log entering
- logger.entering(this.getClass().getName(), "HandlerThread",
- handleSocket);
-
- // remember parameter
- this.handleSocket = handleSocket;
-
- // log exiting
- logger.exiting(this.getClass().getName(), "HandlerThread");
- }
-
- @Override
- public void run() {
-
- // log entering
- logger.entering(this.getClass().getName(), "run");
-
- try {
-
- // wait for request (don't mind the content)
- final BufferedReader in =
- new BufferedReader(new InputStreamReader(handleSocket
- .getInputStream()));
- in.read();
-
- // send event to event manager
- eventManager.observeInternalEvent(System.currentTimeMillis(),
- getServerApplicationName(),
- ServerEventType.SERVER_RECEIVING_REQUEST_SENDING_REPLY,
- "Receiving request.");
-
- // write response
- final PrintStream out =
- new PrintStream(handleSocket.getOutputStream());
- out.print("HTTP/1.0 200 OK\r\n");
-
- } catch (final IOException e) {
- logger.log(Level.SEVERE,
- "I/O exception while handling incoming request!");
- // we can't do more, because nobody takes notice of this thread.
-
- // log exiting
- logger.exiting(this.getClass().getName(), "run");
- return;
- // TODO do we need more?
- }
-
- // close socket
- try {
- handleSocket.close();
- } catch (final IOException e) {
- logger
- .log(Level.WARNING,
- "I/O exception while closing socket!");
-
- // log exiting
- logger.exiting(this.getClass().getName(), "run");
- return;
- }
-
- // log exiting
- logger.exiting(this.getClass().getName(), "run");
- }
- }
-
- /**
- * Internal thread class that is used to listen for requests.
- */
- private class ListenThread extends Thread {
-
- /**
- * Flag to remember whether this thread listens for requests at the
- * moment (<code>true</code>), or has been stopped (<code>false</code>).
- */
- private boolean connected;
-
- /**
- * Creates a new thread to listen for requests, but does not start
- * listening, yet.
- */
- ListenThread() {
-
- // log entering
- logger.entering(this.getClass().getName(), "ListenThread");
-
- // start connected
- connected = true;
-
- // log exiting
- logger.exiting(this.getClass().getName(), "ListenThread");
- }
-
- @Override
- public void run() {
-
- // log entering
- logger.entering(this.getClass().getName(), "run");
-
- try {
-
- // create server socket
- ServerSocket serverSocket = null;
- try {
- serverSocket = new ServerSocket(serverPort);
- } catch (final IOException ioe) {
- logger.log(Level.SEVERE,
- "Can't open server socket on port " + serverPort
- + "!");
-
- // log exiting
- logger.exiting(this.getClass().getName(), "run");
- return;
- }
-
- // as long as we are connected, accept incoming requests
- logger.log(Level.FINE, "Listening on port " + serverPort
- + "...");
- while (connected) {
- Socket incomingConnection = null;
- try {
- incomingConnection = serverSocket.accept();
- } catch (final Exception e) {
- logger
- .log(
- Level.SEVERE,
- "Exception while accepting socket requests! Stopping listening!",
- e);
- break;
- }
- new HandlerThread(incomingConnection).start();
- }
-
- } catch (final Exception e) {
-
- // log that we have been interrupted
- logger.log(Level.WARNING, "Server has been interrupted!", e);
- }
-
- // mark as disconnected
- connected = false;
-
- // log exiting
- logger.exiting(this.getClass().getName(), "run");
- }
-
- /**
- * Stops listening on server socket.
- */
- public void stopListening() {
-
- // log entering
- logger.entering(this.getClass().getName(), "stopListening");
-
- // change connected state to false and interrupt thread
- connected = false;
- interrupt();
-
- // log exiting
- logger.exiting(this.getClass().getName(), "stopListening");
- }
- }
-
- /**
- * Event manager that handles all events concerning this server application.
- */
- private final EventManagerImpl eventManager;
-
- /**
- * Logger for this server which is called "server." plus the name of this
- * server application.
- */
- private final Logger logger;
-
- /**
- * Name of this server application that is used as logger name of this node.
- */
- private final String serverApplicationName;
-
- /**
- * Port on which this server will listen for incoming requests.
- */
- private final int serverPort;
-
- /**
- * Thread that listens for requests in the background.
- */
- private Thread serverThread;
-
- /**
- * Creates a new HTTP server application within this JVM, but does not yet
- * listen for incoming requests.
- *
- * @param network
- * Network to which this HTTP server belongs; at the moment this
- * is only used to determine the event manager instance.
- * @param serverApplicationName
- * Name of this server that is used as part of the logger name.
- * @param serverPort
- * Port on which this server will listen for incoming requests.
- * @throws IllegalArgumentException
- * If at least one of the parameters is <code>null</code> or
- * has an invalid value.
- */
- ServerApplicationImpl(final NetworkImpl network,
- final String serverApplicationName, final int serverPort) {
-
- // check if serverApplicationName can be used as logger name
- if (serverApplicationName == null
- || serverApplicationName.length() == 0) {
- throw new IllegalArgumentException(
- "Invalid serverApplicationName: " + serverApplicationName);
- }
-
- // create logger
- logger = Logger.getLogger("server." + serverApplicationName);
-
- // log entering
- logger.entering(this.getClass().getName(), "ServerApplicationImpl",
- new Object[] { network, serverApplicationName, serverPort });
-
- // check parameters
- if (network == null || serverPort < 0 || serverPort > 65535) {
- final IllegalArgumentException e = new IllegalArgumentException();
- logger.throwing(this.getClass().getName(), "ServerApplicationImpl",
- e);
- throw e;
- }
-
- // remember parameters
- this.serverApplicationName = serverApplicationName;
- this.serverPort = serverPort;
-
- // obtain reference on event manager
- eventManager = network.getEventManagerImpl();
-
- // log exiting
- logger.exiting(this.getClass().getName(), "ServerApplicationImpl");
- }
-
- public synchronized void startListening() {
-
- // log entering
- logger.entering(this.getClass().getName(), "listen");
-
- // check if we are already listening
- if (serverThread != null) {
- final IllegalStateException e =
- new IllegalStateException("We are already listening!");
- logger.throwing(this.getClass().getName(), "listen", e);
- throw e;
- }
-
- // create a thread that listens in the background
- serverThread = new ListenThread();
- serverThread.setName("Reply Thread");
- serverThread.setDaemon(true);
- serverThread.start();
-
- // log exiting
- logger.exiting(this.getClass().getName(), "listen");
- }
-
- public synchronized void stopListening() {
-
- // log entering
- logger.entering(this.getClass().getName(), "stopListening");
-
- // check if we are listening
- if (serverThread == null) {
- final IllegalStateException e =
- new IllegalStateException("We are not listening!");
- logger.throwing(this.getClass().getName(), "stopListening", e);
- throw e;
- }
-
- // log this event
- logger.log(Level.FINE, "Shutting down server");
-
- // interrupt thread
- serverThread.interrupt();
-
- // unset listening thread
- serverThread = null;
-
- // log exiting
- logger.exiting(this.getClass().getName(), "stopListening");
- }
-
- public synchronized boolean isListening() {
- return serverThread != null;
- }
-
- @Override
- public String toString() {
- return this.getClass().getSimpleName() + ": serverApplicationName=\""
- + serverApplicationName + "\", serverPort=" + serverPort;
- }
-
- public String getServerApplicationName() {
- return serverApplicationName;
- }
-
- public int getServerPort() {
- return serverPort;
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/AbstractMasterFactory.java b/src/de/uniba/wiai/lspi/puppetor/rmi/AbstractMasterFactory.java
deleted file mode 100644
index 220ab7e..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/AbstractMasterFactory.java
+++ /dev/null
@@ -1,78 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi;
-
-import java.rmi.RemoteException;
-
-import de.uniba.wiai.lspi.puppetor.rmi.tests.TestRegistration;
-
-/**
- * Factory to create all objects needed by the master side of a PuppeTor
- * network.
- *
- * @author Sebastian Hahn
- */
-public abstract class AbstractMasterFactory {
-
- /**
- * Hold the concrete RemotePuppeTorFactory that will be used to create the
- * implementation classes.
- */
- private static AbstractMasterFactory factory;
-
- /**
- * @return a new concrete
- * <code>AbstractRemotePuppeTorFactory<code>subclass as
- * specified by the initialization
- */
- final public synchronized static AbstractMasterFactory getInstance() {
- return factory;
- }
-
- /**
- * @param factory
- * save this as the factory if this hasn't been called before.
- */
- final public synchronized static void initialize(
- final AbstractMasterFactory factory) {
- if (AbstractMasterFactory.factory != null) {
- throw new RuntimeException("Don't call initialize twice!");
- }
- AbstractMasterFactory.factory = factory;
- }
-
- /**
- * Override this to create a subclass of <code>RemotePuppeTor</code>
- *
- * @param slave
- * Create the server represantation for this connected slave
- * @return The new <code>RemotePuppeTor</code> instance
- */
- public abstract Master createMaster(Slave slave) throws RemoteException;
-
- /**
- * Override this to create a subclass of <code>TestExecutor</code>.
- *
- * @return The <code>TestExecutor</code> instance.
- */
- public abstract TestExecutor getTestExecutorInstance();
-
- /**
- * Override this to create a subclass of <code>TestRegistration</code>.
- *
- * @return The <code>TestRegistration</code> instance.
- */
- public abstract TestRegistration getTestRegistrationInstance();
-
- /**
- * XXX just for now so I don't forget.-SH
- */
- public abstract NetworkDescription createNetworkDescription();
-
-
-
-// public abstract Network createNetwork(String name);
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/AbstractSlaveFactory.java b/src/de/uniba/wiai/lspi/puppetor/rmi/AbstractSlaveFactory.java
deleted file mode 100644
index 45e7f0e..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/AbstractSlaveFactory.java
+++ /dev/null
@@ -1,88 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi;
-
-import java.util.Set;
-
-import de.uniba.wiai.lspi.puppetor.rmi.Slave.OS;
-
-/**
- * Create a new representation of the server. Initialize this once with a
- * concrete subclass of itself to create the actual <code>RemotePuppeTor</code>
- * instance.
- *
- * @author Sebastian Hahn
- */
-public abstract class AbstractSlaveFactory {
-
- /**
- * Hold the concrete RemotePuppeTorFactory that will be used to create
- * <code>Slave</code> instances. Uses
- * <code>AbstractSlaveFactory.class</code> for locking.
- */
- private static AbstractSlaveFactory factory;
-
- /**
- * Hold the remote master object. We only have one per slave. XXX Maybe this
- * entire thing is more like a SlaveManager, not a factory. Oh well. Rename
- * later, I guess -SH
- */
- private static RemoteMaster master;
-
- /**
- * @return a new concrete <code>AbstractSlaveFactory<code>subclass as
- * specified by the initialization
- */
- final public synchronized static AbstractSlaveFactory getInstance() {
- return factory;
- }
-
- /**
- * @param factory
- * save this as the factory if this hasn't been called before.
- * XXX - We could throw an exception here, too. I will think
- * about that.-SH
- */
- final public synchronized static void initialize(
- final AbstractSlaveFactory factory) {
- if (AbstractSlaveFactory.factory == null) {
- AbstractSlaveFactory.factory = factory;
- }
- }
-
- /**
- * Register a new master. Note that calling this twice for different master
- * doesn't indicate an error, but trying to to set the same master again is
- * a bug. Setting this to null means the connection to the master was
- * closed.
- *
- * @param master
- * The master that should be used from now on.
- */
- final public synchronized static void setMaster(final RemoteMaster master) {
- AbstractSlaveFactory.master = master;
- }
-
- /**
- * @return the RemoteMaster currently registered.
- */
- final public synchronized static RemoteMaster getMaster() {
- if (AbstractSlaveFactory.master == null) {
- throw new IllegalStateException("No master set yet!");
- }
- return AbstractSlaveFactory.master;
- }
-
- /**
- * Override this to create a subclass of <code>Slave</code>
- *
- * @return The new <code>Slave</code> instance
- */
- public abstract Slave createSlave(final String slaveName, final OS os,
- final String ip, final Set<Integer> ports,
- final Set<Integer> availableRevisions) throws NullPointerException;
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/LocalMaster.java b/src/de/uniba/wiai/lspi/puppetor/rmi/LocalMaster.java
deleted file mode 100644
index 54afc42..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/LocalMaster.java
+++ /dev/null
@@ -1,36 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi;
-
-/**
- * Represent a connection between the locally running PuppeTor network Master
- * and a slave. This is the master part of the <code>Master</code> interface.
- *
- * @author Sebastian Hahn
- */
-public interface LocalMaster {
- /**
- * @return the slave instance this connection refers to.
- */
- public Slave getSlave();
-
- /**
- * Add a new AbstractTaskImpl for the connected slave.
- *
- * @param abstractTaskImpl
- */
- public void addTask(Task task);
-
- /**
- * @param taskId
- * The task id that we want to get
- * @return Get the TaskResult for the Task with the specified taskId.
- * @throws InterruptedException
- * if interrupted while waiting for the next TestResult
- */
- public TaskResult getTaskResult(int taskId) throws InterruptedException;
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/Master.java b/src/de/uniba/wiai/lspi/puppetor/rmi/Master.java
deleted file mode 100644
index 2e722da..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/Master.java
+++ /dev/null
@@ -1,15 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi;
-
-/**
- * Any subclass implementing this must also implement <code>RemoteMaster</code>
- * and <code>LocalMaster</code>. XXX This isn't a good way to do it. Think
- * about it more later-SH & Karsten
- *
- * @author Sebastian Hahn
- */
-public interface Master {}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/MasterConnector.java b/src/de/uniba/wiai/lspi/puppetor/rmi/MasterConnector.java
deleted file mode 100644
index 446da5d..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/MasterConnector.java
+++ /dev/null
@@ -1,31 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi;
-
-import java.rmi.Remote;
-import java.rmi.RemoteException;
-
-/**
- * Handle the initial handshake with the slaves and pass them back an object
- * that the slave uses to refer to the master. The master will create exactly
- * one <code>MasterConnector</code>, slaves don't ever need one.
- *
- * @author Sebastian Hahn
- */
-public interface MasterConnector extends Remote {
-
- /**
- * Called once by every connecting client so that the master knows about it
- * and has a chance to pass back a representation of itself.
- *
- * @param slave
- * A slave representation as passed from the connecting slave
- * @return The representation of the master
- * @throws RemoteException
- */
- public RemoteMaster registerClient(final Slave slave)
- throws RemoteException;
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/Network.java b/src/de/uniba/wiai/lspi/puppetor/rmi/Network.java
deleted file mode 100644
index 83605d3..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/Network.java
+++ /dev/null
@@ -1,79 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi;
-
-import java.io.Serializable;
-import java.util.List;
-import java.util.Set;
-
-/**
- * This class describes all Tor instances on the slaves that are used to form a
- * private Tor network, or the Tor instances that are added to the public
- * network and are used in a set of tests.
- *
- * @author Sebastian Hahn
- */
-public interface Network extends Serializable {
-
- /**
- * @param torInstance
- * add this <code>TorInstance</code> to the network.
- */
- public void addTorInstance(TorInstance torInstance);
-
- /**
- * @return this network's name, mainly to identify it in logs.
- */
- public String getName();
-
- /**
- * @param test
- * add this <code>test</code> to the network.
- */
- public void addTest(Test test);
-
- /**
- * @return a link for each slave that is part of this network.
- */
- public Set<LocalMaster> getMasters();
-
- /**
- * @return all Tor instances that are authorities
- */
- public Set<TorInstance> getDirectoryAuthorities();
-
- /**
- * @return all Tor instances that are routers, but not authorities
- */
- public Set<TorInstance> getRoutersOnly();
-
- /**
- * @return all Tor instances that are routers (including authorities)
- */
- public Set<TorInstance> getRouters();
-
- /**
- * @return all Tor instances that are proxis (but not authorities or
- * routers)
- */
- public Set<TorInstance> getProxiesOnly();
-
- /**
- * @return all Tor instances that are proxies (including authorities and
- * routers)
- */
- public Set<TorInstance> getProxies();
-
- /**
- * @return all Tests that can run on this network -SH
- */
- public List<Test> getTests();
-
- /**
- * Is this a private network or part of the public one?
- */
- public boolean isPrivateNetwork();
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/NetworkDescription.java b/src/de/uniba/wiai/lspi/puppetor/rmi/NetworkDescription.java
deleted file mode 100644
index a3f9949..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/NetworkDescription.java
+++ /dev/null
@@ -1,16 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi;
-
-import java.io.Serializable;
-
-public interface NetworkDescription extends Serializable {
- public int necessaryProxies();
-
- public int necessaryRouters();
-
- public int necessaryDirectoryAuthorities();
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/RemoteMaster.java b/src/de/uniba/wiai/lspi/puppetor/rmi/RemoteMaster.java
deleted file mode 100644
index 4384485..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/RemoteMaster.java
+++ /dev/null
@@ -1,47 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi;
-
-import java.rmi.Remote;
-import java.rmi.RemoteException;
-
-/**
- * <code>RemotePuppeTor</code> is the main interface that drives the
- * interaction between the interconnected master and its slave instances. It is
- * published by the master's instance, and represents the view of the master a
- * slave has. It provides ways for the slave to register a <code>Slave</code>
- * to provide information about its capabilities, as well as a way for the slave
- * to poll the master for new work and then report back results. XXX We will
- * need a way to allow the master to access this, as well as a way for clients
- * with open ports to register for callback support instead of polling for new
- * tasks =SH
- *
- * @author Sebastian Hahn
- */
-public interface RemoteMaster extends Remote {
-
- /**
- * Slaves that must poll because they are not reachable can use this method
- * to ask the master for new work.
- *
- * @throws RemoteException
- * RMI...
- * @throws InterruptedException
- * when interrupted while waiting for the next job to arrive.
- */
- public Task getNewTask() throws RemoteException, InterruptedException;
-
- /**
- * Report back the result of the task to the master. If possible, add an
- * error message. XXX Maybe down the road an error object is
- * needed/wanted-SH
- *
- * @throws RemoteException
- * RMI
- */
- public void reportTaskResult(TaskResult task) throws RemoteException,
- InterruptedException;
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/Slave.java b/src/de/uniba/wiai/lspi/puppetor/rmi/Slave.java
deleted file mode 100644
index f6fb417..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/Slave.java
+++ /dev/null
@@ -1,51 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi;
-
-import java.io.Serializable;
-import java.net.InetAddress;
-import java.util.Set;
-
-/**
- * A <code>Slave</code> represents a slave instance ready for tests. It
- * carries information about the testing capabilities of a given slave. A slave
- * may not provide information except for its name, which is always mandatory.
- * The slave will not be used for non-client traffic, though.
- *
- * @author Sebastian Hahn
- */
-public interface Slave extends Serializable {
-
- public enum OS {
- UNDEFINED, LINUX_I86_32, LINUX_I86_64, MAC_OSX_PPC, MAC_OSX_I86,
- WINDOWS_XP_32, WINDOWS_XP_64, WINDOWS_VISTA_32, WINDOWS_VISTA_64
- }
-
- /**
- * @return the slave's name.
- */
- public String getName();
-
- /**
- * @return the slave's operating system as defined in <code>Slave.OS</code>.
- */
- public OS getOS();
-
- /**
- * @return the slave's IP address. XXX Test this with IPv6-SH
- */
- public InetAddress getIP();
-
- /**
- * @return the ports this slave can open
- */
- public Set<Integer> openablePorts();
-
- /**
- * @return Tor revisions available on the slave
- */
- public Set<Integer> availableRevisions();
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/Task.java b/src/de/uniba/wiai/lspi/puppetor/rmi/Task.java
deleted file mode 100644
index e57a438..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/Task.java
+++ /dev/null
@@ -1,38 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi;
-
-import java.io.Serializable;
-import java.rmi.RemoteException;
-import java.util.Collection;
-
-public interface Task extends Serializable {
-
- /**
- * Execute the task. Must be called by the slave.
- *
- * @throws InterruptedException
- * if interrupted while the task is running.
- * @throws RemoteException
- * RMI
- */
- public void execute() throws InterruptedException, RemoteException;
-
- /**
- * Returns this tasks id. Task ids must be globally unique and larger than 0
- * or 0 to indicate a task that was sent out by the slave, not the master.
- */
- public int getId();
-
- /**
- * This method will be called by the <code>TestExecutor</code> to allow
- * tasks to collect additional information if they need it.
- * @param additionalInformation
- * offer this information.
- */
- public void addAdditionalInformation(
- Collection<String> additionalInformation);
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/TaskExecutionNotSuccessfulException.java b/src/de/uniba/wiai/lspi/puppetor/rmi/TaskExecutionNotSuccessfulException.java
deleted file mode 100644
index d287fdb..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/TaskExecutionNotSuccessfulException.java
+++ /dev/null
@@ -1,23 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi;
-
-import java.util.Set;
-
-public class TaskExecutionNotSuccessfulException extends Exception {
- private static final long serialVersionUID = 1L;
- final private Set<TaskResult> taskResults;
-
- public TaskExecutionNotSuccessfulException(final Set<TaskResult> taskResults) {
- this.taskResults = taskResults;
- }
-
- public Set<TaskResult> getTaskResults() {
- return taskResults;
- }
-
- // XXX Add some logging capabilities -SH
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/TaskResult.java b/src/de/uniba/wiai/lspi/puppetor/rmi/TaskResult.java
deleted file mode 100644
index e707e65..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/TaskResult.java
+++ /dev/null
@@ -1,39 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi;
-
-import java.io.Serializable;
-
-/**
- * After the completion of a <code>Task</code>, a <code>TaskResult</code> is
- * created to report the result of the task as well as logging information to
- * the Master.
- *
- * @author Sebastian Hahn
- */
-public interface TaskResult extends Serializable {
-
- /**
- * @return
- * the task that we have a result for
- */
- public Task getTask();
-
- /**
- * @return
- * <code>true</code> if the task execution was successful,
- * false otherwise.
- */
- public boolean wasSuccessful();
-
- /**
- * A <code>Task</code> may optionally pass back a result that can be used
- * by the Master to manage further Task execution.
- * @return
- * the result of this task or null if no result is set.
- */
- public String getTaskResult();
-}
\ No newline at end of file
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/Test.java b/src/de/uniba/wiai/lspi/puppetor/rmi/Test.java
deleted file mode 100644
index f67842f..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/Test.java
+++ /dev/null
@@ -1,37 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi;
-
-import java.io.Serializable;
-
-/**
- * <code>Test</code> describes a test and the kind of slave it can run on, as
- * well as other network settings that are required for the test to work. A test
- * should be able to tell whether it was successfully run or not when passed a
- * TestResult.
- *
- * @author Sebastian Hahn
- */
-public interface Test extends Serializable {
- /**
- * @return
- * a description of what this test is useful for.
- */
- public String getDescription();
-
- /**
- * @return
- * a description of the network that this test requires to run, so the
- * </code>TestExecutor</code> can pick or set up one.
- */
- public NetworkDescription getNetworkDescription();
-
- /**
- * Call this when the network has been set up and you want to run the test
- * with the current network.
- */
- public void doTest();
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/TestExecutor.java b/src/de/uniba/wiai/lspi/puppetor/rmi/TestExecutor.java
deleted file mode 100644
index b17dc0d..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/TestExecutor.java
+++ /dev/null
@@ -1,114 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi;
-
-import java.util.Collection;
-import java.util.Map;
-import java.util.Set;
-
-/**
- * A master uses the <code>TestExecutor</code> instance to load all registered
- * tests and check whether the currently running PuppeTor network can satisfy
- * the precondition necessary to run the test. If yes, it sets up the testing
- * environment and passes <code>PuppeTorJob</code>s to the slaves as the test
- * instructs it to do. There must be only a single instance of
- * <code>TestExecutor</code> at any time.
- *
- * @author Sebastian Hahn
- */
-public interface TestExecutor {
-
- /**
- * Register a test with the executor so it will consider it for execution.
- * Accept all submitted tests even though some may not be executable at the
- * moment of submission.
- */
- public void registerTest(Test test);
-
- /**
- * Start executing tests. Callers should make sure that the PuppeTor network
- * that is supposed to be used is ready.
- */
- public void startTesting();
-
- /**
- * Return an immutable Map of the nodes that have been run successfully
- * along with the results of the tests.
- */
- public Map<Test, TestResult> getSuccessfulTests();
-
- /**
- * Return an immutable Map of the nodes that have been run but didn't
- * complete successfully along with the results of the tests.
- */
- public Map<Test, TestResult> getFailedTests();
-
- /**
- * Return all tests that could not be executed because the PuppeTorNetwork
- * could not match a test's preconditions.
- */
- public Set<Test> getCannotRunTests();
-
- /**
- * Register a newly connected PuppeTor slave via its
- * <code>LocalMaster</code> connection object.
- */
- public void registerLocalMaster(LocalMaster master);
-
- /**
- * Unregister a PuppeTor slave. Called when a slave disconnects.
- */
- public void unregisterLocalMaster(LocalMaster master);
-
- /**
- * @return
- * the currently running network.
- */
- public Network getCurrentNetwork();
-
- /**
- * Execute the passed <code>Task</code> once per passed
- * <code>LocalMaster</code>. If a <code>LocalMaster</code> is passed
- * more than once, the task will be executed multiple times.
- *
- * @param masters
- * execute the task for those masters.
- * @param taskClass
- * create a task of this class and execute it.
- * @param taskResults
- * place the results of the task in this <code>Set</code> unless
- * <code>null</code> is passed.
- * @param additionalInformation TODO
- * @throws TaskExecutionNotSuccessfulException
- * Indicates a failure during task execution
- */
- public void executeTaskForMasters(final Collection<LocalMaster> masters,
- final Class<? extends Task> taskClass, final Set<TaskResult> taskResults,
- Collection<String> additionalInformation)
- throws TaskExecutionNotSuccessfulException;
-
- /**
- * Execute the passed <code>Task</code> once per passed
- * <code>TorInstance</code>. If a <code>TorInstance</code> is passed
- * more than once, the task will be executed multiple times.
- * @param taskClass
- * create a task of this class and execute it.
- * @param taskResults
- * place the results of the task in this <code>Set</code> unless
- * <code>null</code> is passed.
- * @param additionalInformation
- * call the task's addAdditionalInformation method with this
- * parameter.
- * @param torInstances
- * execute the task for those masters.
- * @throws TaskExecutionNotSuccessfulException
- * Indicates a failure during task execution
- */
- public void executeTaskForTorInstances(Collection<TorInstance> instances,
- Class<? extends Task> taskClass, Set<TaskResult> taskResults,
- Collection<String> additionalInformation)
- throws TaskExecutionNotSuccessfulException;
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/TestResult.java b/src/de/uniba/wiai/lspi/puppetor/rmi/TestResult.java
deleted file mode 100644
index 85e1b9f..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/TestResult.java
+++ /dev/null
@@ -1,15 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi;
-
-import java.io.Serializable;
-
-/**
- * XXX unused for now. this will hold the result of a /<code>Test</code> -SH
- */
-public interface TestResult extends Serializable {
- public Test getJob();
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/TorInstance.java b/src/de/uniba/wiai/lspi/puppetor/rmi/TorInstance.java
deleted file mode 100644
index 936f6ab..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/TorInstance.java
+++ /dev/null
@@ -1,57 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi;
-
-import java.util.List;
-
-/**
- * Describe each tor instance at the client with a <code>TorInstance</code>.
- *
- * @author Sebastian Hahn
- */
-public interface TorInstance {
- /**
- * Additional configuration options that a test wants to be set for a tor
- * instance. <code>TorInstance.shouldReplaceConfiguration()</code>
- * determines whether this configuration should be added to the default tor
- * configuration or replace it.
- *
- * @return An immutable list of extra configuration settings.
- */
- public List<String> getConfiguration();
-
- /**
- * @return the name by which this Tor instance will be identified at the
- * slaves. See de.uniba.wiai.lspi.puppetor.Network.getNode().
- */
- public String getName();
-
- /**
- * Whether the caller should use the result of
- * <code>TorInstance.getExtraConfiguration()</code> to append tor's
- * default configuration or replace it.
- *
- * @return true if configuration should be replaced, false if it should be
- * appended.
- */
- public boolean shouldReplaceConfiguration();
-
- public boolean isDirectoryAuthority();
-
- public boolean isBridgeAuthority();
-
- public boolean isRouter();
-
- public boolean isExit();
-
- public boolean isOnPrivateNetwork();
-
- public int getRevision();
-
- public boolean isBridge();
-
- public LocalMaster getMaster();
-}
\ No newline at end of file
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/execute/PuppeTorMasterProgram.java b/src/de/uniba/wiai/lspi/puppetor/rmi/execute/PuppeTorMasterProgram.java
deleted file mode 100644
index 16b0aae..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/execute/PuppeTorMasterProgram.java
+++ /dev/null
@@ -1,193 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.execute;
-
-import java.rmi.AccessException;
-import java.rmi.NoSuchObjectException;
-import java.rmi.NotBoundException;
-import java.rmi.RemoteException;
-import java.rmi.registry.LocateRegistry;
-import java.rmi.registry.Registry;
-import java.rmi.server.UnicastRemoteObject;
-
-import javax.rmi.ssl.SslRMIClientSocketFactory;
-import javax.rmi.ssl.SslRMIServerSocketFactory;
-
-import de.uniba.wiai.lspi.puppetor.rmi.AbstractMasterFactory;
-import de.uniba.wiai.lspi.puppetor.rmi.LocalMaster;
-import de.uniba.wiai.lspi.puppetor.rmi.MasterConnector;
-import de.uniba.wiai.lspi.puppetor.rmi.Master;
-import de.uniba.wiai.lspi.puppetor.rmi.Slave;
-import de.uniba.wiai.lspi.puppetor.rmi.RemoteMaster;
-import de.uniba.wiai.lspi.puppetor.rmi.TestExecutor;
-import de.uniba.wiai.lspi.puppetor.rmi.impl.MasterImplFactory;
-import de.uniba.wiai.lspi.puppetor.rmi.tests.TestRegistration;
-
-/**
- * The <code>PuppeTorMasterProgram</code> contains the main method for the
- * master instance of a distributed PuppeTor testing infrastructure. Currently,
- * configuration of the the master's network settings are implemented as
- * instance variables of <code>PuppeTorMasterProgram</code>. XXX eventually,
- * using configuration files would be better -SH
- *
- * @author Sebastian Hahn
- */
-public class PuppeTorMasterProgram {
-
- /**
- * The RMI registry.
- */
- private Registry registry;
-
- /**
- * The port the server is supposed to listen on. This port must not be
- * firewalled and must be forwarded to the machine if necessary.
- */
- final private static int serverport = 2050;
-
- /**
- * The address this server should bind on. Use 127.0.0.1 for local testing.
- * XXX Someone should check if this works with IPv6 -SH
- */
- final private static String serveraddress = "127.0.0.1";
-
- /**
- * Master application entry point
- *
- * @param args
- * Command-line arguments - ignored for now.
- * @throws NotBoundException
- * @throws RemoteException
- * @throws AccessException
- */
- public static void main(final String[] args) throws AccessException,
- RemoteException, NotBoundException {
- final PuppeTorMasterProgram server = new PuppeTorMasterProgram();
-
- final TestExecutor exec =
- AbstractMasterFactory.getInstance().getTestExecutorInstance();
- exec.startTesting();
- server.unregisterRMIRegistry();
- // Suggest a garbage collection, may increase shutdown.
- System.gc();
- }
-
- /**
- * Private constructor to set up RMI-related Java-internal variables and
- * create a <code>RemotePuppeTorImpl</code>.
- *
- * @throws NotBoundException
- * @throws RemoteException
- * @throws AccessException
- */
- private PuppeTorMasterProgram() throws AccessException, RemoteException,
- NotBoundException {
- // Set the location of the keystore where your private certificate is.
- System.setProperty("javax.net.ssl.keyStore", "res/keystore");
- // Set the password for your keystore.
- System.setProperty("javax.net.ssl.keyStorePassword", "pleasechange");
-
- /*
- * Set the location of the truststore which contains the exported
- * certificates of your slaves.
- */
- System.setProperty("javax.net.ssl.trustStore", "res/truststore");
- // Use a timeout of 45 seconds to detect disconnected clients.
- System.setProperty("java.rmi.dgc.leaseValue", "45000");
-
- // Tell the RMI system the location of the master
- System.setProperty("java.rmi.server.hostname", serveraddress);
-
- // Set up the factories we want to use
- AbstractMasterFactory.initialize(new MasterImplFactory());
-
- registerTests();
- final InnerMasterConnector master = InnerMasterConnector.getInstance();
-
- try {
- registry =
- LocateRegistry.createRegistry(serverport,
- new SslRMIClientSocketFactory(),
- new SslRMIServerSocketFactory(null, null, true));
- registry.bind("Master", master);
- } catch (final Throwable th) {
- th.printStackTrace();
- throw new RuntimeException(
- "Could not create the registry or bind"
- + "an object in it. Please check your system's configuration. "
- + th);
- }
-
- System.out.println("waiting for slaves.");
- try {
- Thread.sleep(1000 * 60 * 2);
- } catch (final InterruptedException e) {
- e.printStackTrace();
- throw new RuntimeException("This is the main thread. We don't "
- + "want to be interrupted.");
- }
- registry.unbind("Master");
- // Only unbinding is not enough, the object will still be exported.
- UnicastRemoteObject.unexportObject(master, true);
- }
-
- /**
- * Register all the tests you want to execute here.
- */
- private void registerTests() {
- final AbstractMasterFactory fact = AbstractMasterFactory.getInstance();
- final TestExecutor exec = fact.getTestExecutorInstance();
- final TestRegistration registration = fact.getTestRegistrationInstance();
- registration.registerTests(exec);
- }
-
- private void unregisterRMIRegistry() throws NoSuchObjectException {
- UnicastRemoteObject.unexportObject(registry, true);
- registry = null;
- }
-
- static class InnerMasterConnector extends UnicastRemoteObject implements
- MasterConnector {
- /**
- * Required for serialization. Needs to change for new released
- * versions.
- */
- private static final long serialVersionUID = 1L;
-
- private static InnerMasterConnector instance;
-
- static {
- try {
- instance = new InnerMasterConnector();
- } catch (final RemoteException e) {
- e.printStackTrace();
- throw new ExceptionInInitializerError(e);
- }
- }
-
- /**
- * private constructor, Singleton
- *
- * @throws RemoteException
- * RMI
- */
- private InnerMasterConnector() throws RemoteException {}
-
- public static InnerMasterConnector getInstance() {
- return instance;
- }
-
- public RemoteMaster registerClient(final Slave slave)
- throws RemoteException {
- final AbstractMasterFactory fact =
- AbstractMasterFactory.getInstance();
- final Master master = fact.createMaster(slave);
- AbstractMasterFactory.getInstance().getTestExecutorInstance()
- .registerLocalMaster((LocalMaster) master);
- return (RemoteMaster) master;
- }
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/execute/PuppeTorSlaveProgram.java b/src/de/uniba/wiai/lspi/puppetor/rmi/execute/PuppeTorSlaveProgram.java
deleted file mode 100644
index 7baac20..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/execute/PuppeTorSlaveProgram.java
+++ /dev/null
@@ -1,215 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.execute;
-
-import java.rmi.NoSuchObjectException;
-import java.rmi.NotBoundException;
-import java.rmi.RemoteException;
-import java.rmi.registry.LocateRegistry;
-import java.rmi.registry.Registry;
-import java.util.HashSet;
-import java.util.Set;
-import java.util.concurrent.BlockingQueue;
-import java.util.concurrent.CopyOnWriteArraySet;
-import java.util.concurrent.LinkedBlockingQueue;
-
-import javax.rmi.ssl.SslRMIClientSocketFactory;
-
-import de.uniba.wiai.lspi.puppetor.rmi.AbstractSlaveFactory;
-import de.uniba.wiai.lspi.puppetor.rmi.MasterConnector;
-import de.uniba.wiai.lspi.puppetor.rmi.Slave;
-import de.uniba.wiai.lspi.puppetor.rmi.Task;
-import de.uniba.wiai.lspi.puppetor.rmi.impl.SlaveImplFactory;
-import de.uniba.wiai.lspi.puppetor.rmi.tasks.TerminateTask;
-
-/**
- * The <code>PuppeTorSlaveProgram</code> contains the main method for the
- * slave instances of a distributed PuppeTor testing infrastructure. Currently,
- * configuration of the the master's network settings and some slave-specific
- * options are implemented as instance variables of
- * <code>PuppeTorSlaveProgram</code>. XXX eventually, using configuration
- * files would be better -SH
- *
- * @author Sebastian Hahn
- */
-public class PuppeTorSlaveProgram {
-
- /**
- * The port the master server is supposed to listen on. This port must not
- * be firewalled and must be forwarded to the machine if necessary. Use the
- * same value as you used for the master's configuration.
- */
- final private static int serverport = 2050;
-
- /**
- * The address master server listens on. Use the same value as you used for
- * the master's configuration. XXX Someone should check if this works with
- * IPv6 -SH
- */
- final private static String serveraddress = "127.0.0.1";//78.47.18.109";
-
- /**
- * The address this slave listens on. Use 127.0.0.1 for slaves that do not
- * open any ports for the outside world and thus can act as Tor clients
- * only.
- */
- final private static String myaddress = "127.0.0.1";
-
- /**
- * Each slave is identified by its unique <code>slaveName</code>. Be
- * careful to prevent naming collisions between slaves.
- */
- final private static String slaveName = "slave1";
-
- /**
- * A representation of this slave node to be sent to the master.
- */
- final private Slave slave;
-
- final protected static BlockingQueue<Task> tasks =
- new LinkedBlockingQueue<Task>();
-
- /**
- * Slave application entry point
- *
- * @param args
- * Command-line arguments
- * @throws NoSuchObjectException
- */
- public static void main(final String[] args) {
- final PuppeTorSlaveProgram slave = new PuppeTorSlaveProgram();
- while(true) {
- while (slave.connectToMaster() == false) {
- try {
- Thread.sleep(60000);
- } catch (final InterruptedException e) {
- throw new RuntimeException("Bug: Nothing should " +
- "interrupt us here!");
- }
- }
-
- final Thread taskGetter = new Thread(new Runnable() {
- public void run() {
- while (true) {
- try {
- try {
- // XXX I dislike the following lines, but
- // interruption doesn't work while we wait for the
- // master. Maybe there is a better solution that
- // doesn't eat the poison pill twice. -SH
- Task task = AbstractSlaveFactory.getMaster()
- .getNewTask();
- tasks.put(task);
- if (task instanceof TerminateTask) {
- break;
- }
- } catch (final RemoteException e) {
- // Empty the task queue - we cannot send any results
- // back, anyways.
- tasks.clear();
- // A taskid of 0 indicates a local Task
- tasks.put(new TerminateTask(0,
- "local termination"));
- Thread.currentThread().interrupt();
- }
- } catch (final InterruptedException e) {
- Thread.currentThread().interrupt();
- }
- if (Thread.interrupted()) {
- break;
- }
-
- }
- }
- });
-
- taskGetter.start();
- // XXX We use just a single Thread here. That isn't the smartest move.
- // -SH
- try {
- while (true) {
- final Task newTask = tasks.take();
- newTask.execute();
- if (Thread.interrupted()) {
- break;
- }
- }
- } catch (final InterruptedException ignored) {} catch (final RemoteException ignored) {}
- taskGetter.interrupt();
- AbstractSlaveFactory.setMaster(null);
- // We don't shut down, but want the Master to see we disconnected,
- // so hopefully a gc will happen soon.
- System.gc();
- }
-
- }
-
- /**
- * Private constructor to set up RMI-related Java-internal variables.
- */
- private PuppeTorSlaveProgram() {
- /*
- * Set the location of the keystore where your private certificate is.
- */
- System.setProperty("javax.net.ssl.keyStore", "res/keystore");
- /*
- * Set the password for your keystore.
- */
- System.setProperty("javax.net.ssl.keyStorePassword", "pleasechange");
- /*
- * Set the location of the truststore which contains the exported
- * certificates of your slaves
- */
- System.setProperty("javax.net.ssl.trustStore", "res/truststore");
- // Set up the factories we want to use
- AbstractSlaveFactory.initialize(new SlaveImplFactory());
- final AbstractSlaveFactory fac = AbstractSlaveFactory.getInstance();
-
- /*
- * Create a new slave with the properties that are relevant for the
- * computer and networking environment it runs on.
- * XXX unused for now -SH
- */
- final Set<Integer> ports = new HashSet<Integer>();
- /*for (int i = 10025; i < 20000; i++) {
- ports.add(i);
- }*/
- final Set<Integer> availableRevisions =
- new CopyOnWriteArraySet<Integer>();
- slave = fac.createSlave(slaveName, Slave.OS.UNDEFINED, myaddress,
- ports, availableRevisions);
-
- }
-
- /**
- * Attempt to make a connection to the master through RMI. If the master is
- * down, that is not a fatal condition, we want to keep trying until we can
- * reach it. Don't call this except from main().
- *
- * @return true if the connection was successful, false otherwise.
- */
- private boolean connectToMaster() {
- try {
- final Registry registry =
- LocateRegistry.getRegistry(serveraddress, serverport,
- new SslRMIClientSocketFactory());
- AbstractSlaveFactory.setMaster(((MasterConnector) registry
- .lookup("Master")).registerClient(slave));
- } catch (final NotBoundException e) {
- e.printStackTrace();
- System.out.println("We could connect, but the master has not "
- + "exported the Master Object yet." + e);
- return false;
- } catch (final RemoteException e) {
- return false;
- } catch (final IllegalArgumentException e) {
- throw new RuntimeException(
- "A slave with this name is already connected "
- + "to the server! " + e);
- }
- return true;
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/AbstractTaskImpl.java b/src/de/uniba/wiai/lspi/puppetor/rmi/impl/AbstractTaskImpl.java
deleted file mode 100644
index 008ee49..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/AbstractTaskImpl.java
+++ /dev/null
@@ -1,61 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.impl;
-
-import java.rmi.RemoteException;
-import java.util.Collection;
-
-import de.uniba.wiai.lspi.puppetor.rmi.AbstractSlaveFactory;
-import de.uniba.wiai.lspi.puppetor.rmi.Task;
-import de.uniba.wiai.lspi.puppetor.rmi.TaskResult;
-
-public abstract class AbstractTaskImpl implements Task {
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- /**
- * A human-readable name for the task (to be included in log-messages, etc).
- */
- final protected String name;
-
- /**
- * The globally unique id for this task.
- */
- final protected int id;
-
- public AbstractTaskImpl(final int id, final String name) {
- this.name = name;
- this.id = id;
- }
-
- public int getId() {
- return id;
- }
-
- public abstract void execute() throws InterruptedException, RemoteException;
-
- protected void reportResult(final boolean success)
- throws InterruptedException, RemoteException {
- reportResult(new TaskResultImpl(success, this, null));
- }
-
- protected void reportResult(final boolean success, final String result)
- throws InterruptedException, RemoteException {
- reportResult(new TaskResultImpl(success, this, result));
- }
-
- // XXX protect against double-reporting of results -SH
- protected void reportResult( final TaskResult taskResult)
- throws InterruptedException, RemoteException {
- AbstractSlaveFactory.getMaster().reportTaskResult(taskResult);
- }
-
- public void addAdditionalInformation(
- Collection<String> additionalInformation) {}
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/AbstractTestImpl.java b/src/de/uniba/wiai/lspi/puppetor/rmi/impl/AbstractTestImpl.java
deleted file mode 100644
index eea5b36..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/AbstractTestImpl.java
+++ /dev/null
@@ -1,52 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.impl;
-
-import de.uniba.wiai.lspi.puppetor.rmi.AbstractMasterFactory;
-import de.uniba.wiai.lspi.puppetor.rmi.NetworkDescription;
-import de.uniba.wiai.lspi.puppetor.rmi.Test;
-import de.uniba.wiai.lspi.puppetor.rmi.TestExecutor;
-
-/**
- * Extend <code>AbstractTestImpl</code> for your own tests. Make sure you also add them
- * to the TestRegistration.registerTest method.
- *
- * @author Sebastian Hahn
- */
-abstract public class AbstractTestImpl implements Test {
-
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- /**
- * A test description for log messages.
- */
- final private String description;
-
- /**
- * The network that must exist for the test to be executed.
- */
- final private NetworkDescription networkDescription;
-
- final protected TestExecutor exec;
-
- public AbstractTestImpl(final String description, final NetworkDescription network) {
- this.description = description;
- networkDescription = network;
- exec = AbstractMasterFactory.getInstance().
- getTestExecutorInstance();
- }
-
- public final NetworkDescription getNetworkDescription() {
- return networkDescription;
- }
-
- public String getDescription() {
- return description;
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/MasterImpl.java b/src/de/uniba/wiai/lspi/puppetor/rmi/impl/MasterImpl.java
deleted file mode 100644
index 4de305c..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/MasterImpl.java
+++ /dev/null
@@ -1,104 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.impl;
-
-import java.rmi.RemoteException;
-import java.rmi.server.UnicastRemoteObject;
-import java.rmi.server.Unreferenced;
-import java.util.concurrent.BlockingQueue;
-import java.util.concurrent.LinkedBlockingQueue;
-
-import de.uniba.wiai.lspi.puppetor.rmi.AbstractMasterFactory;
-import de.uniba.wiai.lspi.puppetor.rmi.LocalMaster;
-import de.uniba.wiai.lspi.puppetor.rmi.Master;
-import de.uniba.wiai.lspi.puppetor.rmi.Slave;
-import de.uniba.wiai.lspi.puppetor.rmi.RemoteMaster;
-import de.uniba.wiai.lspi.puppetor.rmi.Task;
-import de.uniba.wiai.lspi.puppetor.rmi.TaskResult;
-
-/**
- * Represent the master at the slave, and give the master a way to interact with
- * the respective slave.
- *
- * @author Sebastian Hahn
- */
-public class MasterImpl extends UnicastRemoteObject implements LocalMaster,
- RemoteMaster, Master, Unreferenced {
-
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- /**
- * Store the reference to the connected slave
- */
- private final Slave slave;
-
- private final BlockingQueue<Task> tasks;
-
- private final BlockingQueue<TaskResult> completedTasks;
-
- private final Object completedTasksLock = new Object();
-
- public Slave getSlave() {
- return slave;
- }
-
- /**
- * Check the slave object for validity and save it for further reference
- *
- * @throws RemoteException
- * RMI
- */
- public MasterImpl(final Slave slave) throws RemoteException {
- if (slave.getName() == null) {
- throw new NullPointerException("The slave's name must not be null!");
- }
- this.slave = slave;
- tasks = new LinkedBlockingQueue<Task>();
- completedTasks = new LinkedBlockingQueue<TaskResult>();
- }
-
- public Task getNewTask() throws RemoteException, InterruptedException {
- return tasks.take();
- }
-
- public void reportTaskResult(final TaskResult task) throws RemoteException,
- InterruptedException {
- synchronized (completedTasksLock) {
- completedTasks.put(task);
- completedTasksLock.notifyAll();
- }
- }
-
- public TaskResult getTaskResult(final int taskId)
- throws InterruptedException {
-
- synchronized (completedTasksLock) {
- TaskResult res;
- while ((res = completedTasks.peek()) == null
- || res.getTask().getId() != taskId) {
- completedTasksLock.wait();
- }
- return completedTasks.take();
- }
- }
-
- /**
- * Remove the slave from the Map of slaves and destroy its workqueues.
- * Inform the TestExecutor that all still-running tests failed because the
- * slave went away.
- */
- public void unreferenced() {
- AbstractMasterFactory.getInstance().getTestExecutorInstance()
- .unregisterLocalMaster(this);
- }
-
- public void addTask(final Task task) {
- tasks.add(task);
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/MasterImplFactory.java b/src/de/uniba/wiai/lspi/puppetor/rmi/impl/MasterImplFactory.java
deleted file mode 100644
index 2e4e67b..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/MasterImplFactory.java
+++ /dev/null
@@ -1,55 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.impl;
-
-import java.rmi.RemoteException;
-import de.uniba.wiai.lspi.puppetor.rmi.AbstractMasterFactory;
-import de.uniba.wiai.lspi.puppetor.rmi.Master;
-import de.uniba.wiai.lspi.puppetor.rmi.NetworkDescription;
-import de.uniba.wiai.lspi.puppetor.rmi.Slave;
-import de.uniba.wiai.lspi.puppetor.rmi.TestExecutor;
-import de.uniba.wiai.lspi.puppetor.rmi.tests.TestRegistration;
-
-/**
- * Create <code>RemotePuppeTorImpl</code> instances
- *
- * @author Sebastian Hahn
- */
-public class MasterImplFactory extends AbstractMasterFactory {
-
- @Override
- public Master createMaster(final Slave slave) throws RemoteException {
- return new MasterImpl(slave);
- }
-
- @Override
- public TestExecutor getTestExecutorInstance() {
- return TestExecutorImpl.getInstance();
- }
-
- @Override
- public TestRegistration getTestRegistrationInstance() {
- return TestRegistration.getInstance();
- }
-
-/* @Override
- public Test createPublicNetworkTestForClientsOnly(final String description) {
- return null;
- return new AbstractTestImpl(description, new NetworkDescriptionImpl(1, 0, 0));
- }*/
-
- /*@Override
- public Network createNetwork(final String name) {
- return new NetworkImpl(name);
- }*/
-
- @Override
- public NetworkDescription createNetworkDescription() {
- // XXX implement -SH
- return null;
- }
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/NetworkDescriptionImpl.java b/src/de/uniba/wiai/lspi/puppetor/rmi/impl/NetworkDescriptionImpl.java
deleted file mode 100644
index be3926d..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/NetworkDescriptionImpl.java
+++ /dev/null
@@ -1,43 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.impl;
-
-import java.util.concurrent.atomic.AtomicInteger;
-
-import de.uniba.wiai.lspi.puppetor.rmi.NetworkDescription;
-
-public class NetworkDescriptionImpl implements NetworkDescription {
-
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- private final AtomicInteger numProxies;
- private final AtomicInteger numRouters;
- private final AtomicInteger numDirectoryAuthorities;
-
- public int necessaryProxies() {
- return numProxies.get();
- }
-
- public int necessaryRouters() {
- return numRouters.get();
- }
-
- public int necessaryDirectoryAuthorities() {
- return numDirectoryAuthorities.get();
- }
-
- public NetworkDescriptionImpl(final int numProxies, final int numRouters,
- final int numDirectoryAuthorities) {
- this.numProxies = new AtomicInteger(numProxies);
- this.numRouters = new AtomicInteger(numRouters);
- this.numDirectoryAuthorities =
- new AtomicInteger(numDirectoryAuthorities);
- }
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/NetworkImpl.java b/src/de/uniba/wiai/lspi/puppetor/rmi/impl/NetworkImpl.java
deleted file mode 100644
index 5340a5b..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/NetworkImpl.java
+++ /dev/null
@@ -1,125 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.impl;
-
-import java.util.Collections;
-import java.util.HashSet;
-import java.util.List;
-import java.util.Set;
-import java.util.concurrent.CopyOnWriteArrayList;
-import java.util.concurrent.CopyOnWriteArraySet;
-
-import de.uniba.wiai.lspi.puppetor.rmi.LocalMaster;
-import de.uniba.wiai.lspi.puppetor.rmi.Network;
-import de.uniba.wiai.lspi.puppetor.rmi.Test;
-import de.uniba.wiai.lspi.puppetor.rmi.TorInstance;
-
-public class NetworkImpl implements Network {
-
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- /**
- * The name of the network, so we can identify it in logs etc.
- */
- final private String name;
-
- /**
- * All the tests that will run with this network. The individual Tor
- * instances will have the tests they need to execute broken down further.
- */
- private final List<Test> tests;
-
- /**
- * All the Tor instances running at the slaves that are linked to this
- * network.
- */
- private final Set<TorInstance> torInstances;
-
- private final boolean privateNetwork;
-
- public NetworkImpl(final String name, boolean privateNetwork) {
- this.name = name;
- tests = new CopyOnWriteArrayList<Test>();
- torInstances = new CopyOnWriteArraySet<TorInstance>();
- this.privateNetwork = privateNetwork;
- }
-
- public void addTorInstance(final TorInstance torInstance) {
- torInstances.add(torInstance);
- }
-
- public String getName() {
- return name;
- }
-
- public void addTest(final Test test) {
- tests.add(test);
- }
-
- public Set<LocalMaster> getMasters() {
- final Set<LocalMaster> masters = new HashSet<LocalMaster>();
- for (final TorInstance torInstance : torInstances) {
- masters.add(torInstance.getMaster());
- }
- return Collections.unmodifiableSet(masters);
- }
-
- public Set<TorInstance> getDirectoryAuthorities() {
- final Set<TorInstance> authorities = new HashSet<TorInstance>();
- for (final TorInstance torInstance : torInstances) {
- if (torInstance.isDirectoryAuthority()) {
- authorities.add(torInstance);
- }
- }
- return Collections.unmodifiableSet(authorities);
- }
-
- public Set<TorInstance> getProxies() {
- return Collections.unmodifiableSet(torInstances);
- }
-
- public Set<TorInstance> getProxiesOnly() {
- final Set<TorInstance> proxies = new HashSet<TorInstance>();
- for (final TorInstance torInstance : torInstances) {
- if (!torInstance.isRouter()) {
- proxies.add(torInstance);
- }
- }
- return Collections.unmodifiableSet(proxies);
- }
-
- public Set<TorInstance> getRouters() {
- final Set<TorInstance> routers = new HashSet<TorInstance>();
- for (final TorInstance torInstance : torInstances) {
- if (!torInstance.isRouter()) {
- routers.add(torInstance);
- }
- }
- return Collections.unmodifiableSet(routers);
- }
-
- public Set<TorInstance> getRoutersOnly() {
- final Set<TorInstance> routers = new HashSet<TorInstance>();
- for (final TorInstance torInstance : torInstances) {
- if (torInstance.isRouter() && !torInstance.isDirectoryAuthority()) {
- routers.add(torInstance);
- }
- }
- return Collections.unmodifiableSet(routers);
- }
-
- public List<Test> getTests() {
- return Collections.unmodifiableList(tests);
- }
-
- public boolean isPrivateNetwork() {
- return privateNetwork;
- }
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/SlaveImpl.java b/src/de/uniba/wiai/lspi/puppetor/rmi/impl/SlaveImpl.java
deleted file mode 100644
index 2e0829c..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/SlaveImpl.java
+++ /dev/null
@@ -1,131 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.impl;
-
-import java.net.InetAddress;
-import java.net.UnknownHostException;
-import java.util.Collections;
-import java.util.Set;
-import java.util.concurrent.CopyOnWriteArraySet;
-
-import de.uniba.wiai.lspi.puppetor.rmi.Slave;
-
-/**
- * Describe a client including its capabilities and network location to the
- * master. Also allow the master to query that information. A slave is not
- * required to give more than its name, but it can only be used for very limited
- * tests.
- *
- * @author Sebastian Hahn
- */
-public final class SlaveImpl implements Slave {
-
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- /**
- * The slave's name
- */
- protected final String name;
-
- /**
- * The slave's ip address
- */
- protected final InetAddress ip;
-
- /**
- * The operating system of the slave. See <code>Slave.OS</code>.
- */
- protected final OS os;
-
- /**
- * The ports this slave can open. Currently, we choose to use 256 as a
- * maximum that "should be" enough so we don't transfer so much data when
- * serializing. See the constructor. Make sure this is immutable.
- */
- protected final Set<Integer> ports;
-
- /**
- * The different revisions of Tor clients this slave can execute.
- */
- protected final Set<Integer> availableRevisions;
-
- /**
- * Create a new <code>Slave</code> with the metrics we currently use,
- * meaning it's operating system, it's IP, and the ports it can open. We may
- * in the future include other things like Tor executable version numbers
- * and the ability to run certain extra scripts.
- *
- * @param slaveName
- * the slave's name. Each slave must have it's own unique name in
- * a given PuppeTor network.
- * @param os
- * the slave's operating system.
- * @param ip
- * A String representation of the slave's IP (v4 or v6).
- * @param ports
- * All the ports this slave can open. Currently, we pick 256 out
- * of those at random.
- * @throws NullPointerException
- * if the slavename is null
- */
- public SlaveImpl(final String slaveName, final OS os, final String ip,
- final Set<Integer> ports, final Set<Integer> availableRevisions)
- throws NullPointerException {
- if (slaveName == null) {
- throw new NullPointerException("slaveName must not be null");
- }
- name = slaveName;
- this.os = os;
- try {
- this.ip = InetAddress.getByName(ip);
- } catch (final UnknownHostException e) {
- e.printStackTrace();
- throw new RuntimeException("You specified an invalid IP address!");
- }
-
- if (ports.size() > 256) {
- final Set<Integer> tmp = new CopyOnWriteArraySet<Integer>();
- int i = 0;
- for (final Integer j : tmp) {
- if (i++ == 256) {
- break;
- }
- tmp.add(j);
- }
- this.ports = Collections.unmodifiableSet(tmp);
- } else {
- this.ports =
- Collections
- .unmodifiableSet(new CopyOnWriteArraySet<Integer>(
- ports));
- }
- this.availableRevisions =
- Collections.unmodifiableSet(availableRevisions);
- }
-
- public String getName() {
- return name;
- }
-
- public OS getOS() {
- return os;
- }
-
- public InetAddress getIP() {
- return ip;
- }
-
- public Set<Integer> openablePorts() {
- return ports;
- }
-
- public Set<Integer> availableRevisions() {
- return availableRevisions;
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/SlaveImplFactory.java b/src/de/uniba/wiai/lspi/puppetor/rmi/impl/SlaveImplFactory.java
deleted file mode 100644
index 1fcb461..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/SlaveImplFactory.java
+++ /dev/null
@@ -1,31 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.impl;
-
-import java.util.Set;
-
-import de.uniba.wiai.lspi.puppetor.rmi.AbstractSlaveFactory;
-import de.uniba.wiai.lspi.puppetor.rmi.Slave;
-import de.uniba.wiai.lspi.puppetor.rmi.Slave.OS;
-
-/**
- * Create <code>RemotePuppeTorImpl</code> instances
- *
- * @author Sebastian Hahn
- */
-public class SlaveImplFactory extends AbstractSlaveFactory {
-
- /**
- * @return a new <code>RemotePuppeTorImpl</code> instance.
- */
- @Override
- public Slave createSlave(final String slaveName, final OS os,
- final String ip, final Set<Integer> ports,
- final Set<Integer> availableRevisions) throws NullPointerException {
- return new SlaveImpl(slaveName, os, ip, ports, availableRevisions);
- }
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/TaskResultImpl.java b/src/de/uniba/wiai/lspi/puppetor/rmi/impl/TaskResultImpl.java
deleted file mode 100644
index fc3124f..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/TaskResultImpl.java
+++ /dev/null
@@ -1,51 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.impl;
-
-import de.uniba.wiai.lspi.puppetor.rmi.Task;
-import de.uniba.wiai.lspi.puppetor.rmi.TaskResult;
-
-public class TaskResultImpl implements TaskResult {
-
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- /**
- * The task this result is used for.
- */
- final private Task task;
-
- /**
- * Store the result of the <code>task</code>, or <code>null</code> if no
- * result is necessary.
- */
- final private String result;
-
- /**
- * Whether the task execution was successful.
- */
- final private boolean wasSuccessful;
-
- public TaskResultImpl(final boolean success, final Task task, final String result) {
- wasSuccessful = success;
- this.task = task;
- this.result = result;
- }
-
- public Task getTask() {
- return task;
- }
-
- public boolean wasSuccessful() {
- return wasSuccessful;
- }
-
- public String getTaskResult() {
- return result;
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/TestExecutorImpl.java b/src/de/uniba/wiai/lspi/puppetor/rmi/impl/TestExecutorImpl.java
deleted file mode 100644
index 605dda0..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/TestExecutorImpl.java
+++ /dev/null
@@ -1,554 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.impl;
-
-import java.lang.reflect.Constructor;
-import java.lang.reflect.InvocationTargetException;
-import java.rmi.NoSuchObjectException;
-import java.rmi.server.UnicastRemoteObject;
-import java.util.Collection;
-import java.util.Collections;
-import java.util.HashSet;
-import java.util.LinkedList;
-import java.util.List;
-import java.util.Map;
-import java.util.Set;
-import java.util.Vector;
-import java.util.concurrent.Callable;
-import java.util.concurrent.ConcurrentHashMap;
-import java.util.concurrent.ConcurrentMap;
-import java.util.concurrent.CopyOnWriteArraySet;
-import java.util.concurrent.ExecutionException;
-import java.util.concurrent.FutureTask;
-import java.util.concurrent.atomic.AtomicBoolean;
-import java.util.concurrent.atomic.AtomicInteger;
-
-import de.uniba.wiai.lspi.puppetor.rmi.Network;
-import de.uniba.wiai.lspi.puppetor.rmi.LocalMaster;
-import de.uniba.wiai.lspi.puppetor.rmi.NetworkDescription;
-import de.uniba.wiai.lspi.puppetor.rmi.RemoteMaster;
-import de.uniba.wiai.lspi.puppetor.rmi.Task;
-import de.uniba.wiai.lspi.puppetor.rmi.TaskExecutionNotSuccessfulException;
-import de.uniba.wiai.lspi.puppetor.rmi.TaskResult;
-import de.uniba.wiai.lspi.puppetor.rmi.Test;
-import de.uniba.wiai.lspi.puppetor.rmi.TestResult;
-import de.uniba.wiai.lspi.puppetor.rmi.TestExecutor;
-import de.uniba.wiai.lspi.puppetor.rmi.TorInstance;
-import de.uniba.wiai.lspi.puppetor.rmi.tasks.BuildCircuitsTask;
-import de.uniba.wiai.lspi.puppetor.rmi.tasks.ConfigureAsPrivateTask;
-import de.uniba.wiai.lspi.puppetor.rmi.tasks.CreateDirectoryAuthorityTask;
-import de.uniba.wiai.lspi.puppetor.rmi.tasks.CreateNetworkTask;
-import de.uniba.wiai.lspi.puppetor.rmi.tasks.CreateProxyTask;
-import de.uniba.wiai.lspi.puppetor.rmi.tasks.CreateRouterTask;
-import de.uniba.wiai.lspi.puppetor.rmi.tasks.ShutdownNodesTask;
-import de.uniba.wiai.lspi.puppetor.rmi.tasks.StartNodesTask;
-import de.uniba.wiai.lspi.puppetor.rmi.tasks.TerminateTask;
-
-/**
- * @author Sebastian Hahn
- */
-public final class TestExecutorImpl implements TestExecutor {
-
- /**
- * The TestExecutor instance that we'll use
- */
- private static final TestExecutor instance = new TestExecutorImpl();
-
- /**
- * Store all local representations of connections to slaves. Use the
- * slave's name as key.
- */
- private static final ConcurrentMap<String, LocalMaster> masters =
- new ConcurrentHashMap<String, LocalMaster>();
-
- /**
- * Store all registered tests until they are started.
- */
- private static final List<Test> newTests = new Vector<Test>();
-
- /**
- * Store all currently running tests.
- */
- // private static final List<Test> runningTests = new Vector<Test>();
- /**
- * Store all tests that cannot run because the slave's cannot build such a
- * network.
- */
- private static final Set<Test> cannotRunTests =
- new CopyOnWriteArraySet<Test>();
-
- /**
- * Store the networks that are created out of the NetworkDescriptions.
- */
- private static final Set<Network> networks =
- new CopyOnWriteArraySet<Network>();
-
- /**
- * Store the currently active network when executing a test. Null indicates
- * that either testing hasn't started yet, or is already finished.
- */
- private static Network currentNetwork = null;
-
- /**
- * Store all failed tests.
- */
- private static final Map<Test, TestResult> failedTests =
- new ConcurrentHashMap<Test, TestResult>();
-
- /**
- * Store all successfully completed tests.
- */
- private static final Map<Test, TestResult> successfulTests =
- new ConcurrentHashMap<Test, TestResult>();
-
- /**
- * Whether testing has started. This cannot be reset.
- */
- private static AtomicBoolean startedTesting = new AtomicBoolean(false);
-
- /**
- * Store the id for the next task. Every task gets its own unique id > 0. A
- * task id equal to 0 indicates that the task was created by the slave to
- * stop execution.
- */
- private static AtomicInteger nextTaskId = new AtomicInteger(1);
-
- /**
- * Singleton
- */
- private TestExecutorImpl() {}
-
- public static TestExecutor getInstance() {
- return instance;
- }
-
- public void registerLocalMaster(final LocalMaster master) {
- if (startedTesting.get() == true) {
- throw new RuntimeException("Bug: Testing has started already.");
- }
- System.out.println(master.getSlave().getName() + " registered");
- if (null != masters.putIfAbsent(master.getSlave().getName(), master)) {
- throw new IllegalArgumentException(master.getSlave().getName()
- + " has already registered with this Server");
- }
- }
-
- public void unregisterLocalMaster(final LocalMaster master) {
- // XXX Don't forget to mark all tests as failed that depended on this
- // slave if we have already started with testing-SH
- try {
- UnicastRemoteObject.unexportObject((RemoteMaster) master, true);
- } catch (final NoSuchObjectException e) {
- throw new RuntimeException(
- "This is a bug. The master should have been exported.");
- }
- if (masters.remove(master.getSlave().getName()) == null) {
- throw new RuntimeException(
- "This is a bug. unregisterLocalMaster must never be " +
- "called twice for the same master.");
- }
- }
-
- /**
- * Evaluate which tests can be run with the currently connected slaves.
- * Cancel all currently running tests and report them as failed if they
- * cannot be restarted with the new (reduced) set of connected slaves,
- * restart those tests otherwise. XXX This should be merged with
- * makeNetworks -SH
- */
- protected void evaluateNetwork() {
- if (startedTesting.get() == false) {
- throw new RuntimeException("Bug: Testing has not been started.");
- // XXX We need to actually check the network.-SH
- }
- }
-
- public void startTesting() {
- if (startedTesting.get() == true) {
- throw new RuntimeException("Bug: Testing has started already.");
- }
- startedTesting.set(true);
- // check which tests we can actually execute
- evaluateNetwork();
- for (final Test test : newTests) {
- makeNetworks(test);
- }
- // XXX cancelled networks will still appear here...=SH
- for (final Network nw : networks) {
- synchronized(TestExecutorImpl.class) {
- currentNetwork = nw;
- }
- try {
- executeTaskForMasters(nw.getMasters(),
- CreateNetworkTask.class, null, null);
- // The network has been created at the slaves. Set up
- // authorities, routers and proxies.
-
- Set<TaskResult> authorityResults =
- new CopyOnWriteArraySet<TaskResult>();
- executeTaskForTorInstances(nw.getDirectoryAuthorities(),
- CreateDirectoryAuthorityTask.class, authorityResults,
- null);
- executeTaskForTorInstances(nw.getRoutersOnly(),
- CreateRouterTask.class, null, null);
- executeTaskForTorInstances(nw.getProxiesOnly(),
- CreateProxyTask.class, null, null);
- if(nw.isPrivateNetwork()) {
- List<String> authorities = new LinkedList<String>();
- for(TaskResult result : authorityResults) {
- authorities.add(result.getTaskResult());
- }
- executeTaskForMasters(nw.getMasters(),
- ConfigureAsPrivateTask.class, null, authorities);
- }
- // Start the nodes ...
- executeTaskForMasters(nw.getMasters(), StartNodesTask.class,
- null, null);
- // ... and build circuits. This may take some time.
- executeTaskForMasters(nw.getMasters(), BuildCircuitsTask.class,
- null, null);
-
- // Start using the individual tests that were set up for this
- // network.
- for (final Test test : nw.getTests()) {
- test.doTest();
- }
-
- // We don't need the network anymore, make sure it's shut down
- // properly.
- executeTaskForMasters(nw.getMasters(), ShutdownNodesTask.class,
- null, null);
-
- } catch (final TaskExecutionNotSuccessfulException e) {
- e.printStackTrace();
- // XXX Log this properly. -SH
- }
-
- }
- // The terminate Task interrupts the slave's main process to
- // indicate that it should disconnect.
- synchronized(TestExecutorImpl.class) {
- currentNetwork = null;
- }
- try {
- executeTaskForMasters(masters.values(), TerminateTask.class, null,
- null);
- } catch (TaskExecutionNotSuccessfulException e) {
- e.printStackTrace();
- // XXX Log this properly. -SH
- }
- }
-
- //XXX This method and the next look very similar. See what we can do. -SH
- public void executeTaskForMasters(final Collection<LocalMaster> masters,
- final Class<? extends Task> taskClass,
- final Set<TaskResult> taskResults,
- Collection<String> additionalInformation)
- throws TaskExecutionNotSuccessfulException {
-
- if (masters.isEmpty()) {
- return;
- }
- final Set<MasterTask> mTasks = new HashSet<MasterTask>();
- Constructor<? extends Task> ctor = null;
- try {
- ctor = taskClass.getConstructor(int.class, String.class);
- String networkName = "";
- synchronized(TestExecutorImpl.class) {
- if(currentNetwork == null) {
- networkName = "";
- } else {
- networkName = currentNetwork.getName();
- }
- }
- for (final LocalMaster master : masters) {
- Task task = ctor.newInstance(
- nextTaskId.getAndIncrement(), networkName);
- task.addAdditionalInformation(additionalInformation);
- mTasks.add(new MasterTask(master, task));
- }
-
- final TaskManager man = new TaskManager(mTasks);
- final FutureTask<Boolean> futureTask =
- new FutureTask<Boolean>(man);
- new Thread(futureTask).start();
- try {
- if (futureTask.get() == false) {
- throw new TaskExecutionNotSuccessfulException(man
- .getTaskResults());
- }
- } catch (final InterruptedException e) {
- throw new TaskExecutionNotSuccessfulException(null);
- } catch (final ExecutionException e) {
- // XXX Log this with the cause. Should not happen. -SH
- e.printStackTrace();
- throw new TaskExecutionNotSuccessfulException(null);
- }
- if(taskResults != null) {
- taskResults.addAll(man.getTaskResults());
- }
- } catch (final SecurityException e) {
- throw new RuntimeException(
- "BUG: We're not using a security manager, so where " +
- "does this exception come from?");
- } catch (final NoSuchMethodException e) {
- e.printStackTrace();
- throw new RuntimeException("BUG: Expected constructor undefined!");
- } catch (final IllegalArgumentException e) {
- e.printStackTrace();
- throw new RuntimeException("BUG: Expected constructor undefined!");
- } catch (final InstantiationException e) {
- e.printStackTrace();
- throw new RuntimeException("BUG: InstantiationException.");
- } catch (final IllegalAccessException e) {
- e.printStackTrace();
- throw new RuntimeException("BUG: Expected constructor undefined!");
- } catch (final InvocationTargetException e) {
- e.printStackTrace();
- throw new RuntimeException("BUG: Constructor threw an exception.");
- }
- }
-
- public Network getCurrentNetwork() {
- synchronized(TestExecutorImpl.class) {
- if(currentNetwork == null) {
- throw new RuntimeException("There is no network currently!");
- }
- return currentNetwork;
- }
- }
-
- public void executeTaskForTorInstances(
- final Collection<TorInstance> torInstances,
- final Class<? extends Task> taskClass,
- final Set<TaskResult> taskResults,
- Collection<String> additionalInformation)
- throws TaskExecutionNotSuccessfulException {
-
- if (torInstances.isEmpty()) {
- return;
- }
- final Set<MasterTask> mTasks = new HashSet<MasterTask>();
- Constructor<? extends Task> ctor = null;
- try {
- ctor = taskClass.getConstructor(int.class, String.class,
- String.class);
- String networkName = "";
- synchronized(TestExecutorImpl.class) {
- if(currentNetwork == null) {
- networkName = "";
- } else {
- networkName = currentNetwork.getName();
- }
- }
- for (final TorInstance torInstance : torInstances) {
- Task task = ctor.newInstance(nextTaskId.getAndIncrement(),
- networkName, torInstance.getName());
- task.addAdditionalInformation(additionalInformation);
- mTasks.add(new MasterTask(torInstance.getMaster(), task));
- }
-
- final TaskManager man = new TaskManager(mTasks);
- final FutureTask<Boolean> futureTask =
- new FutureTask<Boolean>(man);
- new Thread(futureTask).start();
- try {
- if (futureTask.get() == false) {
- throw new TaskExecutionNotSuccessfulException(man
- .getTaskResults());
- }
- } catch (final InterruptedException e) {
- throw new TaskExecutionNotSuccessfulException(null);
- } catch (final ExecutionException e) {
- // XXX Log this with the cause. Should not happen. -SH
- e.printStackTrace();
- throw new TaskExecutionNotSuccessfulException(null);
- }
- if(taskResults != null) {
- taskResults.addAll(man.getTaskResults());
- }
- } catch (final SecurityException e) {
- throw new RuntimeException(
- "BUG: We're not using a security manager, " +
- "so where does this exception come from?");
- } catch (final NoSuchMethodException e) {
- e.printStackTrace();
- throw new RuntimeException("BUG: Expected constructor undefined!");
- } catch (final IllegalArgumentException e) {
- e.printStackTrace();
- throw new RuntimeException("BUG: Expected constructor undefined!");
- } catch (final InstantiationException e) {
- e.printStackTrace();
- throw new RuntimeException("BUG: InstantiationException.");
- } catch (final IllegalAccessException e) {
- e.printStackTrace();
- throw new RuntimeException("BUG: Expected constructor undefined!");
- } catch (final InvocationTargetException e) {
- e.printStackTrace();
- throw new RuntimeException("BUG: Constructor threw an exception.");
- }
- }
-
- protected void makeNetworks(final Test test) {
- final NetworkDescription nd = test.getNetworkDescription();
- // XXX we want to do something much smarter here.-SH
-
- // XXX obviously, the following code adds the specified amount of
- // authorities, routers, and proxies for each slave, regardless of
- // capabilities. That isn't good. -SH
- for (final LocalMaster master : masters.values()) {
- int necessaryDirectoryAuthorities =
- nd.necessaryDirectoryAuthorities();
- boolean onPrivateNetwork = (necessaryDirectoryAuthorities > 0);
- // XXX the name choosing doesn't make sense. -SH
- final Network nw =
- new NetworkImpl(test.getDescription() + " slavename:"
- + master.getSlave().getName(), onPrivateNetwork);
- while (necessaryDirectoryAuthorities >= 1) {
- TorInstanceImpl authority = new TorInstanceImpl(master,
- "Authority" + necessaryDirectoryAuthorities);
- authority.setDirectoryAuthority(true);
- authority.setOnPrivateNetwork(onPrivateNetwork);
- nw.addTorInstance(authority);
- necessaryDirectoryAuthorities--;
- }
- int necessaryRouters = nd.necessaryRouters();
- while (necessaryRouters >= 1) {
- TorInstanceImpl router = new TorInstanceImpl(master, "Router" +
- necessaryRouters);
- router.setRouter(true);
- router.setOnPrivateNetwork(onPrivateNetwork);
- nw.addTorInstance(router);
- necessaryRouters--;
- }
- int necessaryProxies = nd.necessaryProxies();
- while (necessaryProxies >= 1) {
- TorInstanceImpl proxy = new TorInstanceImpl(master, "Proxy" +
- necessaryProxies);
- proxy.setOnPrivateNetwork(onPrivateNetwork);
- nw.addTorInstance(proxy);
- necessaryProxies--;
- }
- nw.addTest(test);
- networks.add(nw);
- }
- }
-
- public Map<Test, TestResult> getFailedTests() {
- return failedTests;
- }
-
- public Set<Test> getCannotRunTests() {
- return cannotRunTests;
- }
-
- public Map<Test, TestResult> getSuccessfulTests() {
- return successfulTests;
- }
-
- public void registerTest(final Test test) {
- if (startedTesting.get() == true) {
- throw new RuntimeException("Bug: Testing has started already.");
- }
- if (test == null) {
- throw new NullPointerException("tests must not be null");
- }
- newTests.add(test);
- }
-
- private static class TaskManager implements Callable<Boolean> {
-
- final private Set<TaskResult> taskResults =
- new CopyOnWriteArraySet<TaskResult>();
-
- final private Object taskResultsLock = new Object();
-
- final private Set<MasterTask> tasks;
-
- TaskManager(final Set<MasterTask> tasks) {
- if (tasks.isEmpty()) {
- throw new IllegalArgumentException("No tasks passed!");
- }
- this.tasks =
- Collections.synchronizedSet(
- new HashSet<MasterTask>(tasks));
- }
-
- public Boolean call() {
- for (final MasterTask task : tasks) {
- task.getMaster().addTask(task.getTask());
- // XXX Asking for trouble here. task could throw an exception!
- // -SH
- new Thread(new TaskCollector(task)).start();
- }
- try {
- synchronized (taskResultsLock) {
- // XXX See above. This needs more thinking -SH
- while (tasks.size() != taskResults.size()) {
- taskResultsLock.wait();
- }
- // XXX We want some logic here.-SH
- for (final TaskResult taskResult : taskResults) {
- if (!taskResult.wasSuccessful()) {
- return false;
- }
- }
- return true;
- }
- } catch (final InterruptedException e) {
- // XXX figure out a better solution here -SH
- e.printStackTrace();
- }
- return false;
- }
-
- public Set<TaskResult> getTaskResults() {
- return taskResults;
- }
-
- private class TaskCollector implements Runnable {
- private final MasterTask task;
-
- public void run() {
- try {
- final TaskResult taskResult =
- task.getMaster().getTaskResult(
- task.getTask().getId());
- synchronized (taskResultsLock) {
- taskResults.add(taskResult);
- taskResultsLock.notifyAll();
- }
- } catch (final InterruptedException e) {
- e.printStackTrace();
- Thread.currentThread().interrupt();
- }
-
- }
-
- protected TaskCollector(final MasterTask task) {
- this.task = task;
- }
- }
- }
-
- private static class MasterTask {
- final private LocalMaster master;
- final private Task task;
-
- MasterTask(final LocalMaster master, final Task task) {
- this.master = master;
- this.task = task;
- }
-
- public LocalMaster getMaster() {
- return master;
- }
-
- public Task getTask() {
- return task;
- }
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/TorInstanceImpl.java b/src/de/uniba/wiai/lspi/puppetor/rmi/impl/TorInstanceImpl.java
deleted file mode 100644
index 4d1ff5b..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/impl/TorInstanceImpl.java
+++ /dev/null
@@ -1,174 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.impl;
-
-import java.util.Collections;
-import java.util.LinkedList;
-import java.util.List;
-import java.util.concurrent.atomic.AtomicBoolean;
-import java.util.concurrent.atomic.AtomicInteger;
-
-import de.uniba.wiai.lspi.puppetor.rmi.LocalMaster;
-import de.uniba.wiai.lspi.puppetor.rmi.TorInstance;
-
-/**
- * Implements a Tor node. Threadsafe.
- *
- * @author Sebastian Hahn
- */
-public class TorInstanceImpl implements TorInstance {
-
- /**
- * Saves configuration options in an "option value" format.
- */
- private final List<String> configuration = new LinkedList<String>();
-
- /**
- * The name of this node.
- */
- private final String name;
-
- /**
- * The svn revision of the Tor build. 0 means no specified revision.
- */
- private final AtomicInteger revision = new AtomicInteger(0);
-
- /**
- * Whether we are a bridge authority.
- */
- private final AtomicBoolean bridgeAuthority = new AtomicBoolean(false);
-
- /**
- * Whether we are a directory authority.
- */
- final private AtomicBoolean directoryAuthority = new AtomicBoolean(false);
-
- /**
- * Whether we are an exit.
- */
- private final AtomicBoolean exit = new AtomicBoolean(false);
-
- /**
- * Whether we are a relay.
- */
- private final AtomicBoolean router = new AtomicBoolean(false);
-
- /**
- * Whether we are a bridge.
- */
- private final AtomicBoolean bridge = new AtomicBoolean(false);
-
- /**
- * Whether we are on a private network.
- */
- private AtomicBoolean privateNetwork = new AtomicBoolean(false);
-
- /**
- * Whether the configuration of the tor instance should be replaced by the
- * config returned from <code>this.getConfiguration()</code>.
- */
- private AtomicBoolean replaceConfiguration = new AtomicBoolean(false);
-
- /**
- * Provide a link to the slave where this torInstance will run.
- */
- private final LocalMaster master;
-
- /**
- * @param master
- * the <code>LocalMaster</code> that represents the slave on which
- * this Tor instance is running.
- * @param name
- * A name to identify the node.
- * XXX think about duplicates, we need a unique name to identify the
- * node, but is it smart to use that name as the Tor proxy's name as
- * well? -SH
- */
- TorInstanceImpl(final LocalMaster master, final String name) {
- this.master = master;
- this.name = name;
- }
-
- public synchronized List<String> getConfiguration() {
- return Collections.unmodifiableList(configuration);
- }
-
- public synchronized void addConfiguration(List<String> newConfig) {
- configuration.addAll(newConfig);
- }
-
- public int getRevision() {
- return revision.get();
- }
-
- public void setRevision(final int revision) {
- this.revision.set(revision);
- }
-
- public boolean isBridgeAuthority() {
- return bridgeAuthority.get();
- }
-
- public void setBridgeAuthority(boolean isBridgeAuthority) {
- bridgeAuthority.set(isBridgeAuthority);
- }
-
- public boolean isDirectoryAuthority() {
- return directoryAuthority.get();
- }
-
- public void setDirectoryAuthority(boolean isDirectoryAuthority) {
- if(isDirectoryAuthority) {
- setRouter(true);
- }
- directoryAuthority.set(isDirectoryAuthority);
- }
-
- public boolean isExit() {
- return exit.get();
- }
-
- public boolean isRouter() {
- return router.get();
- }
-
- public void setRouter(boolean isRouter) {
- if(!isRouter) {
- setDirectoryAuthority(false);
- }
- router.set(isRouter);
- }
-
- public boolean isOnPrivateNetwork() {
- return privateNetwork.get();
- }
-
- public void setOnPrivateNetwork(boolean onPrivateNetwork) {
- privateNetwork.set(onPrivateNetwork);
- }
-
- public boolean shouldReplaceConfiguration() {
- return replaceConfiguration.get();
- }
-
- public void setShouldReplaceConfiguration(
- boolean shouldReplaceConfiguration) {
- replaceConfiguration.set(shouldReplaceConfiguration);
- }
-
- public boolean isBridge() {
- return bridge.get();
- }
-
- public LocalMaster getMaster() {
- return master;
- }
-
- public synchronized String getName() {
- return name;
- }
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/AccessGoogleTask.java b/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/AccessGoogleTask.java
deleted file mode 100644
index 52bf7eb..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/AccessGoogleTask.java
+++ /dev/null
@@ -1,73 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.tasks;
-
-import java.rmi.RemoteException;
-
-import de.uniba.wiai.lspi.puppetor.ClientApplication;
-import de.uniba.wiai.lspi.puppetor.ClientEventType;
-import de.uniba.wiai.lspi.puppetor.Event;
-import de.uniba.wiai.lspi.puppetor.EventListener;
-import de.uniba.wiai.lspi.puppetor.EventManager;
-import de.uniba.wiai.lspi.puppetor.Network;
-import de.uniba.wiai.lspi.puppetor.NetworkFactory;
-import de.uniba.wiai.lspi.puppetor.rmi.impl.AbstractTaskImpl;
-
-public class AccessGoogleTask extends AbstractTaskImpl {
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- /**
- * The name of this tor node.
- */
- private final String nodeName;
-
- public AccessGoogleTask(final int id, final String networkName,
- final String nodeName) {
- super(id, networkName);
- this.nodeName = nodeName;
- }
-
- @Override
- public void execute() throws InterruptedException, RemoteException {
- final Network network = NetworkFactory.getNetworkByName(name);
-
- final ClientApplication client =
- network.createClient("client", "www.google.com", 80, network.getNode(nodeName)
- .getSocksPort());
-
- // create event listener to listen for client application events
- final EventListener clientEventListener = new EventListener() {
-
- // remember time when request was sent
- //private long before;
-
- public void handleEvent(Event event) {
- if (event.getType() == ClientEventType.CLIENT_SENDING_REQUEST) {
- //before = System.currentTimeMillis();
- } else if (event.getType() == ClientEventType.CLIENT_REPLY_RECEIVED) {
- // XXX Log how long requests took, see whether we need this all -SH
- ;
- }
- }
- };
-
- // obtain reference to event manager to be able to respond to events
- final EventManager manager = network.getEventManager();
-
- manager.addEventListener(client.getClientApplicationName(),
- clientEventListener);
-
- // perform at most three request with a timeout of 20 seconds each
- client.startRequests(3, 20000, true);
-
- // block this thread as long as client requests are running, but don't wait forever.
- reportResult(manager.waitForAnyOccurence(client.getClientApplicationName(),
- ClientEventType.CLIENT_REQUESTS_PERFORMED, 75000));
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/AddHiddenServiceTask.java b/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/AddHiddenServiceTask.java
deleted file mode 100644
index fc30b5d..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/AddHiddenServiceTask.java
+++ /dev/null
@@ -1,105 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.tasks;
-
-import java.rmi.RemoteException;
-import java.util.Set;
-
-import de.uniba.wiai.lspi.puppetor.ClientApplication;
-import de.uniba.wiai.lspi.puppetor.ClientEventType;
-import de.uniba.wiai.lspi.puppetor.Event;
-import de.uniba.wiai.lspi.puppetor.EventListener;
-import de.uniba.wiai.lspi.puppetor.EventManager;
-import de.uniba.wiai.lspi.puppetor.HiddenService;
-import de.uniba.wiai.lspi.puppetor.HiddenServiceEventType;
-import de.uniba.wiai.lspi.puppetor.Network;
-import de.uniba.wiai.lspi.puppetor.NetworkFactory;
-import de.uniba.wiai.lspi.puppetor.ProxyNode;
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-import de.uniba.wiai.lspi.puppetor.ServerApplication;
-import de.uniba.wiai.lspi.puppetor.rmi.impl.AbstractTaskImpl;
-
-public class AddHiddenServiceTask extends AbstractTaskImpl {
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- /**
- * The name of this tor node.
- */
- private final String nodeName;
-
- public AddHiddenServiceTask(final int id, final String networkName,
- final String nodeName) {
- super(id, networkName);
- this.nodeName = nodeName;
- }
-
- @Override
- public void execute() throws InterruptedException, RemoteException {
- final Network network = NetworkFactory.getNetworkByName(name);
- ProxyNode node = network.getNode(nodeName);
- try {
- // add hidden service
- final HiddenService hidServ = node.addHiddenService("hidServ");
- node.writeConfiguration();
- node.hup();
- final EventManager manager = network.getEventManager();
-
- // wait for 6 minutes that the proxy has published its first RSD
- if (!manager.waitForAnyOccurence(node.getNodeName(),
- HiddenServiceEventType.BOB_DESC_PUBLISHED_RECEIVED,
- 6L * 60L * 1000L)) {
- // failed to publish an RSD
- reportResult(false);
- return;
- }
-
- // create server application
- final ServerApplication server =
- network.createServer("server", hidServ.getServicePort());
-
- // create client application, pick one of the nodes.
- // XXX fix this so we can choose which node we want. -SH
- Set<String> nodes = network.getAllNodes().keySet();
- String[] localNodeNames = new String[10];
- localNodeNames = nodes.toArray(localNodeNames);
- final ClientApplication client =
- network.createClient("client",
- hidServ.determineOnionAddress(), hidServ
- .getVirtualPort(), network.getNode(localNodeNames[0]).getSocksPort());
-
- // register event listener
- final EventListener clientAndServerEventListener = new EventListener() {
- public void handleEvent(Event event) {
- // XXX log -SH
- }
- };
- manager.addEventListener(client.getClientApplicationName(),
- clientAndServerEventListener);
- manager.addEventListener(server.getServerApplicationName(),
- clientAndServerEventListener);
-
- // start server
- server.startListening();
-
- // perform at most five request with a timeout of 45 seconds each
- client.startRequests(5, 45000, true);
-
- // wait for request to be performed
- manager.waitForAnyOccurence(client.getClientApplicationName(),
- ClientEventType.CLIENT_REQUESTS_PERFORMED);
- reportResult(true);
- } catch (final PuppeTorException e) {
- e.printStackTrace();
- reportResult(false);
- }
-
-
- }
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/BuildCircuitsTask.java b/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/BuildCircuitsTask.java
deleted file mode 100644
index 5c002c7..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/BuildCircuitsTask.java
+++ /dev/null
@@ -1,40 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.tasks;
-
-import java.rmi.RemoteException;
-
-import de.uniba.wiai.lspi.puppetor.Network;
-import de.uniba.wiai.lspi.puppetor.NetworkFactory;
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-import de.uniba.wiai.lspi.puppetor.rmi.impl.AbstractTaskImpl;
-
-public class BuildCircuitsTask extends AbstractTaskImpl {
-
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- public BuildCircuitsTask(final int id, final String networkName) {
- super(id, networkName);
- }
-
- @Override
- public void execute() throws InterruptedException, RemoteException {
- final Network network = NetworkFactory.getNetworkByName(name);
- try {
- if (!network.hupUntilUp(50, 10000)) {
- reportResult(false);
- }
- } catch (final PuppeTorException e) {
- e.printStackTrace();
- reportResult(false);
- }
- reportResult(true);
- }
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/ConfigureAsPrivateTask.java b/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/ConfigureAsPrivateTask.java
deleted file mode 100644
index 464df65..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/ConfigureAsPrivateTask.java
+++ /dev/null
@@ -1,49 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.tasks;
-
-import java.rmi.RemoteException;
-import java.util.Collection;
-import java.util.LinkedList;
-import java.util.List;
-
-import de.uniba.wiai.lspi.puppetor.Network;
-import de.uniba.wiai.lspi.puppetor.NetworkFactory;
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-import de.uniba.wiai.lspi.puppetor.rmi.impl.AbstractTaskImpl;
-
-public class ConfigureAsPrivateTask extends AbstractTaskImpl {
-
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- final List<String> authorityInformation = new LinkedList<String>();
-
- public ConfigureAsPrivateTask(final int id, final String networkName) {
- super(id, networkName);
- }
-
- @Override
- public void execute() throws InterruptedException, RemoteException {
- final Network network = NetworkFactory.getNetworkByName(name);
- network.configureAsPartOfPrivateNetwork(authorityInformation);
- try {
- network.writeConfigurations();
- reportResult(true);
- } catch (final PuppeTorException e) {
- e.printStackTrace();
- reportResult(false);
- }
- }
-
- @Override
- public void addAdditionalInformation(
- Collection<String> additionalInformation) {
- authorityInformation.addAll(additionalInformation);
- }
-}
\ No newline at end of file
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/CreateDirectoryAuthorityTask.java b/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/CreateDirectoryAuthorityTask.java
deleted file mode 100644
index 7183d7d..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/CreateDirectoryAuthorityTask.java
+++ /dev/null
@@ -1,47 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.tasks;
-
-import java.rmi.RemoteException;
-
-import de.uniba.wiai.lspi.puppetor.DirectoryNode;
-import de.uniba.wiai.lspi.puppetor.Network;
-import de.uniba.wiai.lspi.puppetor.NetworkFactory;
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-import de.uniba.wiai.lspi.puppetor.rmi.impl.AbstractTaskImpl;
-
-public class CreateDirectoryAuthorityTask extends AbstractTaskImpl {
-
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- /**
- * The name of this Tor authority.
- */
- private final String authorityName;
-
- public CreateDirectoryAuthorityTask(final int id, final String networkName,
- final String authorityName) {
- super(id, networkName);
- this.authorityName = authorityName;
- }
-
- @Override
- public void execute() throws InterruptedException, RemoteException {
- final Network network = NetworkFactory.getNetworkByName(name);
- DirectoryNode node = network.createDirectory(authorityName);
- // write configuration of proxy node
- try {
- network.writeConfigurations();
- reportResult(true, node.getDirServerString());
- } catch (final PuppeTorException e) {
- e.printStackTrace();
- reportResult(false);
- }
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/CreateNetworkTask.java b/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/CreateNetworkTask.java
deleted file mode 100644
index 5c9f919..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/CreateNetworkTask.java
+++ /dev/null
@@ -1,30 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.tasks;
-
-import java.rmi.RemoteException;
-
-import de.uniba.wiai.lspi.puppetor.NetworkFactory;
-import de.uniba.wiai.lspi.puppetor.rmi.impl.AbstractTaskImpl;
-
-public class CreateNetworkTask extends AbstractTaskImpl {
-
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- public CreateNetworkTask(final int id, final String name) {
- super(id, name);
- }
-
- @Override
- public void execute() throws InterruptedException, RemoteException {
- NetworkFactory.createNetwork(name);
- reportResult(true);
- }
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/CreateProxyTask.java b/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/CreateProxyTask.java
deleted file mode 100644
index 90fd6a1..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/CreateProxyTask.java
+++ /dev/null
@@ -1,47 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.tasks;
-
-import java.rmi.RemoteException;
-
-import de.uniba.wiai.lspi.puppetor.Network;
-import de.uniba.wiai.lspi.puppetor.NetworkFactory;
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-import de.uniba.wiai.lspi.puppetor.rmi.impl.AbstractTaskImpl;
-
-public class CreateProxyTask extends AbstractTaskImpl {
-
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- /**
- * The name of this tor proxy.
- */
- private final String proxyName;
-
- public CreateProxyTask(final int id, final String networkName,
- final String proxyName) {
- super(id, networkName);
- this.proxyName = proxyName;
- }
-
- @Override
- public void execute() throws InterruptedException, RemoteException {
- final Network network = NetworkFactory.getNetworkByName(name);
- network.createProxy(proxyName);
-
- // write configuration of proxy node
- try {
- network.writeConfigurations();
- reportResult(true);
- } catch (final PuppeTorException e) {
- e.printStackTrace();
- reportResult(false);
- }
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/CreateRouterTask.java b/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/CreateRouterTask.java
deleted file mode 100644
index 5e1c170..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/CreateRouterTask.java
+++ /dev/null
@@ -1,47 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.tasks;
-
-import java.rmi.RemoteException;
-
-import de.uniba.wiai.lspi.puppetor.Network;
-import de.uniba.wiai.lspi.puppetor.NetworkFactory;
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-import de.uniba.wiai.lspi.puppetor.rmi.impl.AbstractTaskImpl;
-
-public class CreateRouterTask extends AbstractTaskImpl {
-
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- /**
- * The name of this router.
- */
- private final String routerName;
-
- public CreateRouterTask(final int id, final String networkName,
- final String routerName) {
- super(id, networkName);
- this.routerName = routerName;
- }
-
- @Override
- public void execute() throws InterruptedException, RemoteException {
- final Network network = NetworkFactory.getNetworkByName(name);
- network.createRouter(routerName);
- // write configuration of router node
- try {
- network.writeConfigurations();
- reportResult(true);
- } catch (final PuppeTorException e) {
- e.printStackTrace();
- reportResult(false);
- }
-
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/ShutdownNodesTask.java b/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/ShutdownNodesTask.java
deleted file mode 100644
index 9308340..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/ShutdownNodesTask.java
+++ /dev/null
@@ -1,37 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.tasks;
-
-import java.rmi.RemoteException;
-
-import de.uniba.wiai.lspi.puppetor.Network;
-import de.uniba.wiai.lspi.puppetor.NetworkFactory;
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-import de.uniba.wiai.lspi.puppetor.rmi.impl.AbstractTaskImpl;
-
-public class ShutdownNodesTask extends AbstractTaskImpl {
-
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- public ShutdownNodesTask(final int id, final String name) {
- super(id, name);
- }
-
- @Override
- public void execute() throws InterruptedException, RemoteException {
- final Network network = NetworkFactory.getNetworkByName(name);
- try {
- network.shutdownNodes();
- reportResult(true);
- } catch (final PuppeTorException e) {
- reportResult(false);
- }
- }
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/StartNodesTask.java b/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/StartNodesTask.java
deleted file mode 100644
index 39716a0..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/StartNodesTask.java
+++ /dev/null
@@ -1,41 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.tasks;
-
-import java.rmi.RemoteException;
-
-import de.uniba.wiai.lspi.puppetor.Network;
-import de.uniba.wiai.lspi.puppetor.NetworkFactory;
-import de.uniba.wiai.lspi.puppetor.PuppeTorException;
-import de.uniba.wiai.lspi.puppetor.rmi.impl.AbstractTaskImpl;
-
-public class StartNodesTask extends AbstractTaskImpl {
-
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- public StartNodesTask(final int id, final String networkName) {
- super(id, networkName);
- }
-
- @Override
- public void execute() throws InterruptedException, RemoteException {
- final Network network = NetworkFactory.getNetworkByName(name);
- try {
- if (!network.startNodes(5000)) {
- // failed to start the proxy
- reportResult(false);
- }
- } catch (final PuppeTorException e) {
- e.printStackTrace();
- reportResult(false);
- }
- reportResult(true);
- }
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/TerminateTask.java b/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/TerminateTask.java
deleted file mode 100644
index 8fc5b95..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/tasks/TerminateTask.java
+++ /dev/null
@@ -1,29 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.tasks;
-
-import java.rmi.RemoteException;
-
-import de.uniba.wiai.lspi.puppetor.rmi.impl.AbstractTaskImpl;
-
-public class TerminateTask extends AbstractTaskImpl {
-
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- public TerminateTask(final int id, final String name) {
- super(id, name);
- }
-
- @Override
- public void execute() throws InterruptedException, RemoteException {
- Thread.currentThread().interrupt();
- reportResult(true);
- }
-
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/tests/AccessGoogleOverPublicTorNetwork.java b/src/de/uniba/wiai/lspi/puppetor/rmi/tests/AccessGoogleOverPublicTorNetwork.java
deleted file mode 100644
index 2af3c74..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/tests/AccessGoogleOverPublicTorNetwork.java
+++ /dev/null
@@ -1,39 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.tests;
-
-import java.util.Set;
-
-import de.uniba.wiai.lspi.puppetor.rmi.Network;
-import de.uniba.wiai.lspi.puppetor.rmi.TaskExecutionNotSuccessfulException;
-import de.uniba.wiai.lspi.puppetor.rmi.TorInstance;
-import de.uniba.wiai.lspi.puppetor.rmi.impl.AbstractTestImpl;
-import de.uniba.wiai.lspi.puppetor.rmi.impl.NetworkDescriptionImpl;
-import de.uniba.wiai.lspi.puppetor.rmi.tasks.AccessGoogleTask;
-
-public class AccessGoogleOverPublicTorNetwork extends AbstractTestImpl {
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- public AccessGoogleOverPublicTorNetwork() {
- super("Use the public Tor network to connect to google.com and wait" +
- " for a response", new NetworkDescriptionImpl(1, 0, 0));
- }
-
- public void doTest() {
- Network nw = exec.getCurrentNetwork();
- Set<TorInstance> proxies = nw.getProxies();
- try {
- exec.executeTaskForTorInstances(proxies, AccessGoogleTask.class, null, null);
- System.out.println("Accessing google succeeded");
- } catch (TaskExecutionNotSuccessfulException e) {
- System.out.println("Accessing google failed");
- // Log this properly, especially where we failed
- }
- }
-}
\ No newline at end of file
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/tests/StartingAndAccessingHiddenServiceOverPrivateTorNetworkTest.java b/src/de/uniba/wiai/lspi/puppetor/rmi/tests/StartingAndAccessingHiddenServiceOverPrivateTorNetworkTest.java
deleted file mode 100644
index d8f5738..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/tests/StartingAndAccessingHiddenServiceOverPrivateTorNetworkTest.java
+++ /dev/null
@@ -1,40 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.tests;
-
-import java.util.Set;
-
-import de.uniba.wiai.lspi.puppetor.rmi.Network;
-import de.uniba.wiai.lspi.puppetor.rmi.TaskExecutionNotSuccessfulException;
-import de.uniba.wiai.lspi.puppetor.rmi.TorInstance;
-import de.uniba.wiai.lspi.puppetor.rmi.impl.AbstractTestImpl;
-import de.uniba.wiai.lspi.puppetor.rmi.impl.NetworkDescriptionImpl;
-import de.uniba.wiai.lspi.puppetor.rmi.tasks.AddHiddenServiceTask;
-
-public class StartingAndAccessingHiddenServiceOverPrivateTorNetworkTest extends
- AbstractTestImpl {
- /**
- * Required for serialization. Needs to change for new released versions.
- */
- private static final long serialVersionUID = 1L;
-
- public StartingAndAccessingHiddenServiceOverPrivateTorNetworkTest() {
- super("Create a private Tor network, set up a hidden service, " +
- "and access it", new NetworkDescriptionImpl(1, 3, 2));
- }
-
- public void doTest() {
- Network nw = exec.getCurrentNetwork();
- Set<TorInstance> proxies = nw.getProxiesOnly();
- try {
- exec.executeTaskForTorInstances(proxies, AddHiddenServiceTask.class, null, null);
- System.out.println("Accessing the hidden service succeeded.");
- } catch (TaskExecutionNotSuccessfulException e) {
- System.out.println("Accessing the hidden service failed.");
- // Log this properly, especially where we failed
- }
- }
-}
diff --git a/src/de/uniba/wiai/lspi/puppetor/rmi/tests/TestRegistration.java b/src/de/uniba/wiai/lspi/puppetor/rmi/tests/TestRegistration.java
deleted file mode 100644
index 3e55a32..0000000
--- a/src/de/uniba/wiai/lspi/puppetor/rmi/tests/TestRegistration.java
+++ /dev/null
@@ -1,33 +0,0 @@
-/*
- * Copyright 2007 Karsten Loesing
- * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
- * See LICENSE for licensing information
- */
-package de.uniba.wiai.lspi.puppetor.rmi.tests;
-
-import de.uniba.wiai.lspi.puppetor.rmi.TestExecutor;
-
-public class TestRegistration {
-
- private static TestRegistration instance = new TestRegistration();
-
- public void registerTests(TestExecutor testExecutor) {
- testExecutor.registerTest(new AccessGoogleOverPublicTorNetwork());
- testExecutor.registerTest(new
- StartingAndAccessingHiddenServiceOverPrivateTorNetworkTest());
- }
-
- /**
- * Singleton
- */
- private TestRegistration() {}
-
- /**
- * @return
- * the <code>TestRegistration</code> instance.
- */
- public static TestRegistration getInstance() {
- return instance;
- }
-
-}
diff --git a/src/org/torproject/puppetor/ClientApplication.java b/src/org/torproject/puppetor/ClientApplication.java
new file mode 100644
index 0000000..a1b27a0
--- /dev/null
+++ b/src/org/torproject/puppetor/ClientApplication.java
@@ -0,0 +1,100 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+/**
+ * The <code>ClientApplication</code> can be used to simulate simple
+ * <code>HTTP GET</code> requests by a virtual local client. Therefore, an
+ * address and a port are given to which the client shall connect. Requests are
+ * performed by a background thread, so that multiple requests could be
+ * performed at the same time.
+ *
+ * @author kloesing
+ */
+public interface ClientApplication {
+
+ /**
+ * <p>
+ * Performs one or more HTTP requests to a previously provided address and
+ * port. All requests are performed by a thread in the background, so that
+ * this method returns immediately. That thread will try for
+ * <code>retries</code> times to make the request with a timeout of
+ * <code>timeoutForEachRetry</code> milliseconds each. If an attempt is
+ * not successful, the thread nevertheless waits for the timeout to expire
+ * before performing the next attempt. If <code>stopOnSuccess</code> is
+ * set to <code>true</code>, the thread will quit performing requests
+ * immediately after the first successful request.
+ * </p>
+ *
+ * <p>
+ * For each sent request the application fires a
+ * <event>ClientEventType.CLIENT_SENDING_REQUEST</code> event. On receiving
+ * a reply it fires an event of type <code>ClientEventType.CLIENT_REPLY_RECEIVED</code>,
+ * if a request is not successful or times out, a <code>ClientEventType.CLIENT_GAVE_UP_REQUEST</code>
+ * event is fired. After all requests have been performed (either
+ * successfully, or not) an event of type <code>ClientEventType.CLIENT_REQUESTS_PERFORMED</code>
+ * is fired.
+ * </p>
+ *
+ * TODO may this method only be invoked once?!
+ *
+ * @param retries
+ * The number of retries that this client will perform. Must be
+ * <code>1</code> or greater.
+ * @param timeoutForEachRetry
+ * The timeout for each request. If a request is not successful,
+ * the thread nevertheless waits for the timeout to expire. Must
+ * not be negative.
+ * @param stopOnSuccess
+ * If set to <code>true</code>, the client quits performing
+ * requests after the first successful request, if <code>false</code>
+ * it continues until all <code>retries</code> have been
+ * processed.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given for either of the
+ * parameters.
+ */
+ public abstract void startRequests(int retries, long timeoutForEachRetry,
+ boolean stopOnSuccess);
+
+ /**
+ * Stops all requests that are currently running.
+ *
+ * @throws IllegalStateException
+ * Thrown if no requests have been started before.
+ */
+ public abstract void stopRequest();
+
+ /**
+ * Returns the name of this client.
+ *
+ * @return The name of this client.
+ */
+ public abstract String getClientApplicationName();
+
+ /**
+ * Returns the SOCKS port of the local Tor node to which requests are sent.
+ *
+ * @return The SOCKS port of the local Tor node to which requests are sent.
+ */
+ public abstract int getSocksPort();
+
+ /**
+ * Returns the target name for the requests sent by this client; can be
+ * either a server name/address or an onion address.
+ *
+ * @return The target name for the requests sent by this client.
+ */
+ public abstract String getTargetName();
+
+ /**
+ * Returns the target port for the requests sent by this client; can be
+ * either a server port or a virtual port of a hidden service.
+ *
+ * @return The target port for the requests sent by this client.
+ */
+ public abstract int getTargetPort();
+}
diff --git a/src/org/torproject/puppetor/ClientEventType.java b/src/org/torproject/puppetor/ClientEventType.java
new file mode 100644
index 0000000..0b5c81e
--- /dev/null
+++ b/src/org/torproject/puppetor/ClientEventType.java
@@ -0,0 +1,64 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+/**
+ * Event types that can be fired by a client application running as thread in
+ * the background.
+ */
+@SuppressWarnings("serial")
+public class ClientEventType implements EventType {
+
+ /**
+ * String identifying the type of the event type.
+ */
+ String typeString;
+
+ /**
+ * Creates a new event type with the given type string.
+ *
+ * @param typeString
+ * String identifying the type of the event type.
+ */
+ public ClientEventType(final String typeString) {
+ this.typeString = typeString;
+ }
+
+ public String getTypeName() {
+ return typeString;
+ }
+
+ /**
+ * The client application is sending a request; this event is fired
+ * internally and not parsed from a log statement from Tor.
+ */
+ public static final ClientEventType CLIENT_SENDING_REQUEST =
+ new ClientEventType("CLIENT_SENDING_REQUEST");
+
+ /**
+ * The client application has received a reply to a previously sent request;
+ * this event is fired internally and not parsed from a log statement from
+ * Tor.
+ */
+ public static final ClientEventType CLIENT_REPLY_RECEIVED =
+ new ClientEventType("CLIENT_REPLY_RECEIVED");
+
+ /**
+ * The client application has given up waiting for the reply to a previously
+ * sent request; this event is fired internally and not parsed from a log
+ * statement from Tor.
+ */
+ public static final ClientEventType CLIENT_GAVE_UP_REQUEST =
+ new ClientEventType("CLIENT_GAVE_UP_REQUEST");
+
+ /**
+ * The client application has completed a series of requests, whether they
+ * were successful or not; this event is fired internally and not parsed
+ * from a log statement from Tor.
+ */
+ public static final ClientEventType CLIENT_REQUESTS_PERFORMED =
+ new ClientEventType("CLIENT_REQUESTS_PERFORMED");
+}
diff --git a/src/org/torproject/puppetor/DirectoryNode.java b/src/org/torproject/puppetor/DirectoryNode.java
new file mode 100644
index 0000000..41fa6a2
--- /dev/null
+++ b/src/org/torproject/puppetor/DirectoryNode.java
@@ -0,0 +1,49 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+import java.util.Set;
+
+/**
+ * A <code>DirectoryNode</code> represents a Tor process that acts as
+ * <code>RouterNode</code> and which is further a directory authoritative
+ * server for the (private) Tor network. It inherits most of the configuration
+ * and behavior from <code>RouterNode</code> and adds some directory-specific
+ * configurations and behavior.
+ *
+ * @author kloesing
+ */
+public interface DirectoryNode extends RouterNode {
+
+ /**
+ * Combines the fingerprint of this node to a <code>DirServer</code>
+ * string that can be used to configure this or other nodes to use this node
+ * as directory server.
+ *
+ * @return <code>DirServer</code> string to configure a node to use this
+ * node as directory server.
+ * @throws PuppeTorException
+ * Thrown if a problem occurs when determining the fingerprint
+ * of this node.
+ */
+ public abstract String getDirServerString() throws PuppeTorException;
+
+ /**
+ * Adds the given (possibly empty) set of onion router fingerprints to the
+ * set of approved routers to confirm to directory clients, that the given
+ * routers can be trusted. Changes are only stored locally and not written
+ * to the <code>approved-routers</code> file to disk which will be done
+ * when writing the configuration of this node.
+ *
+ * @param approvedRouters
+ * The set of approved routers to be added. Each provided string
+ * must be formatted as
+ * <code>nickname 0000 0000 0000 0000 0000 0000 0000 0000 0000 0000</code>.
+ * @throws IllegalArgumentException
+ * Thrown if <code>null</code> is passed as parameter.
+ */
+ public void addApprovedRouters(Set<String> approvedRouters);
+}
diff --git a/src/org/torproject/puppetor/Event.java b/src/org/torproject/puppetor/Event.java
new file mode 100644
index 0000000..03626b1
--- /dev/null
+++ b/src/org/torproject/puppetor/Event.java
@@ -0,0 +1,54 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+import java.io.Serializable;
+
+/**
+ * An <code>Event</code> is created for every state change of an asynchronous
+ * system component, e.g. a Tor process or a client/server application running
+ * as thread in the background. In contrast to <code>NodeState</code> an
+ * <code>Event</code> cannot be a pre- or postconditions for a method
+ * invocation. There is no prescribed order in which events are fired by a
+ * certain process or application. Some events can be fired only once, others
+ * possibly multiple times. All management operations for events are contained
+ * in the <code>EventManager</code>.
+ *
+ * @author kloesing
+ */
+public interface Event extends Serializable {
+
+ /**
+ * Returns the name of the source of this event, which can be a Tor process
+ * or a client/server application running as thread in the background.
+ *
+ * @return The event source.
+ */
+ public abstract String getSource();
+
+ /**
+ * Returns the event type.
+ *
+ * @return The event type.
+ */
+ public abstract EventType getType();
+
+ /**
+ * Returns the original message that lead to firing this event.
+ *
+ * @return The original message.
+ */
+ public abstract String getMessage();
+
+ /**
+ * Returns the occurrence time of this event, which is either the parsed
+ * time from a Tor log statement, or the current system time when a
+ * client/server application fired this event.
+ *
+ * @return The occurrence time.
+ */
+ public abstract long getOccurrenceTime();
+}
diff --git a/src/org/torproject/puppetor/EventListener.java b/src/org/torproject/puppetor/EventListener.java
new file mode 100644
index 0000000..099cfec
--- /dev/null
+++ b/src/org/torproject/puppetor/EventListener.java
@@ -0,0 +1,25 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+/**
+ * This interface must be implemented by any object in a test application that
+ * shall be registered as event listener.
+ *
+ * @author kloesing
+ */
+public interface EventListener {
+
+ /**
+ * Is invoked when an asynchronous event is fired by the source (or one of
+ * the sources) for which this listener was registered.
+ *
+ * @param event
+ * The event that was fired.
+ */
+ public void handleEvent(Event event);
+
+}
diff --git a/src/org/torproject/puppetor/EventManager.java b/src/org/torproject/puppetor/EventManager.java
new file mode 100644
index 0000000..eafb9c1
--- /dev/null
+++ b/src/org/torproject/puppetor/EventManager.java
@@ -0,0 +1,231 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+import java.util.List;
+
+/**
+ * The <code>EventManager</code> is the central place for a test run to manage
+ * asynchronous events by Tor processes and client or server applications
+ * running as threads in the background. A test application can either register
+ * event listeners to be notified asynchronously about events when they occur,
+ * or synchronize with an event by being blocked until a certain event occurs.
+ *
+ * @author kloesing
+ */
+public interface EventManager {
+
+ /**
+ * Registers the given <code>listener</code> as event listener for events
+ * originating from the given <code>source</code>. This method returns a
+ * list of all previously fired events by this source, so that each event
+ * fired by this source is either included in the returned list or
+ * signalized in a later invocation on the event listener, but not in both.
+ * This prevents race conditions by eliminating the gap between registration
+ * of an event handler and asking if an event has been fired before
+ * registering.
+ *
+ * @param source
+ * The name of the source of events that the listener is
+ * interested in. May not be <code>null</code> and must be the
+ * name of a previously created node, client, or server.
+ * @param listener
+ * The listener that wants to be notified about events from the
+ * given <code>source</code>. If the <code>listener</code>
+ * is already registered for the same <code>source</code>,
+ * nothing happens, i.e. the <code>listener</code> will not
+ * receive multiple invocations for the same event. May not be
+ * <code>null</code>.
+ * @return A list of all previously fired events for the given
+ * <code>source</code>. If no event has been fired before, an
+ * empty list is returned instead of <code>null</code>.
+ * @throws IllegalArgumentException
+ * Thrown if <code>null</code> is passed for either of the
+ * parameters or if the <code>source</code> is unknown.
+ */
+ public abstract List<Event> addEventListener(String source,
+ EventListener listener);
+
+ /**
+ * Registers the given <code>listener</code> as event listener for future
+ * events originating from any source.
+ *
+ * @param listener
+ * The listener that wants to be notified about events from the
+ * given <code>source</code>. If the <code>listener</code>
+ * is already registered for all sources, nothing happens, i.e.
+ * the <code>listener</code> will not receive multiple
+ * invocations for the same event. May not be <code>null</code>.
+ * @throws IllegalArgumentException
+ * Thrown if <code>null</code> is passed for the parameter.
+ */
+ public abstract void addEventListener(EventListener listener);
+
+ /**
+ * Returns the list of all previously observed events from the given
+ * <code>source</code>.
+ *
+ * @param source
+ * The source of the events that the invoking thread is
+ * interested in. May not be <code>null</code> and must be the
+ * name of a previously created node, client, or server.
+ * @return List of all previously observed events from the given
+ * <code>source</code>.
+ * @throws IllegalArgumentException
+ * Thrown if <code>null</code> is passed as parameter or if
+ * the <code>source</code> is unknown.
+ */
+ public abstract List<Event> getEventHistory(String source);
+
+ /**
+ * Returns whether the given <code>eventType</code> has been observed from
+ * the given <code>source</code> before, or not.
+ *
+ * @param source
+ * The source of the event that the invoking thread is interested
+ * in. May not be <code>null</code> and must be the name of a
+ * previously created node, client, or server.
+ * @param eventType
+ * The event type that the invoking thread is interested in. May
+ * not be <code>null</code>.
+ * @throws IllegalArgumentException
+ * Thrown if <code>null</code> is passed for either of the
+ * parameters or if <code>source</code> is unknown.
+ * @return <code>true</code> if the event has been observed from the
+ * source before, <code>false</code> otherwise.
+ */
+ public abstract boolean hasEventOccured(String source, EventType eventType);
+
+ /**
+ * Removes the given <code>listener</code> as event listener from all
+ * previously registered sources. If this listener is not registered for any
+ * source, nothing happens.
+ *
+ * @param listener
+ * The listener that shall be removed from the list of registered
+ * listeners. May not be <code>null</code>.
+ * @throws IllegalArgumentException
+ * Thrown if <code>null</code> is passed as parameter.
+ */
+ public abstract void removeEventListener(EventListener listener);
+
+ /**
+ * Checks if the given <code>eventType</code> has been observed from the
+ * given <code>source</code> before; if not, blocks the invoking thread
+ * until the next event of this type is fired from that source. Note that
+ * this method does not restrict waiting to a timeout, so that it could
+ * potentially block forever!
+ *
+ * @param source
+ * The source of the event that the invoking thread is willing to
+ * wait for. May not be <code>null</code> and must be the name
+ * of a previously created node, client, or server.
+ * @param eventType
+ * The event type that the invoking thread is willing to wait for
+ * from the given <code>source</code>. May not be
+ * <code>null</code>.
+ * @throws IllegalArgumentException
+ * Thrown if <code>null</code> is passed for either of the
+ * parameters or if the <code>source</code> is unknown.
+ */
+ public abstract void waitForAnyOccurence(String source, EventType eventType);
+
+ /**
+ * Checks if the given <code>eventType</code> has been observed from the
+ * given <code>source</code> before; if not, blocks the invoking thread
+ * until the next event of this type is fired from that source or the given
+ * timeout of <code>maximumTimeToWaitInMillis</code> milliseconds has
+ * expired.
+ *
+ * @param source
+ * The source of the event that the invoking thread is willing to
+ * wait for. May not be <code>null</code> and must be the name
+ * of a previously created node, client, or server.
+ * @param eventType
+ * The event type that the invoking thread is willing to wait for
+ * from the given <code>source</code>. May not be
+ * <code>null</code>.
+ * @param maximumTimeToWaitInMillis
+ * The maximum time to wait in milliseconds. A positive value or
+ * zero restricts waiting to this time. If this value is
+ * negative, we will wait potentially forever.
+ * @return <code>true</code> if an event of the given type has been fired
+ * by the <code>source</code> within the given timeout,
+ * <code>false</code> otherwise.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is passed for either of the
+ * parameters or if the <code>source</code> is unknown.
+ */
+ public abstract boolean waitForAnyOccurence(String source,
+ EventType eventType, long maximumTimeToWaitInMillis);
+
+ /**
+ * Blocks the invoking thread until the next <code>event</code> is fired
+ * from the given <code>source</code>. This method only waits for the
+ * next occurence of an event, regardless of previous occurrences. Note that
+ * this method does not restrict waiting to a timeout, so that it could
+ * potentially block forever!
+ *
+ * @param source
+ * The source of the event that the invoking thread is willing to
+ * wait for. May not be <code>null</code> and must be the name
+ * of a previously created node, client, or server.
+ * @param eventType
+ * The event type that the invoking thread is willing to wait for
+ * from the given <code>source</code>. May not be
+ * <code>null</code>.
+ * @throws IllegalArgumentException
+ * Thrown if <code>null</code> is passed for either of the
+ * parameters or if the <code>source</code> is unknown.
+ */
+ public abstract void waitForNextOccurence(String source, EventType eventType);
+
+ /**
+ * Blocks the invoking thread until the next <code>event</code> is fired
+ * from the given <code>source</code> or the given timeout of
+ * <code>maximumTimeToWaitInMillis</code> milliseconds has expired. This
+ * method only waits for the next occurence of an event, regardless of
+ * previous occurrences.
+ *
+ * @param source
+ * The source of the event that the invoking thread is willing to
+ * wait for. May not be <code>null</code> and must be the name
+ * of a previously created node, client, or server.
+ * @param eventType
+ * The event type that the invoking thread is willing to wait for
+ * from the given <code>source</code>. May not be
+ * <code>null</code>.
+ * @param maximumTimeToWaitInMillis
+ * The maximum time to wait in milliseconds. A positive value or
+ * zero restricts waiting to this time. If this value is
+ * negative, we will wait potentially forever.
+ * @return <code>true</code> if an event of the given type has been fired
+ * by the <code>source</code> within the given timeout,
+ * <code>false</code> otherwise.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is passed for either of the
+ * parameters or if the <code>source</code> is unknown.
+ */
+ public abstract boolean waitForNextOccurence(String source,
+ EventType eventType, long maximumTimeToWaitInMillis);
+
+ /**
+ * Registers a new event type by passing a pattern string that can be
+ * applied to a regular expression when parsing Tor log statements. This is
+ * useful for log statements that are only included in modified Tor
+ * versions. Therefore, the event type may be an instance of a self-defined
+ * class that implements <code>EventType</code>.
+ *
+ * @param patternString
+ * The pattern string that will be used for parsing Tor log
+ * statements; the syntax corresponds to java.util.regex.Pattern.
+ * @param eventType
+ * The event type of the event that will be fired when a log
+ * statement was parsed that includes the given pattern.
+ */
+ public abstract void registerEventTypePattern(String patternString,
+ EventType eventType);
+}
diff --git a/src/org/torproject/puppetor/EventType.java b/src/org/torproject/puppetor/EventType.java
new file mode 100644
index 0000000..4923984
--- /dev/null
+++ b/src/org/torproject/puppetor/EventType.java
@@ -0,0 +1,26 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+import java.io.Serializable;
+
+/**
+ * The super interface of possible event types that are fired on a state change
+ * of an asynchronous system component, e.g. a Tor process or a client/server
+ * application running as thread in the background.
+ *
+ * @author kloesing
+ */
+public interface EventType extends Serializable {
+
+ /**
+ * Returns a string representation of the event type name for display
+ * purposes.
+ *
+ * @return String representation of the event type name.
+ */
+ public abstract String getTypeName();
+}
diff --git a/src/org/torproject/puppetor/HiddenService.java b/src/org/torproject/puppetor/HiddenService.java
new file mode 100644
index 0000000..26975fb
--- /dev/null
+++ b/src/org/torproject/puppetor/HiddenService.java
@@ -0,0 +1,61 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+/**
+ * A <code>HiddenService</code> instance contains all configurations of a
+ * hidden service that is registered at a node.
+ *
+ * @author kloesing
+ */
+public interface HiddenService {
+
+ /**
+ * Determines the onion address for a previously added hidden service with
+ * name <code>serviceName</code>. Requires that the node has been
+ * started, i.e. is in state <code>NodeState.RUNNING</code>, and is
+ * configured to provide this hidden service.
+ *
+ * @return The onion address string consisting of 16 base32 chars plus
+ * ".onion" for hidden service versions 0 and 1 or 16 base32 chars
+ * plus "." plus 24 base32 chars plus ".onion" for hidden service
+ * version 2.
+ * @throws IllegalArgumentException
+ * Thrown if <code>null</code> or a zero-length string is
+ * passed as parameter.
+ * @throws IllegalStateException
+ * Thrown if the node at which this hidden service is configured
+ * is not in state <code>NodeState.RUNNING</code>.
+ * @throws PuppeTorException
+ * Thrown if the onion address of this hidden service could not
+ * be read, which is also the case when the node's configuration
+ * has not been written and the node has not been HUP'ed after
+ * configuring the hidden service.
+ */
+ public abstract String determineOnionAddress() throws PuppeTorException;
+
+ /**
+ * Returns the name of the hidden service.
+ *
+ * @return The name of the hidden service.
+ */
+ public String getServiceName();
+
+ /**
+ * Returns the port on which the service listens for requests.
+ *
+ * @return The port on which the service listens for requests.
+ */
+ public int getServicePort();
+
+ /**
+ * Returns the virtual port that this hidden service runs on as it is
+ * announced to clients.
+ *
+ * @return The virtual port of this hidden service.
+ */
+ public int getVirtualPort();
+}
diff --git a/src/org/torproject/puppetor/HiddenServiceEventType.java b/src/org/torproject/puppetor/HiddenServiceEventType.java
new file mode 100644
index 0000000..5d29f1f
--- /dev/null
+++ b/src/org/torproject/puppetor/HiddenServiceEventType.java
@@ -0,0 +1,215 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+/**
+ * Event types that can be fired by all Tor processes performing hidden-service
+ * operations.
+ */
+@SuppressWarnings("serial")
+public class HiddenServiceEventType implements EventType {
+
+ /**
+ * String identifying the type of the event type.
+ */
+ String typeString;
+
+ /**
+ * Creates a new event type with the given type string.
+ *
+ * @param typeString
+ * String identifying the type of the event type.
+ */
+ public HiddenServiceEventType(final String typeString) {
+ this.typeString = typeString;
+ }
+
+ public String getTypeName() {
+ return typeString;
+ }
+
+ /**
+ * Alice has received an onion request; this event is parsed from a log
+ * statement in connection_ap_handshake_rewrite_and_attach().
+ */
+ public static final HiddenServiceEventType ALICE_ONION_REQUEST_RECEIVED =
+ new HiddenServiceEventType("ALICE_ONION_REQUEST_RECEIVED");
+
+ /**
+ * Alice sends a fetch request for a hidden service descriptor to a
+ * directory server; this event is parsed from a log statement in
+ * rend_client_refetch_renddesc().
+ */
+ public static final HiddenServiceEventType ALICE_SENDING_FETCH_DESC =
+ new HiddenServiceEventType("ALICE_SENDING_FETCH_DESC");
+
+ /**
+ * Alice receives a reply to a previous fetch request for a hidden service
+ * descriptors from a directory server; this event is parsed from a log
+ * statement in connection_dir_client_reached_eof().
+ */
+ public static final HiddenServiceEventType ALICE_DESC_FETCHED_RECEIVED =
+ new HiddenServiceEventType("ALICE_DESC_FETCHED_RECEIVED");
+
+ /**
+ * Alice has built a circuit to a rendezvous point and sends an
+ * ESTABLISH_RENDEZVOUS cell; this event is parsed from a log statement in
+ * rend_client_send_establish_rendezvous().
+ */
+ public static final HiddenServiceEventType ALICE_BUILT_REND_CIRC_SENDING_ESTABLISH_RENDEZVOUS =
+ new HiddenServiceEventType(
+ "ALICE_BUILT_REND_CIRC_SENDING_ESTABLISH_RENDEZVOUS");
+
+ /**
+ * Alice receives a RENDEZVOUS_ESTABLISHED cell from a rendezvous point;
+ * this event is parsed from a log statement in
+ * rend_client_rendezvous_acked().
+ */
+ public static final HiddenServiceEventType ALICE_RENDEZVOUS_ESTABLISHED_RECEIVED =
+ new HiddenServiceEventType("ALICE_RENDEZVOUS_ESTABLISHED_RECEIVED");
+
+ /**
+ * Alice has built a circuit to an introduction point (which does not
+ * automatically lead to sending an INTRODUCE1 cell, because the rendezvous
+ * circuit might not be ready); this event is parsed from a log statement in
+ * rend_client_introcirc_has_opened().
+ */
+ public static final HiddenServiceEventType ALICE_BUILT_INTRO_CIRC =
+ new HiddenServiceEventType("ALICE_BUILT_INTRO_CIRC");
+
+ /**
+ * Alice sends an INTRODUCE1 cell to an introduction point; this event is
+ * parsed from a log statement in rend_client_send_introduction().
+ */
+ public static final HiddenServiceEventType ALICE_SENDING_INTRODUCE1 =
+ new HiddenServiceEventType("ALICE_SENDING_INTRODUCE1");
+
+ /**
+ * Alice has received an INTRODUCE_ACK cell as an acknowledgement to a
+ * previously sent INTRODUCE1 cell; this event is parsed from a log
+ * statement in rend_client_introduction_acked().
+ */
+ public static final HiddenServiceEventType ALICE_INTRODUCE_ACK_RECEIVED =
+ new HiddenServiceEventType("ALICE_INTRODUCE_ACK_RECEIVED");
+
+ /**
+ * Alice has received a RENDEZVOUS2 cell and can now open an application
+ * connection to the client; this event is parsed from a log statement in
+ * rend_client_receive_rendezvous().
+ */
+ public static final HiddenServiceEventType ALICE_RENDEZVOUS2_RECEIVED_APP_CONN_OPENED =
+ new HiddenServiceEventType(
+ "ALICE_RENDEZVOUS2_RECEIVED_APP_CONN_OPENED");
+
+ /**
+ * Bob has built a circuit to an introduction point and sends an
+ * ESTABLISH_INTRO cell; this event is parsed from a log statement in
+ * rend_service_intro_has_opened().
+ */
+ public static final HiddenServiceEventType BOB_BUILT_INTRO_CIRC_SENDING_ESTABLISH_INTRO =
+ new HiddenServiceEventType(
+ "BOB_BUILT_INTRO_CIRC_SENDING_ESTABLISH_INTRO");
+
+ /**
+ * Bob has received an INTRO_ESTABLISHED cell, i.e. a node has confirmed to
+ * work as introduction point; this event is parsed from a log statement in
+ * rend_service_intro_established().
+ */
+ public static final HiddenServiceEventType BOB_INTRO_ESTABLISHED_RECEIVED =
+ new HiddenServiceEventType("BOB_INTRO_ESTABLISHED_RECEIVED");
+
+ /**
+ * Bob posts a hidden service descriptor to the directory servers; this
+ * event is parsed from a log statement in upload_service_descriptor().
+ */
+ public static final HiddenServiceEventType BOB_SENDING_PUBLISH_DESC =
+ new HiddenServiceEventType("BOB_SENDING_PUBLISH_DESC");
+
+ /**
+ * Bob received a response from a directory server to a previous publish
+ * request; this event is parsed from a log statement in
+ * connection_dir_client_reached_eof().
+ */
+ public static final HiddenServiceEventType BOB_DESC_PUBLISHED_RECEIVED =
+ new HiddenServiceEventType("BOB_DESC_PUBLISHED_RECEIVED");
+
+ /**
+ * Bob has received an INTRODUCE2 cell, i.e. a node wants to establish a
+ * connection, and will now try to establish a circuit to the client's
+ * rendezvous point; this event is parsed from a log statement in
+ * rend_service_introduce().
+ */
+ public static final HiddenServiceEventType BOB_INTRODUCE2_RECEIVED =
+ new HiddenServiceEventType("BOB_INTRODUCE2_RECEIVED");
+
+ /**
+ * Bob has built a circuit to a rendezvous point and sends a RENDEZVOUS1
+ * cell; this event is parsed from a log statement in
+ * rend_service_rendezvous_has_opened().
+ */
+ public static final HiddenServiceEventType BOB_BUILT_REND_CIRC_SENDING_RENDEZVOUS1 =
+ new HiddenServiceEventType(
+ "BOB_BUILT_REND_CIRC_SENDING_RENDEZVOUS1");
+
+ /**
+ * Bob opens a connection to the actual hidden server; this event is parsed
+ * from a log statement in connection_exit_begin_conn().
+ */
+ public static final HiddenServiceEventType BOB_APP_CONN_OPENED =
+ new HiddenServiceEventType("BOB_APP_CONN_OPENED");
+
+ /**
+ * The directory server has received a descriptor post request; this event
+ * is parsed from a log statement in directory_handle_command_post().
+ */
+ public static final HiddenServiceEventType DIR_PUBLISH_DESC_RECEIVED =
+ new HiddenServiceEventType("DIR_PUBLISH_DESC_RECEIVED");
+
+ /**
+ * The directory server has received a descriptor fetch request; this event
+ * is parsed from a log statement in directory_handle_command_get().
+ */
+ public static final HiddenServiceEventType DIR_FETCH_DESC_RECEIVED =
+ new HiddenServiceEventType("DIR_FETCH_DESC_RECEIVED");
+
+ /**
+ * The node received an ESTABLISH_INTRO cell, i.e. was requested to work as
+ * introduction point, and replies with an INTRO_ESTABLISHED cell; this
+ * event is parsed from a log statement in rend_mid_establish_intro().
+ */
+ public static final HiddenServiceEventType IPO_RECEIVED_ESTABLISH_INTRO_SENDING_INTRO_ESTABLISHED =
+ new HiddenServiceEventType(
+ "IPO_RECEIVED_ESTABLISH_INTRO_SENDING_INTRO_ESTABLISHED");
+
+ /**
+ * The introduction point received an INTRODUCE1 cell and reacts by sending
+ * an INTRODUCE2 cell to Bob and an INTRODUCE_ACK cell to Alice; this event
+ * is parsed from a log statement in rend_mid_introduce().
+ */
+ public static final HiddenServiceEventType IPO_RECEIVED_INTRODUCE1_SENDING_INTRODUCE2_AND_INTRODUCE_ACK =
+ new HiddenServiceEventType(
+ "IPO_RECEIVED_INTRODUCE1_SENDING_INTRODUCE2_AND_INTRODUCE_ACK");
+
+ /**
+ * The node received an ESTABLISH_RENDEZVOUS cell, i.e. was requested to
+ * work as rendezvous point, and replies with a RENDEZVOUS_ESTABLISHED cell;
+ * this event is parsed from a log statement in
+ * rend_mid_establish_rendezvous().
+ */
+ public static final HiddenServiceEventType RPO_RECEIVED_ESTABLISH_RENDEZVOUS_SENDING_RENDEZVOUS_ESTABLISHED =
+ new HiddenServiceEventType(
+ "RPO_RECEIVED_ESTABLISH_RENDEZVOUS_SENDING_RENDEZVOUS_ESTABLISHED");
+
+ /**
+ * The rendezvous point received a RENDEZVOUS1 cell and reacts by sending a
+ * RENDEZVOUS2 cell to Alice; this event is parsed from a log statement in
+ * rend_mid_rendezvous().
+ */
+ public static final HiddenServiceEventType RPO_RECEIVING_RENDEZVOUS1_SENDING_RENDEZVOUS2 =
+ new HiddenServiceEventType(
+ "RPO_RECEIVING_RENDEZVOUS1_SENDING_RENDEZVOUS2");
+
+}
\ No newline at end of file
diff --git a/src/org/torproject/puppetor/Network.java b/src/org/torproject/puppetor/Network.java
new file mode 100644
index 0000000..743c67c
--- /dev/null
+++ b/src/org/torproject/puppetor/Network.java
@@ -0,0 +1,657 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+import java.io.File;
+import java.util.List;
+import java.util.Map;
+
+/**
+ * A Network instance constitutes the central object of any test run and is
+ * aware of the node configuration. It creates all nodes for this configuration
+ * and is able to perform common operations on these nodes. Apart from the
+ * factory methods, all other operations could also be performed manually by an
+ * application using the appropriate interfaces.
+ *
+ * @author kloesing
+ */
+public interface Network {
+
+ /**
+ * <p>
+ * Configures this network as private Tor network by exchanging directory
+ * strings and router fingerprints between the nodes of this network.
+ * Afterwards, the nodes will be able to run a private Tor network,
+ * separated from public directory servers and onion routers.
+ * </p>
+ *
+ * <p>
+ * The configuration is done in two steps:
+ * <ol>
+ * <li>Directory strings of directory nodes are added to the configurations
+ * of all nodes in the other network.</li>
+ * <li>Router fingerprints of all router and directory nodes are added to
+ * the <code>approved-routers</code> files of the directory nodes.</li>
+ * </ol>
+ * </p>
+ *
+ * <p>
+ * This operation may be invoked in any state of the contained nodes.
+ * However, a network that does not have directory nodes of its own but
+ * relies on directory nodes of a merged network <b>should not be started
+ * before being configured as private network!</b> Otherwise it would
+ * connect to the public Tor network before being merged with the other
+ * private Tor network. However, it may also be invoked at a later time,
+ * e.g. to admit new nodes.
+ * </p>
+ *
+ * <p>
+ * This operation does not write any configurations to disk and neither
+ * starts a nodes nor sends HUP signals to running nodes. These operations
+ * are left to the application, so that they have more control over the
+ * network behavior.
+ * </p>
+ *
+ * <p>
+ * Applications need to ensure that there are enough directory nodes (2) and
+ * router nodes (3) in the network to allow normal operation.
+ * </p>
+ *
+ * @throws PuppeTorException
+ * Thrown if an I/O problem occurs while determining the nodes'
+ * fingerprints.
+ */
+ public abstract void configureAsPrivateNetwork() throws PuppeTorException;
+
+ /**
+ * XXX document properly -SH
+ */
+ public void configureAsPartOfPrivateNetwork(
+ List<String> authorizedDirectoriesFingerprints);
+
+
+ /**
+ * Creates a new client application, but does not yet perform a request.
+ *
+ * @param clientApplicationName
+ * The name for this client application, which is used for
+ * logging purposes and as event source. May neither be
+ * <code>null</code> or a zero-length string. The name needs to
+ * be unique in this network.
+ * @param targetAddress
+ * The target for requests sent by this client application. Can
+ * be a publicly available URL or an onion address. May neither
+ * be <code>null</code> or a zero-length string.
+ * @param targetPort
+ * The TCP port for requests sent by this client application. If
+ * the target address is an onion address, this port is the
+ * virtual port that the hidden service has announced. May not be
+ * negative or greater than 65535.
+ * @param socksPort
+ * The TCP port on which a local Tor process is waiting for
+ * incoming SOCKS requests. May not be negative or greater than
+ * 65535.
+ * @return Reference to the created client application.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given for either of the
+ * parameters.
+ */
+ public abstract ClientApplication createClient(
+ String clientApplicationName, String targetAddress, int targetPort,
+ int socksPort);
+
+ /**
+ * Creates a new directory node with automatically assigned ports and adds
+ * it to the network, but does not yet write its configuration to disk or
+ * start the corresponding Tor process.
+ *
+ * @param nodeName
+ * The name for this node, which is used as name for the working
+ * directory, for logging purposes, as node nickname, and as
+ * event source. May neither be <code>null</code> or have zero
+ * or more than 19 alpha-numeric characters. The node name needs
+ * to be unique in this network.
+ * @return Reference to the created directory node.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given as node name.
+ */
+ public abstract DirectoryNode createDirectory(String nodeName);
+
+ /**
+ * Creates a new directory node with automatically assigned ports that will
+ * listen on the given IP address and adds it to the network, but does not
+ * yet write its configuration to disk or start the corresponding Tor
+ * process.
+ *
+ * @param nodeName
+ * The name for this node, which is used as name for the working
+ * directory, for logging purposes, as node nickname, and as
+ * event source. May neither be <code>null</code> or have zero
+ * or more than 19 alpha-numeric characters. The node name needs
+ * to be unique in this network.
+ * @param serverIpAddress
+ * The IP address on which the node will listen. Must be a valid
+ * IP v4 address in dotted decimal notation. May not be
+ * <code>null</code>.
+ * @return Reference to the created directory node.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given as node name.
+ */
+ public abstract DirectoryNode createDirectory(String nodeName,
+ String serverIpAddress);
+
+ /**
+ * Creates a new directory node and adds it to the network, but does not yet
+ * write its configuration to disk or start the corresponding Tor process.
+ *
+ * @param nodeName
+ * The name for this node, which is used as name for the working
+ * directory, for logging purposes, as node nickname, and as
+ * event source. May neither be <code>null</code> or have zero
+ * or more than 19 alpha-numeric characters. The node name needs
+ * to be unique in this network.
+ * @param controlPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for a controller. May not be negative or greater than 65535.
+ * @param socksPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for incoming SOCKS requests. May not be negative or greater
+ * than 65535.
+ * @param orPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for incoming requests from other onion routers. May not be
+ * negative or greater than 65535.
+ * @param dirPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for incoming directory requests. May not be negative or
+ * greater than 65535.
+ * @return Reference to the created directory node.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given for either of the
+ * parameters.
+ */
+ public abstract DirectoryNode createDirectory(String nodeName,
+ int controlPort, int socksPort, int orPort, int dirPort);
+
+ /**
+ * Creates a new directory node that will listen on the given IP address and
+ * adds it to the network, but does not yet write its configuration to disk
+ * or start the corresponding Tor process.
+ *
+ * @param nodeName
+ * The name for this node, which is used as name for the working
+ * directory, for logging purposes, as node nickname, and as
+ * event source. May neither be <code>null</code> or have zero
+ * or more than 19 alpha-numeric characters. The node name needs
+ * to be unique in this network.
+ * @param controlPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for a controller. May not be negative or greater than 65535.
+ * @param socksPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for incoming SOCKS requests. May not be negative or greater
+ * than 65535.
+ * @param orPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for incoming requests from other onion routers. May not be
+ * negative or greater than 65535.
+ * @param dirPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for incoming directory requests. May not be negative or
+ * greater than 65535.
+ * @param serverIpAddress
+ * The IP address on which the node will listen. Must be a valid
+ * IP v4 address in dotted decimal notation. May not be
+ * <code>null</code>.
+ * @return Reference to the created directory node.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given for either of the
+ * parameters.
+ */
+ public abstract DirectoryNode createDirectory(String nodeName,
+ int controlPort, int socksPort, int orPort, int dirPort,
+ String serverIpAddress);
+
+ /**
+ * Creates a new <code>ProxyNode</code> with automatically assigned ports
+ * and adds it to the network, but does not yet write its configuration to
+ * disk or start the corresponding Tor process.
+ *
+ * @param nodeName
+ * The name for this node, which is used as name for the working
+ * directory, for logging purposes, and as event source. May
+ * neither be <code>null</code> or have zero or more than 19
+ * alpha-numeric characters. The node name needs to be unique in
+ * this network.
+ * @return Reference to the created proxy node.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given as node name.
+ */
+ public abstract ProxyNode createProxy(String nodeName);
+
+ /**
+ * Creates a new <code>ProxyNode</code> and adds it to the network, but
+ * does not yet write its configuration to disk or start the corresponding
+ * Tor process.
+ *
+ * @param nodeName
+ * The name for this node, which is used as name for the working
+ * directory, for logging purposes, and as event source. May
+ * neither be <code>null</code> or have zero or more than 19
+ * alpha-numeric characters. The node name needs to be unique in
+ * this network.
+ * @param controlPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for a controller. May not be negative or greater than 65535.
+ * @param socksPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for incoming SOCKS requests. May not be negative or greater
+ * than 65535.
+ * @return Reference to the created proxy node.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given for either of the
+ * parameters.
+ */
+ public abstract ProxyNode createProxy(String nodeName, int controlPort,
+ int socksPort);
+
+ /**
+ * Creates a new <code>RouterNode</code> with automatically assigned ports
+ * and adds it to the network, but does not yet write its configuration to
+ * disk or start the corresponding Tor process.
+ *
+ * @param nodeName
+ * The name for this node, which is used as name for the working
+ * directory, for logging purposes, as node nickname, and as
+ * event source. May neither be <code>null</code> or have zero
+ * or more than 19 alpha-numeric characters. The node name needs
+ * to be unique in this network.
+ * @return Reference to the created router node.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given as node name.
+ */
+ public abstract RouterNode createRouter(String nodeName);
+
+ /**
+ * Creates a new <code>RouterNode</code> and adds it to the network, but
+ * does not yet write its configuration to disk or start the corresponding
+ * Tor process.
+ *
+ * @param nodeName
+ * The name for this node, which is used as name for the working
+ * directory, for logging purposes, as node nickname, and as
+ * event source. May neither be <code>null</code> or have zero
+ * or more than 19 alpha-numeric characters. The node name needs
+ * to be unique in this network.
+ * @param controlPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for a controller. May not be negative or greater than 65535.
+ * @param socksPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for incoming SOCKS requests. May not be negative or greater
+ * than 65535.
+ * @param orPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for incoming requests from other onion routers. May not be
+ * negative or greater than 65535.
+ * @param dirPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for incoming directory requests which in fact are requests for
+ * the mirrored directory. May not be negative or greater than
+ * 65535.
+ * @return Reference to the created router node.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given for either of the
+ * parameters.
+ */
+ public abstract RouterNode createRouter(String nodeName, int controlPort,
+ int socksPort, int orPort, int dirPort);
+
+ /**
+ * Creates a new <code>RouterNode</code> with automatically assigned ports
+ * that will listen on the given IP address and adds it to the network, but
+ * does not yet write its configuration to disk or start the corresponding
+ * Tor process.
+ *
+ * @param nodeName
+ * The name for this node, which is used as name for the working
+ * directory, for logging purposes, as node nickname, and as
+ * event source. May neither be <code>null</code> or have zero
+ * or more than 19 alpha-numeric characters. The node name needs
+ * to be unique in this network.
+ * @param serverIpAddress
+ * The IP address on which the node will listen. Must be a valid
+ * IP v4 address in dotted decimal notation. May not be
+ * <code>null</code>.
+ * @return Reference to the created router node.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given as node name.
+ */
+ public abstract RouterNode createRouter(String nodeName,
+ String serverIpAddress);
+
+ /**
+ * Creates a new <code>RouterNode</code> that will listen on the given IP
+ * address and adds it to the network, but does not yet write its
+ * configuration to disk or start the corresponding Tor process.
+ *
+ * @param nodeName
+ * The name for this node, which is used as name for the working
+ * directory, for logging purposes, as node nickname, and as
+ * event source. May neither be <code>null</code> or have zero
+ * or more than 19 alpha-numeric characters. The node name needs
+ * to be unique in this network.
+ * @param controlPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for a controller. May not be negative or greater than 65535.
+ * @param socksPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for incoming SOCKS requests. May not be negative or greater
+ * than 65535.
+ * @param orPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for incoming requests from other onion routers. May not be
+ * negative or greater than 65535.
+ * @param dirPort
+ * The TCP port on which the corresponding Tor process will wait
+ * for incoming directory requests which in fact are requests for
+ * the mirrored directory. May not be negative or greater than
+ * 65535.
+ * @param serverIpAddress
+ * The IP address on which the node will listen. Must be a valid
+ * IP v4 address in dotted decimal notation. May not be
+ * <code>null</code>.
+ * @return Reference to the created router node.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given for either of the
+ * parameters.
+ */
+ public abstract RouterNode createRouter(String nodeName, int controlPort,
+ int socksPort, int orPort, int dirPort, String serverIpAddress);
+
+ /**
+ * Creates a new <code>ServerApplication</code> with automatically
+ * assigned ports, but does not start listening for incoming requests.
+ *
+ * @param serverApplicationName
+ * The name for this server application, which is used for
+ * logging purposes and as event source. May neither be
+ * <code>null</code> or a zero-length string. The name needs to
+ * be unique in this network.
+ * @return Reference to the created server application.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given as server application
+ * name.
+ */
+ public abstract ServerApplication createServer(String serverApplicationName);
+
+ /**
+ * Creates a new <code>ServerApplication</code>, but does not start
+ * listening for incoming requests.
+ *
+ * @param serverApplicationName
+ * The name for this server application, which is used for
+ * logging purposes and as event source. May neither be
+ * <code>null</code> or a zero-length string. The name needs to
+ * be unique in this network.
+ * @param serverPort
+ * The TCP port on which the server will wait for incoming
+ * requests. May not be negative or greater than 65535.
+ * @return Reference to the created server application.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given for either of the
+ * parameters.
+ */
+ public abstract ServerApplication createServer(
+ String serverApplicationName, int serverPort);
+
+ /**
+ * Returns a reference on the (single) event manager for this network.
+ *
+ * @return Reference on the (single) event manager for this network.
+ */
+ public abstract EventManager getEventManager();
+
+ /**
+ * Returns (a copy of) the map containing the names of all directory nodes
+ * as keys and the corresponding directory nodes as values.
+ *
+ * @return Map containing all directory nodes.
+ */
+ public abstract Map<String, DirectoryNode> getAllDirectoryNodes();
+
+ /**
+ * Returns (a copy of) the map containing the names of all router nodes
+ * (only those that are not acting as directory nodes at the same time) as
+ * keys and the corresponding router nodes as values.
+ *
+ * @return Map containing all router nodes.
+ */
+ public abstract Map<String, RouterNode> getAllRouterNodes();
+
+ /**
+ * Returns (a copy of) the map containing the names of all proxy nodes (only
+ * those that are not acting as router or directory nodes at the same time)
+ * as keys and the corresponding proxy nodes as values.
+ *
+ * @return Map containing all proxy nodes.
+ */
+ public abstract Map<String, ProxyNode> getAllProxyNodes();
+
+ /**
+ * Returns (a copy of) the map containing the names of all nodes as keys and
+ * the corresponding proxy nodes as values.
+ *
+ * @return Map containing all nodes.
+ */
+ public abstract Map<String, ProxyNode> getAllNodes();
+
+ /**
+ * Returns the node with name <code>nodeName</code> or <code>null</code>
+ * if no such node exists.
+ *
+ * @param nodeName
+ * The node name to look up.
+ * @return The node with name <code>nodeName</code>.
+ */
+ public abstract ProxyNode getNode(String nodeName);
+
+ /**
+ * <p>
+ * Sends a HUP signal to all nodes in the network in regular intervals and
+ * blocks the invoking thread until all nodes have reported to have
+ * successfully opened a circuit.
+ * </p>
+ *
+ * <p>
+ * First, the method waits for <code>hupInterval</code> milliseconds for
+ * the nodes to have successfully opened a circuit. If they do not succeed
+ * within this time, a HUP signal is sent to all nodes and the method waits
+ * for another <code>hupInterval</code> milliseconds. In total, the method
+ * sends at most <code>tries</code> HUP signals before giving up and
+ * returning with <code>false</code>. Thus, the maximum waiting time is
+ * <code>(tries + 1)</code> times <code>hupInterval</code>. As soon as
+ * all nodes have successfully opened circuits, the method returns with
+ * <code>true</code>. This operation can only be invoked, if all nodes in
+ * the network are in state <code>NodeState.RUNNING</code>.
+ * </p>
+ *
+ * @param tries
+ * The maximum number of HUP signals that are sent to the Tor
+ * processes. Negative values are not allowed. A value of zero
+ * means to wait only for the given time of
+ * <code>hupInterval</code> milliseconds without sending a HUP
+ * signal. Typical values depend on the network being a public or
+ * private Tor network and range about 3 to 5 tries.
+ * @param hupInterval
+ * The time in milliseconds that the method will wait between
+ * sending HUP signals. Negative values are not allowed.
+ * Typically, values should not be smaller than 5 seconds to
+ * permit Tor to stabilize.
+ * @throws IllegalStateException
+ * Thrown if at least one node is not in state
+ * <code>NodeState.RUNNING</code>.
+ * @throws IllegalArgumentException
+ * Thrown if a negative value is given for either
+ * <code>tries</code> or <code>hupInterval</code>.
+ * @throws PuppeTorException
+ * Thrown if an I/O problem occurs while sending HUP signals.
+ * @return <code>true</code> if all nodes have reported to have
+ * successfully opened a circuit, <code>false</code> otherwise.
+ */
+ public abstract boolean hupUntilUp(int tries, long hupInterval)
+ throws PuppeTorException;
+
+ /**
+ * Sends a HUP signal to all nodes in the network once. This operation can
+ * only be invoked, if all nodes in the network are in state
+ * <code>NodeState.RUNNING</code>.
+ *
+ * @throws IllegalStateException
+ * Thrown if at least one node is not in state
+ * <code>NodeState.RUNNING</code>.
+ * @throws PuppeTorException
+ * Thrown if an I/O problem occurs while sending HUP signals.
+ */
+ public abstract void hupAllNodes() throws PuppeTorException;
+
+ /**
+ * Sends a HUP signal to all directory nodes in the network once. This
+ * operation can only be invoked, if all directory nodes in the network are
+ * in state <code>NodeState.RUNNING</code>.
+ *
+ * @throws IllegalStateException
+ * Thrown if at least one directory node is not in state
+ * <code>NodeState.RUNNING</code>.
+ * @throws PuppeTorException
+ * Thrown if an I/O problem occurs while sending HUP signals.
+ */
+ public abstract void hupAllDirectories() throws PuppeTorException;
+
+ /**
+ * Attempts to shut down all running nodes. The method blocks until all
+ * shutdown requests have been sent and either returns, or throws the first
+ * exception that has been observed when shutting down nodes. The method can
+ * be assumed to return very quickly. If there are no running nodes in this
+ * network, this operation has no effect.
+ *
+ * @throws PuppeTorException
+ * Thrown if an I/O problem occurs while shutting down the
+ * nodes.
+ */
+ public abstract void shutdownNodes() throws PuppeTorException;
+
+ /**
+ * Attempts to start all nodes within a given timeout of
+ * <code>maximumTimeToWaitInMillis</code> milliseconds. The method returns
+ * as soon as all nodes have started and opened their control port so that
+ * we can connect to them. It returns a boolean that states whether the
+ * operation was either successful or has timed out. This operation can only
+ * be invoked, if all nodes in the network have written their configuration,
+ * i.e. are not in state <code>NodeState.CONFIGURING</code> anymore.
+ *
+ * @param maximumTimeToWaitInMillis
+ * The maximum time to wait in milliseconds. A positive value or
+ * zero restricts waiting to this time. Negative values are not
+ * allowed. Typical values are in the range of a few seconds.
+ * @return <code>true</code> if all nodes could be started successfully,
+ * <code>false</code> if a timeout has occured.
+ * @throws IllegalStateException
+ * Thrown if at least one node in the network is still in state
+ * <code>NodeState.CONFIGURING</code>.
+ * @throws IllegalArgumentException
+ * Thrown if a negative value is given for
+ * <code>maximumTimeToWaitInMillis</code>.
+ * @throws PuppeTorException
+ * Thrown if an I/O problem occurs while starting the nodes.
+ */
+ public abstract boolean startNodes(long maximumTimeToWaitInMillis)
+ throws PuppeTorException;
+
+ /**
+ * Writes the configurations for all nodes in the network to disk, including
+ * <code>torrc</code> and <code>approved-routers</code> files. This
+ * method is assumed to return very quickly. In case of a private network,
+ * <code>configureAsPrivateNetwork</code> should be invoked in advance to
+ * this method!
+ *
+ * @throws PuppeTorException
+ * Thrown if an I/O problem occurs while writing to the nodes'
+ * working directories.
+ */
+ public abstract void writeConfigurations() throws PuppeTorException;
+
+ /**
+ * Returns the working directory of this network configuration which is in
+ * <code>test-env/networkName/</code>.
+ *
+ * @return Working directory of this network.
+ */
+ public abstract File getWorkingDirectory();
+
+ /**
+ * Returns all configuration strings of the template of a node class that
+ * will be added to future instances of this node class and its subclasses.
+ * Note that the result only contains those configuration strings that are
+ * added by this node class to possible superclasses and that parameterized
+ * configuration strings, e.g. port configurations, are not included.
+ *
+ * @param nodeClass
+ * The class which will be configured with the returned template
+ * configuration; may not be <code>null</code>.
+ * @return The template configuration for the given node class.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given for the parameter.
+ */
+ public abstract List<String> getTemplateConfiguration(
+ Class<? extends ProxyNode> nodeClass);
+
+ /**
+ * Adds a configuration string to the template of a node class, so that it
+ * will be added to future instances of this node class and its subclasses.
+ *
+ * @param nodeClass
+ * The class of nodes of which future instances will have the
+ * given configuration string; may not be <code>null</code>.
+ * @param templateConfigurationString
+ * The configuration string to add; may neither be
+ * <code>null</code> nor a zero-length string, and must consist
+ * of configuration key and value.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given for either of the
+ * parameters.
+ */
+ public abstract void addTemplateConfiguration(
+ Class<? extends ProxyNode> nodeClass,
+ String templateConfigurationString);
+
+ /**
+ * Removes a configuration string from the template of a node class, so that
+ * it will not be added to future instances of this node class and its
+ * subclasses.
+ *
+ * @param nodeClass
+ * The class of nodes of which future instances will have the
+ * given configuration string; may not be <code>null</code>.
+ * @param templateConfigurationKey
+ * The configuration key to remove; may neither be
+ * <code>null</code> nor a zero-length key.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given for either of the
+ * parameters.
+ */
+ public abstract void removeTemplateConfiguration(
+ Class<? extends ProxyNode> nodeClass,
+ String templateConfigurationKey);
+
+ /**
+ * Returns the name of this network.
+ *
+ * @return The name of this network.
+ */
+ public abstract String getNetworkName();
+
+}
diff --git a/src/org/torproject/puppetor/NetworkFactory.java b/src/org/torproject/puppetor/NetworkFactory.java
new file mode 100644
index 0000000..2b93ed3
--- /dev/null
+++ b/src/org/torproject/puppetor/NetworkFactory.java
@@ -0,0 +1,83 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+import java.util.concurrent.ConcurrentHashMap;
+import java.util.concurrent.ConcurrentMap;
+
+import org.torproject.puppetor.impl.NetworkImpl;
+
+
+/**
+ * The <code>NetworkFactory</code> is a concrete factory that can create
+ * <code>Network</code> instances.
+ *
+ * TODO At the moment, this class uses the concrete class NetworkImpl to
+ * implement its only factory method. If we want to make this a real abstract
+ * factory, we need to replace the concrete constructor by reading the class
+ * name of the class implementing Network from a property file and invoking its
+ * constructor using reflection. Currently, this is the only place where we
+ * reference a class from the impl package.
+ *
+ * @author kloesing
+ */
+public abstract class NetworkFactory {
+
+ final private static ConcurrentMap<String, Network> networks =
+ new ConcurrentHashMap<String, Network>();
+
+ /**
+ * Creates a new network that is required for a test run. The new network is
+ * initially unpopulated and creates its own working directory at
+ * test-env/randomTestID/. The network automatically assigns port numbers to
+ * newly created nodes starting at <code>7000</code>.
+ *
+ * @param networkName
+ * Name of this network configuration.
+ * @return A new network instance.
+ */
+ public static Network createNetwork(final String networkName) {
+ final Network network = new NetworkImpl(networkName);
+ networks.put(networkName, network);
+ return network;
+ }
+
+ /**
+ * Creates a new network that is required for a test run. The new network is
+ * initially unpopulated and creates its own working directory at
+ * test-env/randomTestID/. The network automatically assigns port numbers to
+ * newly created nodes starting at <code>startPort</code>.
+ *
+ * @param networkName
+ * Name of this network configuration.
+ * @param startPort
+ * The initial value for automatically assigned port numbers of
+ * nodes created by this <code>Network</code>; must be a value
+ * between <code>1024</code> and <code>65535</code>.
+ * Applications need to ensure that there are enough ports left
+ * to the maximum number port <code>65535</code> for all
+ * created nodes.
+ * @return A new network instance.
+ */
+ public static Network createNetwork(final String networkName,
+ final int startPort) {
+ final Network network = new NetworkImpl(networkName, startPort);
+ networks.put(networkName, network);
+ return network;
+ }
+
+ /**
+ *
+ */
+ public static Network getNetworkByName(final String networkName) {
+ final Network network = networks.get(networkName);
+ if (network == null) {
+ throw new IllegalStateException("BUG: The network with name "
+ + networkName + " must exist and be non-null!");
+ }
+ return network;
+ }
+}
diff --git a/src/org/torproject/puppetor/NodeEventType.java b/src/org/torproject/puppetor/NodeEventType.java
new file mode 100644
index 0000000..7b65bd4
--- /dev/null
+++ b/src/org/torproject/puppetor/NodeEventType.java
@@ -0,0 +1,60 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+/**
+ * Event types that can be fired by all Tor processes.
+ */
+@SuppressWarnings("serial")
+public class NodeEventType implements EventType {
+
+ /**
+ * String identifying the type of the event type.
+ */
+ String typeString;
+
+ /**
+ * Creates a new event type with the given type string.
+ *
+ * @param typeString
+ * String identifying the type of the event type.
+ */
+ public NodeEventType(final String typeString) {
+ this.typeString = typeString;
+ }
+
+ public String getTypeName() {
+ return typeString;
+ }
+
+ /**
+ * The node was started and we managed to connect to its control port; this
+ * event is fired internally and not parsed from a log statement from Tor.
+ */
+ public static final NodeEventType NODE_STARTED =
+ new NodeEventType("NODE_STARTED");
+
+ /**
+ * The node has opened its control port; this event is parsed from a log
+ * statement in connection_create_listener().
+ */
+ public static final NodeEventType NODE_CONTROL_PORT_OPENED =
+ new NodeEventType("NODE_CONTROL_PORT_OPENED");
+
+ /**
+ * The node which has successfully opened a circuit; this event is parsed
+ * from a log statement in circuit_send_next_onion_skin().
+ */
+ public static final NodeEventType NODE_CIRCUIT_OPENED =
+ new NodeEventType("NODE_CIRCUIT_OPENED");
+
+ /**
+ * The node was stopped; this event is fired internally and not parsed from
+ * a log statement from Tor.
+ */
+ public static final NodeEventType NODE_STOPPED =
+ new NodeEventType("NODE_STOPPED");
+}
\ No newline at end of file
diff --git a/src/org/torproject/puppetor/NodeState.java b/src/org/torproject/puppetor/NodeState.java
new file mode 100644
index 0000000..720b44f
--- /dev/null
+++ b/src/org/torproject/puppetor/NodeState.java
@@ -0,0 +1,44 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+/**
+ * The <code>NodeState</code> constitutes the state of a single Tor node. In
+ * contrast to <code>EventType</code> the node states depend only on the
+ * methods that have been invoked on these objects, and not on asynchronous
+ * state changes. Most operations of <code>ProxyNode</code> and its subclasses
+ * require a certain <code>NodeState</code> as precondition and may ensure
+ * another <code>NodeState</code> as postcondition. There is a prescribed
+ * order of states.
+ *
+ * @author kloesing
+ */
+public enum NodeState {
+
+ /**
+ * The configuration of this node has not been written to disk. This is the
+ * initial state of a <code>ProxyNode</code> or one of its subclasses.
+ */
+ CONFIGURING,
+
+ /**
+ * The configuration of this node has been written to disk, but the Tor
+ * process has not been started, yet. This state could be useful to review
+ * the configuration that has been written to disk.
+ */
+ CONFIGURATION_WRITTEN,
+
+ /**
+ * The node has been started and is running.
+ */
+ RUNNING,
+
+ /**
+ * The node had been started and shut down. It cannot be started at a later
+ * time anymore.
+ */
+ SHUT_DOWN
+}
diff --git a/src/org/torproject/puppetor/ProxyNode.java b/src/org/torproject/puppetor/ProxyNode.java
new file mode 100644
index 0000000..78ac209
--- /dev/null
+++ b/src/org/torproject/puppetor/ProxyNode.java
@@ -0,0 +1,269 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+import java.util.List;
+
+/**
+ * <p>
+ * A <code>ProxyNode</code> represents a Tor process that is configured as
+ * onion proxy, i.e. to relay traffic from a local application to the Tor
+ * network and vice versa, and does not route traffic on behalf of remote
+ * applications. It is the superclass for other node types that extend the
+ * configuration of a </code>ProxyNode</code>.
+ * </p>
+ *
+ * <p>
+ * <b>Pay extra attention when using in private network!</b> Using proxy nodes
+ * in private networks in the same way as router nodes will fail! Tor has two
+ * different strategies for downloading network status documents: Directory
+ * caches (router nodes) download these documents after every HUP signal and
+ * then accept all contained router entries. But directory clients (proxy nodes)
+ * only download network status documents, if the most recent download lies at
+ * least 30 minutes in the past, and then accept only those of the contained
+ * router entries that are at least 10 minutes old. However, when starting all
+ * nodes of a private network at once, directories cannot contain 10 minutes old
+ * router descriptors. You have at least the following options to cope with this
+ * problem:
+ * </p>
+ *
+ * <ul>
+ * <li>Use router nodes instead of proxy nodes,</li>
+ * <li>start proxy nodes with a delay of at least 10 minutes to be sure that
+ * the router descriptors stored at directory authorities will be accepted by
+ * directory clients, or</li>
+ * <li>change the constants <code>ESTIMATED_PROPAGATION_TIME</code> and
+ * <code>NETWORKSTATUS_CLIENT_DL_INTERVAL</code> in Tor to values smaller than
+ * your overall HUP time for starting the network.</li>
+ * </ul>
+ *
+ * @author kloesing
+ */
+public interface ProxyNode {
+
+ /**
+ * Adds the entries for a hidden service to the configuration of this node.
+ *
+ * @param serviceName
+ * Name of the hidden service that will be used as name for the
+ * hidden service directory. May neither be <code>null</code>
+ * or a zero-length string.
+ * @param servicePort
+ * The TCP port on which the service will be available for
+ * requests. This can, but need not be different from the virtual
+ * port that is announced to clients. May not be negative or
+ * greater than 65535.
+ * @param virtualPort
+ * The virtual TCP port that this hidden service runs on as it is
+ * announced to clients. May not be negative or greater than
+ * 65535.
+ * @return <code>HiddenService</code> object containing the configuration
+ * of the created hidden service.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given for either of the
+ * parameters.
+ */
+ public abstract HiddenService addHiddenService(String serviceName,
+ int servicePort, int virtualPort);
+
+ /**
+ * Adds the entries for a hidden service with virtual port 80 to the
+ * configuration of this node.
+ *
+ * @param serviceName
+ * Name of the hidden service that will be used as name for the
+ * hidden service directory. May neither be <code>null</code>
+ * or a zero-length string.
+ * @param servicePort
+ * The TCP port on which the service will be available for
+ * requests. This can, but need not be different from the virtual
+ * port that is announced to clients. May not be negative or
+ * greater than 65535.
+ * @return <code>HiddenService</code> object containing the configuration
+ * of the created hidden service.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given for either of the
+ * parameters.
+ */
+ public abstract HiddenService addHiddenService(String serviceName,
+ int servicePort);
+
+ /**
+ * Adds the entries for a hidden service with an automatically assigned
+ * service port and virtual port 80 to the configuration of this node.
+ *
+ * service port automatically assigned virtual port 80
+ *
+ * @param serviceName
+ * Name of the hidden service that will be used as name for the
+ * hidden service directory. May neither be <code>null</code>
+ * or a zero-length string.
+ * @return <code>HiddenService</code> object containing the configuration
+ * of the created hidden service.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given for the parameter.
+ */
+ public abstract HiddenService addHiddenService(String serviceName);
+
+ /**
+ * Adds the given configuration string, consisting of "<configuration key>
+ * <configuration value>", to the configuration of this node.
+ *
+ * @param configurationString
+ * The configuration string to be added.
+ * @throws IllegalArgumentException
+ * Thrown if the given configuration string is either
+ * <code>null</code>, a zero-length string, or does not
+ * consist of configuration key and value.
+ */
+ public abstract void addConfiguration(String configurationString);
+
+ /**
+ * Adds the given configuration strings, each consisting of "<configuration
+ * key> <configuration value>", to the configuration of this node.
+ *
+ * @param configurationStrings
+ * A list of the configuration strings to be added.
+ * @throws IllegalArgumentException
+ * Thrown if the given list is <code>null</code>, or any of
+ * the contained strings is either <code>null</code>, a
+ * zero-length string, or does not consist of configuration key
+ * and value.
+ */
+ public abstract void addConfigurations(List<String> configurationStrings);
+
+ /**
+ * Replaces the first configuration string, consisting of "<configuration
+ * key> <configuration value>", that contains the same configuration key as
+ * <code>configurationString</code> by this new configuration string; if
+ * multiple occurrences of the given configuration key are found, only the
+ * first occurrence is replaced; if no configuration can be found, the
+ * configuration string is appended.
+ *
+ * @param configurationString
+ * The replacing configuration string.
+ * @throws IllegalArgumentException
+ * Thrown if the given configuration string is either
+ * <code>null</code>, a zero-length string, or does not
+ * consist of configuration key and value.
+ */
+ public abstract void replaceConfiguration(String configurationString);
+
+ /**
+ * Removes all configuration strings containing the given configuration key
+ * in "<configuration key> <configuration value>", regardless of their
+ * configuration value.
+ *
+ * @param configurationKey
+ * The configuration key to remove.
+ * @throws IllegalArgumentException
+ * Thrown if the given configuration key is either
+ * <code>null</code> or a zero-length key.
+ */
+ public abstract void removeConfiguration(String configurationKey);
+
+ /**
+ * Returns the name of this node.
+ *
+ * @return The name of this node.
+ */
+ public abstract String getNodeName();
+
+ /**
+ * Returns the state of this node.
+ *
+ * @return The state of this node.
+ */
+ public abstract NodeState getNodeState();
+
+ /**
+ * Sends a HUP command to the process via its control port to restart it;
+ * can only be done if the node has already been started, i.e. is in state
+ * <code>NodeState.RUNNING</code>!
+ *
+ * @throws PuppeTorException
+ * Thrown if an I/O problem occurs while sending the HUP signal.
+ * @throws IllegalStateException
+ * Thrown if node is not in state <code>NodeState.RUNNING</code>.
+ */
+ public abstract void hup() throws PuppeTorException;
+
+ /**
+ * Shuts down the Tor process corresponding to this node immediately. This
+ * is done by sending the <code>SHUTDOWN</code> signal twice, so that
+ * those nodes extending <code>ProxyNode</code> which have opened their OR
+ * port shutdown immediately, too.
+ *
+ * @throws IllegalStateException
+ * Thrown if this node is not in state
+ * <code>NodeState.RUNNING</code>.
+ * @throws PuppeTorException
+ * Thrown if an I/O problem occurs while sending the
+ * <code>SHUTDOWN</code> signal.
+ */
+ public abstract void shutdown() throws PuppeTorException;
+
+ /**
+ * Starts the Tor process for this node and connects to the control port as
+ * soon as it is opened. <b>In order for this method to succeed it is
+ * absolutely necessary, that logging on the console is not changed in the
+ * configuration of this node to a higher level than NOTICE, because the
+ * output is parsed to see when the control port is opened.</b>
+ *
+ * @param maximumTimeToWaitInMillis
+ * Maximum time in milliseconds we will wait for the Tor process
+ * to be started and the control port being opened. If this value
+ * is negative or zero, we will wait potentially forever.
+ * @return <code>true</code> if the node could be started successfully,
+ * <code>false</code> otherwise.
+ * @throws IllegalStateException
+ * Thrown if node is not in state
+ * <code>NodeState.CONFIGURATION_WRITTEN</code>, i.e. if
+ * either configuration has not been written or the process has
+ * already been started.
+ * @throws PuppeTorException
+ * Thrown if either the process could not be started, or the
+ * connection to the control port could not be established.
+ */
+ public abstract boolean startNode(long maximumTimeToWaitInMillis)
+ throws PuppeTorException;
+
+ /**
+ * Writes the configuration of this node to the <code>torrc</code> file in
+ * its working directory and changes the state to
+ * <code>NodeState.CONFIGURATION_WRITTEN</code>, if it was in state
+ * <code>NodeState.CONFIGURING</code> before.
+ *
+ * @throws PuppeTorException
+ * Thrown if the configuration file <code>torrc</code> cannot
+ * be written to disk.
+ */
+ public abstract void writeConfiguration() throws PuppeTorException;
+
+ /**
+ * Returns the SOCKS port of this node.
+ *
+ * @return The SOCKS port of this node.
+ */
+ public abstract int getSocksPort();
+
+ /**
+ * Returns the control port of this node.
+ *
+ * @return The control port of this node.
+ */
+ public abstract int getControlPort();
+
+ /**
+ * Returns (a copy of) the list of strings containing the configuration of
+ * this node.
+ *
+ * @return (A copy of) the list of strings containing the configuration of
+ * this node.
+ */
+ public abstract List<String> getConfiguration();
+
+}
diff --git a/src/org/torproject/puppetor/PuppeTorException.java b/src/org/torproject/puppetor/PuppeTorException.java
new file mode 100644
index 0000000..ff4ef72
--- /dev/null
+++ b/src/org/torproject/puppetor/PuppeTorException.java
@@ -0,0 +1,65 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+/**
+ * The <code>PuppeTorException</code> comprises all kinds of checked
+ * exceptions that occur when interacting with the JVM-external Tor processes or
+ * with the local file system. Any occurence of this exception denotes either a
+ * configuration problem that can only be solved outside of the JVM, or an
+ * unexpected problem. In contrast to this, all kinds of programming errors of
+ * an application using this API (invoking a method with wrong parameter values,
+ * in wrong state, etc.) will instead cause appropriate runtime exceptions from
+ * the Java API.
+ *
+ * @author kloesing
+ */
+@SuppressWarnings("serial")
+public class PuppeTorException extends Exception {
+
+ /**
+ * Creates a <code>PuppeTorException</code> without detail message or
+ * cause.
+ */
+ public PuppeTorException() {
+ super();
+ }
+
+ /**
+ * Creates a <code>PuppeTorException</code> with the given detail
+ * <code>message</code> and <code>cause</code>.
+ *
+ * @param message
+ * The detail message of this exception.
+ * @param cause
+ * The cause for this exception.
+ */
+ public PuppeTorException(final String message, final Throwable cause) {
+ super(message, cause);
+ }
+
+ /**
+ * Creates a <code>PuppeTorException</code> with the given detail
+ * <code>message</code>, but without a <code>cause</code>.
+ *
+ * @param message
+ * The detail message of this exception.
+ */
+ public PuppeTorException(final String message) {
+ super(message);
+ }
+
+ /**
+ * Creates a <code>PuppeTorException</code> with the given
+ * <code>cause</code>, but without a detail message.
+ *
+ * @param cause
+ * The cause for this exception.
+ */
+ public PuppeTorException(final Throwable cause) {
+ super(cause);
+ }
+}
diff --git a/src/org/torproject/puppetor/RouterNode.java b/src/org/torproject/puppetor/RouterNode.java
new file mode 100644
index 0000000..ea42a87
--- /dev/null
+++ b/src/org/torproject/puppetor/RouterNode.java
@@ -0,0 +1,55 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+/**
+ * A <code>RouterNode</code> represents a Tor process that is configured to
+ * both, relay traffic from a local application to the Tor network and to route
+ * traffic on behalf of remote applications. It inherits most of its
+ * configuration and behavior from its superclass <code>ProxyNode</code> and
+ * adds some router-specific configurations and behavior.
+ *
+ * @author kloesing
+ */
+public interface RouterNode extends ProxyNode {
+
+ /**
+ * Returns the dir port of this node.
+ *
+ * @return The dir port of this node.
+ */
+ public abstract int getDirPort();
+
+ /**
+ * Returns the onion port of this node.
+ *
+ * @return The onion port of this node.
+ */
+ public abstract int getOrPort();
+
+ /**
+ * <p>
+ * Returns the fingerprint string of this node, formatted like
+ * <code>nickname 0000 0000 0000 0000 0000 0000 0000 0000 0000 0000</code>.
+ * </p>
+ *
+ * <p>
+ * The fingerprint is determined by a background thread that is started as
+ * soon as the node is instantiated. If this background thread has not
+ * finished when this method is invoked, the invoking thread will be blocked
+ * until the fingerprint is available (or determining it has failed,
+ * whereupon an exception will be thrown).
+ * </p>
+ *
+ * @return The fingerprint of this node.
+ * @throws PuppeTorException
+ * Thrown if either the temporary <code>torrc.temp</code>
+ * configuration file cannot be written, the Tor process cannot
+ * be started temporarily, or the fingerprint file cannot be
+ * read.
+ */
+ public abstract String getFingerprint() throws PuppeTorException;
+}
diff --git a/src/org/torproject/puppetor/ServerApplication.java b/src/org/torproject/puppetor/ServerApplication.java
new file mode 100644
index 0000000..2859d97
--- /dev/null
+++ b/src/org/torproject/puppetor/ServerApplication.java
@@ -0,0 +1,58 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+/**
+ * The <code>ServerApplication</code> can be used as simple HTTP server that
+ * answers all <code>HTTP GET</code> requests by empty <code>HTTP OK</code>
+ * replies. Therefore, a thread will be started to listen for incoming requests
+ * in the background.
+ *
+ * @author kloesing
+ */
+public interface ServerApplication {
+
+ /**
+ * Starts listening for incoming <code>HTTP GET</code> requests from
+ * clients. Any incoming request is answered by an empty
+ * <code>HTTP OK</code> reply. This method may only be invoked when the
+ * server is currently not in listening state!
+ *
+ * @throws IllegalStateException
+ * Thrown if the server is currently not in listening state.
+ */
+ public abstract void startListening();
+
+ /**
+ * Stops listening for requests. This method may only be invoked when the
+ * server is currently in listening state!
+ *
+ * @throws IllegalStateException
+ * Thrown if the server is currently in listening state.
+ */
+ public abstract void stopListening();
+
+ /**
+ * Returns whether this server is currently in listening state.
+ *
+ * @return The listening state of this server.
+ */
+ public abstract boolean isListening();
+
+ /**
+ * Returns the name of this server.
+ *
+ * @return The name of this server.
+ */
+ public abstract String getServerApplicationName();
+
+ /**
+ * Returns the port on which this server listens.
+ *
+ * @return The port on which this server listens.
+ */
+ public abstract int getServerPort();
+}
diff --git a/src/org/torproject/puppetor/ServerEventType.java b/src/org/torproject/puppetor/ServerEventType.java
new file mode 100644
index 0000000..e98fd95
--- /dev/null
+++ b/src/org/torproject/puppetor/ServerEventType.java
@@ -0,0 +1,41 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor;
+
+/**
+ * Event types that can be fired by a server application running as thread in
+ * the background.
+ */
+@SuppressWarnings("serial")
+public class ServerEventType implements EventType {
+
+ /**
+ * String identifying the type of the event type.
+ */
+ String typeString;
+
+ /**
+ * Creates a new event type with the given type string.
+ *
+ * @param typeString
+ * String identifying the type of the event type.
+ */
+ public ServerEventType(final String typeString) {
+ this.typeString = typeString;
+ }
+
+ public String getTypeName() {
+ return typeString;
+ }
+
+ /**
+ * The server application has received a request and sent a reply to it;
+ * this event is fired internally and not parsed from a log statement from
+ * Tor.
+ */
+ public static final ServerEventType SERVER_RECEIVING_REQUEST_SENDING_REPLY =
+ new ServerEventType("SERVER_RECEIVING_REQUEST_SENDING_REPLY");
+}
\ No newline at end of file
diff --git a/src/org/torproject/puppetor/examples/AccessingPublicWebServerOverTor.java b/src/org/torproject/puppetor/examples/AccessingPublicWebServerOverTor.java
new file mode 100644
index 0000000..0d0d633
--- /dev/null
+++ b/src/org/torproject/puppetor/examples/AccessingPublicWebServerOverTor.java
@@ -0,0 +1,119 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.examples;
+
+import org.torproject.puppetor.ClientApplication;
+import org.torproject.puppetor.ClientEventType;
+import org.torproject.puppetor.Event;
+import org.torproject.puppetor.EventListener;
+import org.torproject.puppetor.EventManager;
+import org.torproject.puppetor.Network;
+import org.torproject.puppetor.NetworkFactory;
+import org.torproject.puppetor.ProxyNode;
+import org.torproject.puppetor.PuppeTorException;
+
+/**
+ * Example for accessing a public web server (here: <code>www.google.com</code>)
+ * over Tor to measure the access time.
+ *
+ * @author kloesing
+ */
+public class AccessingPublicWebServerOverTor {
+
+ /**
+ * Sets up and runs the test.
+ *
+ * @param args
+ * Command-line arguments (ignored).
+ * @throws PuppeTorException
+ * Thrown if there is a problem with the JVM-external Tor
+ * processes that we cannot handle.
+ */
+ public static void main(final String[] args) throws PuppeTorException {
+
+ // though we only need a single proxy, we always need to create a
+ // network to initialize a test case.
+ final Network network = NetworkFactory.createNetwork("example1");
+
+ // create a single proxy node with name "proxy"
+ final ProxyNode proxy = network.createProxy("proxy");
+
+ // write configuration of proxy node
+ network.writeConfigurations();
+
+ // start proxy node and wait until it has opened a circuit with a
+ // timeout of 5 seconds
+ if (!network.startNodes(5000)) {
+
+ // failed to start the proxy
+ System.out.println("Failed to start the node!");
+ return;
+ }
+ System.out.println("Successfully started the node!");
+
+ // hup until proxy has built circuits (5 retries, 10 seconds timeout
+ // each)
+ if (!network.hupUntilUp(5, 10000)) {
+
+ // failed to build circuits
+ System.out.println("Failed to build circuits!");
+ System.exit(0);
+ }
+ System.out.println("Successfully built circuits!");
+
+ // create client application
+ final ClientApplication client =
+ network.createClient("client", "www.google.com", 80, proxy
+ .getSocksPort());
+
+ // create event listener to listen for client application events
+ final EventListener clientEventListener = new EventListener() {
+
+ // remember time when request was sent
+ private long before;
+
+ public void handleEvent(Event event) {
+ if (event.getType() == ClientEventType.CLIENT_SENDING_REQUEST) {
+ before = System.currentTimeMillis();
+ } else if (event.getType() == ClientEventType.CLIENT_REPLY_RECEIVED) {
+ System.out.println("Request took "
+ + (System.currentTimeMillis() - before)
+ + " milliseconds");
+ }
+ }
+ };
+
+ // obtain reference to event manager to be able to respond to events
+ final EventManager manager = network.getEventManager();
+
+ // register event handler for client application events
+ manager.addEventListener(client.getClientApplicationName(),
+ clientEventListener);
+
+ // perform at most three request with a timeout of 20 seconds each
+ client.startRequests(3, 20000, true);
+
+ // block this thread as long as client requests are running
+ manager.waitForAnyOccurence(client.getClientApplicationName(),
+ ClientEventType.CLIENT_REQUESTS_PERFORMED);
+
+ // wait a second before shutting down the proxy
+ try {
+ Thread.sleep(1000);
+ } catch (final InterruptedException e) {}
+
+ // shut down proxy
+ network.shutdownNodes();
+
+ // wait another second before exiting the application
+ try {
+ Thread.sleep(1000);
+ } catch (final InterruptedException e) {}
+
+ // Shut down the JVM
+ System.out.println("Goodbye.");
+ }
+}
diff --git a/src/org/torproject/puppetor/examples/AdvertisingAndAccessingHiddenServiceOverPrivateTorNetwork.java b/src/org/torproject/puppetor/examples/AdvertisingAndAccessingHiddenServiceOverPrivateTorNetwork.java
new file mode 100644
index 0000000..d2d7016
--- /dev/null
+++ b/src/org/torproject/puppetor/examples/AdvertisingAndAccessingHiddenServiceOverPrivateTorNetwork.java
@@ -0,0 +1,132 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.examples;
+
+import org.torproject.puppetor.ClientApplication;
+import org.torproject.puppetor.ClientEventType;
+import org.torproject.puppetor.Event;
+import org.torproject.puppetor.EventListener;
+import org.torproject.puppetor.EventManager;
+import org.torproject.puppetor.HiddenService;
+import org.torproject.puppetor.HiddenServiceEventType;
+import org.torproject.puppetor.Network;
+import org.torproject.puppetor.NetworkFactory;
+import org.torproject.puppetor.PuppeTorException;
+import org.torproject.puppetor.RouterNode;
+import org.torproject.puppetor.ServerApplication;
+
+/**
+ * Example for advertising and accessing a hidden service over a private Tor
+ * network.
+ *
+ * @author kloesing
+ */
+public class AdvertisingAndAccessingHiddenServiceOverPrivateTorNetwork {
+
+ /**
+ * Sets up and runs the test.
+ *
+ * @param args
+ * Command-line arguments (ignored).
+ * @throws PuppeTorException
+ * Thrown if there is a problem with the JVM-external Tor
+ * processes that we cannot handle.
+ */
+ public static void main(final String[] args) throws PuppeTorException {
+
+ // create a network to initialize the test case
+ final Network network = NetworkFactory.createNetwork("example4");
+
+ // create three router nodes
+ final RouterNode router1 = network.createRouter("router1");
+ network.createRouter("router2");
+ final RouterNode router3 = network.createRouter("router3");
+
+ // create only one directory node
+ network.createDirectory("dir1");
+
+ // add hidden service
+ final HiddenService hidServ1 = router1.addHiddenService("hidServ");
+
+ // configure nodes of this network to be part of a private network
+ network.configureAsPrivateNetwork();
+
+ // write node configurations
+ network.writeConfigurations();
+
+ // start nodes and wait until they have opened a circuit with a timeout
+ // of 5 seconds
+ if (!network.startNodes(5000)) {
+
+ // failed to start the nodes
+ System.out.println("Failed to start nodes!");
+ System.exit(0);
+ }
+ System.out.println("Successfully started nodes!");
+
+ // hup until nodes have built circuits (60 retries, 10 seconds timeout
+ // each)
+ if (!network.hupUntilUp(60, 10000)) {
+
+ // failed to build circuits
+ System.out.println("Failed to build circuits!");
+ System.exit(0);
+ }
+ System.out.println("Successfully built circuits!");
+
+ // obtain reference to event manager to be able to respond to events
+ final EventManager manager = network.getEventManager();
+
+ // wait for 1 hour that the proxy has published its first RSD
+ if (!manager.waitForAnyOccurence(router1.getNodeName(),
+ HiddenServiceEventType.BOB_DESC_PUBLISHED_RECEIVED,
+ 1L * 60L * 60L * 1000L)) {
+ // failed to publish an RSD
+ System.out.println("Failed to publish an RSD!");
+ System.exit(0);
+ }
+ System.out.println("Successfully published an RSD!");
+
+ // create server application
+ final ServerApplication server =
+ network.createServer("server", hidServ1.getServicePort());
+
+ // create client application
+ final ClientApplication client =
+ network.createClient("client",
+ hidServ1.determineOnionAddress(), hidServ1
+ .getVirtualPort(), router3.getSocksPort());
+
+ // register event listener
+ final EventListener clientAndServerEventListener = new EventListener() {
+ public void handleEvent(Event event) {
+ System.out.println("Handling event: " + event.getMessage());
+ }
+ };
+ manager.addEventListener(client.getClientApplicationName(),
+ clientAndServerEventListener);
+ manager.addEventListener(server.getServerApplicationName(),
+ clientAndServerEventListener);
+
+ // start server
+ server.startListening();
+ System.out.println("Started server");
+
+ // perform at most five request with a timeout of 45 seconds each
+ client.startRequests(5, 45000, true);
+
+ // wait for request to be performed
+ manager.waitForAnyOccurence(client.getClientApplicationName(),
+ ClientEventType.CLIENT_REQUESTS_PERFORMED);
+
+ // shut down nodes
+ network.shutdownNodes();
+
+ // Shut down the JVM
+ System.out.println("Goodbye.");
+ System.exit(0);
+ }
+}
diff --git a/src/org/torproject/puppetor/examples/AdvertisingAndAccessingHiddenServiceOverPublicTorNetwork.java b/src/org/torproject/puppetor/examples/AdvertisingAndAccessingHiddenServiceOverPublicTorNetwork.java
new file mode 100644
index 0000000..87675c3
--- /dev/null
+++ b/src/org/torproject/puppetor/examples/AdvertisingAndAccessingHiddenServiceOverPublicTorNetwork.java
@@ -0,0 +1,149 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.examples;
+
+import org.torproject.puppetor.ClientApplication;
+import org.torproject.puppetor.ClientEventType;
+import org.torproject.puppetor.Event;
+import org.torproject.puppetor.EventListener;
+import org.torproject.puppetor.EventManager;
+import org.torproject.puppetor.HiddenService;
+import org.torproject.puppetor.HiddenServiceEventType;
+import org.torproject.puppetor.Network;
+import org.torproject.puppetor.NetworkFactory;
+import org.torproject.puppetor.ProxyNode;
+import org.torproject.puppetor.PuppeTorException;
+import org.torproject.puppetor.ServerApplication;
+import org.torproject.puppetor.ServerEventType;
+
+/**
+ * Example for advertising and accessing a hidden service over the public Tor
+ * network.
+ *
+ * @author kloesing
+ */
+public class AdvertisingAndAccessingHiddenServiceOverPublicTorNetwork {
+
+ /**
+ * Sets up and runs the test.
+ *
+ * @param args
+ * Command-line arguments (ignored).
+ * @throws PuppeTorException
+ * Thrown if there is a problem with the JVM-external Tor
+ * processes that we cannot handle.
+ */
+ public static void main(final String[] args) throws PuppeTorException {
+
+ // create a network to initialize the test case
+ final Network network = NetworkFactory.createNetwork("example3");
+
+ // create two proxy nodes
+ final ProxyNode proxy1 = network.createProxy("proxy1");
+ final ProxyNode proxy2 = network.createProxy("proxy2");
+
+ // add hidden service to the configuration of proxy1
+ final HiddenService hidServ1 = proxy1.addHiddenService("hidServ");
+
+ // write configuration of proxy nodes
+ network.writeConfigurations();
+
+ // start nodes and wait until they have opened a circuit with a timeout
+ // of 5 seconds
+ if (!network.startNodes(5000)) {
+
+ // failed to start the proxy
+ System.out.println("Failed to start nodes!");
+ System.exit(0);
+ }
+ System.out.println("Successfully started nodes!");
+
+ // hup until nodes have built circuits (5 retries, 10 seconds timeout
+ // each)
+ if (!network.hupUntilUp(5, 10000)) {
+
+ // failed to build circuits
+ System.out.println("Failed to build circuits!");
+ System.exit(0);
+ }
+ System.out.println("Successfully built circuits!");
+
+ // obtain reference to event manager to be able to respond to events
+ final EventManager manager = network.getEventManager();
+
+ // wait for 3 minutes that the proxy has published its first RSD
+ if (!manager.waitForAnyOccurence(proxy1.getNodeName(),
+ HiddenServiceEventType.BOB_DESC_PUBLISHED_RECEIVED,
+ 3L * 60L * 1000L)) {
+
+ // failed to publish an RSD
+ System.out.println("Failed to publish an RSD!");
+ System.exit(0);
+ }
+ System.out.println("Successfully published an RSD!");
+
+ // create server application
+ final ServerApplication server =
+ network.createServer("server", hidServ1.getServicePort());
+
+ // create client application
+ final ClientApplication client =
+ network.createClient("client",
+ hidServ1.determineOnionAddress(), hidServ1
+ .getVirtualPort(), proxy2.getSocksPort());
+
+ // create event listener to listen for client and server application
+ // events
+ final EventListener clientAndServerEventListener = new EventListener() {
+
+ private long requestReceivedAtServer;
+
+ // remember time when request was sent and when it was received
+ private long requestSentFromClient;
+
+ public void handleEvent(Event event) {
+ if (event.getType() == ClientEventType.CLIENT_SENDING_REQUEST) {
+ requestSentFromClient = event.getOccurrenceTime();
+ } else if (event.getType() == ServerEventType.SERVER_RECEIVING_REQUEST_SENDING_REPLY) {
+ requestReceivedAtServer = event.getOccurrenceTime();
+ System.out.println("Request took "
+ + (requestReceivedAtServer - requestSentFromClient)
+ + " milliseconds from client to server!");
+ } else if (event.getType() == ClientEventType.CLIENT_REPLY_RECEIVED) {
+ System.out
+ .println("Request took "
+ + (event.getOccurrenceTime() - requestSentFromClient)
+ + " milliseconds for the round-trip and "
+ + (event.getOccurrenceTime() - requestReceivedAtServer)
+ + " milliseconds from server to client!");
+ }
+ }
+ };
+
+ // register event handler for client and server application events
+ manager.addEventListener(client.getClientApplicationName(),
+ clientAndServerEventListener);
+ manager.addEventListener(server.getServerApplicationName(),
+ clientAndServerEventListener);
+
+ // start server
+ server.startListening();
+
+ // perform at most five request with a timeout of 45 seconds each
+ client.startRequests(5, 45000, true);
+
+ // block this thread as long as client requests are running
+ manager.waitForAnyOccurence(client.getClientApplicationName(),
+ ClientEventType.CLIENT_REQUESTS_PERFORMED);
+
+ // shut down proxy
+ network.shutdownNodes();
+
+ // Shut down the JVM
+ System.out.println("Goodbye.");
+ System.exit(0);
+ }
+}
diff --git a/src/org/torproject/puppetor/examples/AdvertisingHiddenServiceToPublicTorNetwork.java b/src/org/torproject/puppetor/examples/AdvertisingHiddenServiceToPublicTorNetwork.java
new file mode 100644
index 0000000..2f9773a
--- /dev/null
+++ b/src/org/torproject/puppetor/examples/AdvertisingHiddenServiceToPublicTorNetwork.java
@@ -0,0 +1,110 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.examples;
+
+import org.torproject.puppetor.Event;
+import org.torproject.puppetor.EventListener;
+import org.torproject.puppetor.EventManager;
+import org.torproject.puppetor.HiddenServiceEventType;
+import org.torproject.puppetor.Network;
+import org.torproject.puppetor.NetworkFactory;
+import org.torproject.puppetor.NodeEventType;
+import org.torproject.puppetor.ProxyNode;
+import org.torproject.puppetor.PuppeTorException;
+
+/**
+ * Example for advertising a hidden service to the public Tor network and
+ * observing the publication of rendezvous service descriptors.
+ *
+ * @author kloesing
+ */
+public class AdvertisingHiddenServiceToPublicTorNetwork {
+
+ /**
+ * Sets up and runs the test.
+ *
+ * @param args
+ * Command-line arguments (ignored).
+ * @throws PuppeTorException
+ * Thrown if there is a problem with the JVM-external Tor
+ * processes that we cannot handle.
+ */
+ public static void main(final String[] args) throws PuppeTorException {
+ // create a network to initialize the test case
+ final Network network = NetworkFactory.createNetwork("example2");
+
+ // create a single proxy node
+ final ProxyNode proxy = network.createProxy("proxy");
+
+ // add hidden service to the configuration
+ proxy.addHiddenService("hidServ");
+
+ // write configuration of proxy node
+ network.writeConfigurations();
+
+ // create event listener to listen for events from our proxy
+ final EventListener proxyEventListener = new EventListener() {
+
+ // remember time when request was sent
+ private long circuitOpened = -1;
+
+ public void handleEvent(Event event) {
+ if (event.getType() == NodeEventType.NODE_CIRCUIT_OPENED) {
+ if (circuitOpened == -1) {
+ circuitOpened = System.currentTimeMillis();
+ }
+ } else if (event.getType() == HiddenServiceEventType.BOB_DESC_PUBLISHED_RECEIVED) {
+ System.out.println("RSD published "
+ + (System.currentTimeMillis() - circuitOpened)
+ + " milliseconds after first circuit was opened");
+ }
+ }
+ };
+
+ // obtain reference to event manager to be able to respond to events
+ final EventManager manager = network.getEventManager();
+
+ // register event handler for proxy events
+ manager.addEventListener(proxy.getNodeName(), proxyEventListener);
+
+ // start proxy node and wait until it has opened a circuit with a
+ // timeout of 5 seconds
+ if (!network.startNodes(5000)) {
+
+ // failed to start the proxy
+ System.out.println("Failed to start the node!");
+ System.exit(0);
+ }
+ System.out.println("Successfully started the node!");
+
+ // hup until proxy has built circuits (5 retries, 10 seconds timeout
+ // each)
+ if (!network.hupUntilUp(5, 10000)) {
+
+ // failed to build circuits
+ System.out.println("Failed to build circuits!");
+ System.exit(0);
+ }
+ System.out.println("Successfully built circuits!");
+
+ // let it run for 2 minutes and observe when RSDs are published...
+ System.out
+ .println("Waiting for 2 minutes and observing RSD publications...");
+
+ try {
+ Thread.sleep(2L * 60L * 1000L);
+ } catch (final InterruptedException e) {
+ // do nothing
+ }
+
+ // shut down proxy
+ network.shutdownNodes();
+
+ // Shut down the JVM
+ System.out.println("Goodbye.");
+ System.exit(0);
+ }
+}
diff --git a/src/org/torproject/puppetor/impl/ClientApplicationImpl.java b/src/org/torproject/puppetor/impl/ClientApplicationImpl.java
new file mode 100644
index 0000000..ef4cbb4
--- /dev/null
+++ b/src/org/torproject/puppetor/impl/ClientApplicationImpl.java
@@ -0,0 +1,458 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.impl;
+
+import java.io.BufferedReader;
+import java.io.IOException;
+import java.io.InputStreamReader;
+import java.io.PrintStream;
+import java.net.InetSocketAddress;
+import java.net.Proxy;
+import java.net.Socket;
+import java.net.SocketTimeoutException;
+import java.net.Proxy.Type;
+import java.util.logging.Level;
+import java.util.logging.Logger;
+
+import org.torproject.puppetor.ClientApplication;
+import org.torproject.puppetor.ClientEventType;
+
+
+/**
+ * Implementation of <code>ClientApplication</code>.
+ *
+ * @author kloesing
+ */
+public class ClientApplicationImpl implements ClientApplication {
+
+ /**
+ * Internal thread class that is used to perform requests.
+ */
+ private class RequestThread extends Thread {
+
+ /**
+ * Flag to remember whether requests are performed at the moment (<code>true</code>),
+ * or have been stopped (<code>false</code>).
+ */
+ private boolean connected;
+
+ /**
+ * Number of retries to be performed.
+ */
+ private final int retries;
+
+ /**
+ * Flag that determines whether requests shall be stopped after the
+ * first successful reply (<code>true</code>), or not (<code>false</code>).
+ */
+ private final boolean stopOnSuccess;
+
+ /**
+ * Timeout in milliseconds for each retry.
+ */
+ private final long timeoutForEachRetry;
+
+ /**
+ * Creates a new thread, but does not start performing requests, yet.
+ *
+ * @param retries
+ * Number of retries to be performed.
+ * @param timeoutForEachRetry
+ * Timeout in milliseconds for each retry.
+ * @param stopOnSuccess
+ * Flag that determines whether requests shall be stopped
+ * after the first successful reply (<code>true</code>),
+ * or not (<code>false</code>).
+ */
+ RequestThread(final int retries, final long timeoutForEachRetry,
+ final boolean stopOnSuccess) {
+
+ // log entering
+ logger
+ .entering(this.getClass().getName(), "RequestThread",
+ new Object[] { retries, timeoutForEachRetry,
+ stopOnSuccess });
+
+ // check parameters
+ if (retries < 0 || timeoutForEachRetry < 0) {
+ final IllegalArgumentException e =
+ new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "RequestThread", e);
+ throw e;
+ }
+
+ // remember parameters
+ this.retries = retries;
+ this.timeoutForEachRetry = timeoutForEachRetry;
+ this.stopOnSuccess = stopOnSuccess;
+
+ // start connected
+ connected = true;
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "RequestThread");
+ }
+
+ /**
+ * Perform one or more requests.
+ */
+ @Override
+ public void run() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "run");
+
+ try {
+
+ // set Tor as proxy
+ final InetSocketAddress isa =
+ new InetSocketAddress("127.0.0.1", socksPort);
+ final Proxy p = new Proxy(Type.SOCKS, isa);
+
+ // create target address for socket -- don't resolve the target
+ // name to an IP address!
+ final InetSocketAddress hs =
+ InetSocketAddress.createUnresolved(targetName,
+ targetPort);
+
+ // start retry loop
+ for (int i = 0; connected && i < retries; i++) {
+
+ // log this try
+ logger.log(Level.FINE, "Trying to perform request");
+
+ // remember when we started
+ final long timeBeforeConnectionAttempt =
+ System.currentTimeMillis();
+
+ // send event to event manager
+ eventManager.observeInternalEvent(
+ timeBeforeConnectionAttempt,
+ getClientApplicationName(),
+ ClientEventType.CLIENT_SENDING_REQUEST,
+ "Sending request.");
+
+ Socket s = null;
+ try {
+
+ // create new socket using Tor as proxy
+ s = new Socket(p);
+
+ // try to connect to remote server
+ s.connect(hs, (int) timeoutForEachRetry);
+
+ // open output stream to write request
+ final PrintStream out =
+ new PrintStream(s.getOutputStream());
+ out.print("GET / HTTP/1.0\r\n\r\n");
+
+ // open input stream to read reply
+ final BufferedReader in =
+ new BufferedReader(new InputStreamReader(s
+ .getInputStream()));
+
+ // only read the first char in the response; this method
+ // blocks until there is a response
+ in.read();
+
+ // send event to event manager
+ eventManager.observeInternalEvent(System
+ .currentTimeMillis(),
+ getClientApplicationName(),
+ ClientEventType.CLIENT_REPLY_RECEIVED,
+ "Received response.");
+
+ // if we should stop on success, stop further connection
+ // attempts
+ if (stopOnSuccess) {
+ connected = false;
+ }
+
+ // clean up socket
+ in.close();
+ out.close();
+ s.close();
+
+ } catch (final SocketTimeoutException e) {
+
+ // log warning
+ logger.log(Level.WARNING,
+ "Connection to remote server timed out!", e);
+
+ // send event to event manager
+ eventManager.observeInternalEvent(System
+ .currentTimeMillis(),
+ getClientApplicationName(),
+ ClientEventType.CLIENT_GAVE_UP_REQUEST,
+ "Giving up request.");
+
+ // try again immediately, if there are retries left
+
+ } catch (final IOException e) {
+
+ // log warning
+ logger.log(Level.WARNING,
+ "Connection to remote server could not be "
+ + "established!", e);
+
+ // send event to event manager
+ eventManager.observeInternalEvent(System
+ .currentTimeMillis(),
+ getClientApplicationName(),
+ ClientEventType.CLIENT_GAVE_UP_REQUEST,
+ "Giving up request.");
+
+ // wait for the rest of the timeout
+ final long timeOfTimeoutLeft =
+ timeBeforeConnectionAttempt
+ + timeoutForEachRetry
+ - System.currentTimeMillis();
+ if (timeOfTimeoutLeft > 0) {
+ try {
+ Thread.sleep(timeOfTimeoutLeft);
+ } catch (final InterruptedException ex) {
+ // do nothing
+ }
+ }
+
+ } finally {
+ if (s != null) {
+ // close connection
+ try {
+
+ // try to close socket
+ logger.log(Level.FINER,
+ "Trying to close socket.");
+ s.close();
+ logger.log(Level.FINE, "Socket closed.");
+
+ } catch (final IOException e) {
+
+ // log warning
+ logger.log(Level.WARNING,
+ "Exception when trying to close socket!",
+ e);
+ }
+ }
+ }
+ }
+
+ } finally {
+
+ // we are done here
+ logger.log(Level.FINE, "Requests performed!");
+
+ // send event to event manager
+ eventManager.observeInternalEvent(System.currentTimeMillis(),
+ getClientApplicationName(),
+ ClientEventType.CLIENT_REQUESTS_PERFORMED,
+ "Requests performed.");
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "run");
+ }
+ }
+
+ /**
+ * Immediately stops this and all possibly subsequent requests.
+ */
+ public void stopRequest() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "stopRequest");
+
+ // change connected state to false and interrupt thread
+ connected = false;
+ interrupt();
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "stopRequest");
+ }
+ }
+
+ /**
+ * Name of this client application that is used as logger name of this node.
+ */
+ private final String clientApplicationName;
+
+ /**
+ * Thread that performs the requests in the background.
+ */
+ private RequestThread clientThread;
+
+ /**
+ * Event manager that handles all events concerning this client application.
+ */
+ private final EventManagerImpl eventManager;
+
+ /**
+ * Logger for this client which is called "client." plus the name of this
+ * client application.
+ */
+ private final Logger logger;
+
+ /**
+ * SOCKS port of the local Tor node to which requests are sent.
+ */
+ private final int socksPort;
+
+ /**
+ * Target name for the requests sent by this client; can be a publicly
+ * available URL or an onion address.
+ */
+ private final String targetName;
+
+ /**
+ * Target port for the requests sent by this client; can be either a server
+ * port or a virtual port of a hidden service.
+ */
+ private final int targetPort;
+
+ /**
+ * Creates a new HTTP client within this JVM, but does not start sending
+ * requests.
+ *
+ * @param network
+ * Network to which this HTTP client belongs; at the moment this
+ * is only used to determine the event manager instance.
+ * @param clientApplicationName
+ * Name of this client that is used as part of the logger name.
+ * @param targetName
+ * Target name for requests; can be either a server name/address
+ * or an onion address.
+ * @param targetPort
+ * Target port for requests; can be either a server port or a
+ * virtual port of a hidden service.
+ * @param socksPort
+ * SOCKS port of the local Tor node.
+ * @throws IllegalArgumentException
+ * If at least one of the parameters is <code>null</code> or
+ * has an invalid value.
+ */
+ ClientApplicationImpl(final NetworkImpl network,
+ final String clientApplicationName, final String targetName,
+ final int targetPort, final int socksPort) {
+
+ // check if clientApplicationName can be used as logger name
+ if (clientApplicationName == null
+ || clientApplicationName.length() == 0) {
+ throw new IllegalArgumentException(
+ "Invalid clientApplicationName: " + clientApplicationName);
+ }
+
+ // create logger
+ logger = Logger.getLogger("client." + clientApplicationName);
+
+ // log entering
+ logger.entering(this.getClass().getName(), "ClientApplicationImpl",
+ new Object[] { network, clientApplicationName, targetName,
+ targetPort, socksPort });
+
+ // check parameters
+ if (network == null || targetName == null || targetName.length() == 0
+ || targetPort < 0 || targetPort > 65535 || socksPort < 0
+ || socksPort > 65535) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "ClientApplicationImpl",
+ e);
+ throw e;
+ }
+
+ // remember parameters
+ this.clientApplicationName = clientApplicationName;
+ this.targetName = targetName;
+ this.targetPort = targetPort;
+ this.socksPort = socksPort;
+
+ // obtain and store reference on event manager
+ eventManager = network.getEventManagerImpl();
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "ClientApplicationImpl");
+ }
+
+ public synchronized void startRequests(final int retries,
+ final long timeoutForEachRetry, final boolean stopOnSuccess) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "performRequest",
+ new Object[] { retries, timeoutForEachRetry, stopOnSuccess });
+
+ // check parameters
+ if (retries <= 0 || timeoutForEachRetry < 0) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "performRequest", e);
+ throw e;
+ }
+
+ // check if we already have started a request (TODO change this to allow
+ // multiple requests in parallel? would be possible)
+ if (clientThread != null) {
+ final IllegalStateException e =
+ new IllegalStateException(
+ "Another request has already been started!");
+ logger.throwing(this.getClass().getName(), "performRequest", e);
+ throw e;
+ }
+
+ // create a thread that performs requests in the background
+ clientThread =
+ new RequestThread(retries, timeoutForEachRetry, stopOnSuccess);
+ clientThread.setName("Request Thread");
+ clientThread.setDaemon(true);
+ clientThread.start();
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "performRequest");
+ }
+
+ public synchronized void stopRequest() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "stopRequest");
+
+ // check if a request is running
+ if (clientThread == null) {
+ final IllegalStateException e =
+ new IllegalStateException("Cannot stop "
+ + "request, because no request has been started!");
+ logger.throwing(this.getClass().getName(), "stopRequest", e);
+ throw e;
+ }
+
+ // log this event
+ logger.log(Level.FINE, "Shutting down client");
+
+ // interrupt thread
+ clientThread.stopRequest();
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "stopRequest");
+ }
+
+ @Override
+ public String toString() {
+ return this.getClass().getSimpleName() + ": clientApplicationName=\""
+ + clientApplicationName + "\", targetAddress=\"" + targetName
+ + "\", targetPort=" + targetPort + ", socksPort=" + socksPort;
+ }
+
+ public String getClientApplicationName() {
+ return clientApplicationName;
+ }
+
+ public int getSocksPort() {
+ return socksPort;
+ }
+
+ public String getTargetName() {
+ return targetName;
+ }
+
+ public int getTargetPort() {
+ return targetPort;
+ }
+}
diff --git a/src/org/torproject/puppetor/impl/DirectoryNodeImpl.java b/src/org/torproject/puppetor/impl/DirectoryNodeImpl.java
new file mode 100644
index 0000000..b8cf15c
--- /dev/null
+++ b/src/org/torproject/puppetor/impl/DirectoryNodeImpl.java
@@ -0,0 +1,432 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.impl;
+
+import java.io.BufferedReader;
+import java.io.BufferedWriter;
+import java.io.File;
+import java.io.FileReader;
+import java.io.FileWriter;
+import java.io.IOException;
+import java.io.InputStream;
+import java.io.InputStreamReader;
+import java.io.OutputStreamWriter;
+import java.util.ArrayList;
+import java.util.List;
+import java.util.Set;
+import java.util.SortedSet;
+import java.util.TreeSet;
+import java.util.logging.Level;
+
+import org.torproject.puppetor.DirectoryNode;
+import org.torproject.puppetor.PuppeTorException;
+
+
+/**
+ * Implementation of <code>DirectoryNode</code>.
+ *
+ * @author kloesing
+ */
+public class DirectoryNodeImpl extends RouterNodeImpl implements DirectoryNode {
+
+ /**
+ * Executable file for generating v3 directory authority certificates.
+ *
+ * TODO make this configurable!
+ */
+ protected static final File torGencertExecutable = new File("tor-gencert");
+
+ /**
+ * Internal thread class that is used to generate v3 directory authority
+ * certificates in parallel, which can take a few seconds.
+ */
+ private class GenerateCertificateThread extends Thread {
+
+ @Override
+ public void run() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "run");
+
+ // run tor-gencert
+ final ProcessBuilder processBuilder =
+ new ProcessBuilder(torGencertExecutable.getPath(),
+ "--create-identity-key"// );
+ , "--passphrase-fd", "0");
+ final File workingDirectory =
+ new File(workingDir.getAbsolutePath() + File.separator
+ + "keys" + File.separator);
+
+ // create working directory
+ workingDirectory.mkdirs();
+
+ processBuilder.directory(workingDirectory);
+ processBuilder.redirectErrorStream(true);
+ Process tmpProcess = null;
+ try {
+ tmpProcess = processBuilder.start();
+ } catch (final IOException e) {
+ final PuppeTorException ex =
+ new PuppeTorException(
+ "Could not start tor-gencert process for generating a "
+ + "v3 directory authority certificate!",
+ e);
+ logger.log(Level.WARNING, "Could not start tor-gencert "
+ + "process for generating a v3 directory authority "
+ + "certificate!", ex);
+ setCaughtException(ex);
+ return;
+ }
+
+ final BufferedWriter writer =
+ new BufferedWriter(new OutputStreamWriter(tmpProcess
+ .getOutputStream()));
+ try {
+ writer.write("somepassword\n");
+ writer.close();
+ } catch (final IOException e1) {
+ System.out.println("Exception at write! " + e1.getMessage());
+ e1.printStackTrace();
+ }
+ // final InputStream read = tmpProcess.getErrorStream();
+
+ final InputStream stderr = tmpProcess.getInputStream();
+ final InputStreamReader isr = new InputStreamReader(stderr);
+ final BufferedReader br = new BufferedReader(isr);
+ String line = null;
+ try {
+ while ((line = br.readLine()) != null) {
+ ;
+ }
+ } catch (final IOException e1) {
+ e1.printStackTrace();
+ }
+
+ // wait for process to terminate
+ int exitValue = 0;
+ try {
+ exitValue = tmpProcess.waitFor();
+ } catch (final InterruptedException e) {
+ final PuppeTorException ex =
+ new PuppeTorException(
+ "Interrupted while waiting for tor-gencert process to exit!",
+ e);
+ logger.log(Level.WARNING,
+ "tor-gencert process was interrupted!", ex);
+ setCaughtException(ex);
+ return;
+ }
+
+ if (exitValue != 0) {
+ final PuppeTorException ex =
+ new PuppeTorException(
+ "Could not start tor-gencert process! tor-gencert exited with "
+ + "exit value " + exitValue + "!");
+ logger.log(Level.WARNING,
+ "Could not start tor-gencert process!", ex);
+ setCaughtException(ex);
+ return;
+ }
+
+ // read fingerprint from file
+ final File authorityCertificateFile =
+ new File(workingDirectory.getAbsolutePath()
+ + File.separator + "authority_certificate");
+ String identity;
+ try {
+ final BufferedReader br2 =
+ new BufferedReader(new FileReader(
+ authorityCertificateFile));
+ while ((line = br2.readLine()) != null
+ && !line.startsWith("fingerprint ")) {
+ ;
+ }
+ if (line == null) {
+ final PuppeTorException ex =
+ new PuppeTorException(
+ "Could not find fingerprint line in file "
+ + "authority_certificate!");
+ logger.log(Level.WARNING,
+ "Could not find fingerprint line in file "
+ + "authority_certificate!", ex);
+ setCaughtException(ex);
+ return;
+ }
+ identity = line.substring(line.indexOf(" ") + 1);
+ br2.close();
+ } catch (final IOException e) {
+ final PuppeTorException ex =
+ new PuppeTorException(
+ "Could not read fingerprint from file!", e);
+ logger.log(Level.WARNING, "Could not read fingerprint file!",
+ ex);
+ setCaughtException(ex);
+ return;
+ }
+
+ setV3Identity(identity);
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "run");
+ }
+ }
+
+ /**
+ * Set of routers that are approved by this directory node.
+ */
+ private final SortedSet<String> approvedRouters;
+
+ /**
+ * Creates a <code>DirectoryNodeImpl</code> and adds it to the given
+ * <code>network</code>, but does not yet write its configuration to disk
+ * or start the corresponding Tor process.
+ *
+ * @param network
+ * Network configuration to which this node belongs.
+ * @param nodeName
+ * The name of the new node which may only consist of between 1
+ * and 19 alpha-numeric characters.
+ * @param controlPort
+ * Port on which the Tor node will be listening for us as its
+ * controller. May not be negative or greater than 65535.
+ * @param socksPort
+ * Port on which the Tor node will be listening for SOCKS
+ * connection requests. May not be negative or greater than
+ * 65535.
+ * @param orPort
+ * Port on which the Tor node will be listening for onion
+ * requests by other Tor nodes. May not be negative or greater
+ * than 65535.
+ * @param dirPort
+ * Port on which the Tor node will be listening for directory
+ * requests from other Tor nodes. May not be negative or greater
+ * than 65535.
+ * @param serverIpAddress
+ * The IP address on which the node will listen. Must be a valid
+ * IP v4 address in dotted decimal notation. May not be
+ * <code>null</code>.
+ * @throws IllegalArgumentException
+ * If at least one of the parameters is <code>null</code> or
+ * has an invalid value.
+ * @throws PuppeTorException
+ * Thrown if an I/O problem occurs while writing the temporary
+ * <code>approved-routers</code> file.
+ */
+ DirectoryNodeImpl(final NetworkImpl network, final String nodeName,
+ final int controlPort, final int socksPort, final int orPort,
+ final int dirPort, final String serverIpAddress) {
+ // create superclass instance; parameter checking is done in super
+ // constructor
+ super(network, nodeName, controlPort, socksPort, orPort, dirPort,
+ serverIpAddress);
+
+ // log entering
+ logger.entering(this.getClass().getName(), "DirectoryNodeImpl",
+ new Object[] { network, nodeName, controlPort, socksPort,
+ orPort, dirPort });
+
+ // initialize attribute
+ approvedRouters = new TreeSet<String>();
+
+ // extend configuration by template configuration of directory nodes
+ configuration.addAll(templateConfiguration);
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "DirectoryNodeImpl");
+ }
+
+ /**
+ * Invoked by the certificate generating thread: sets the generated v3
+ * identity string.
+ *
+ * @param v3Identity
+ * The generated v3 identity string.
+ */
+ private synchronized void setV3Identity(final String v3Identity) {
+ // log entering
+ logger.entering(this.getClass().getName(), "setV3Identity", v3Identity);
+
+ // remember fingerprint and notify all waiting threads
+ this.v3Identity = v3Identity;
+ notifyAll();
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "setV3Identity");
+ }
+
+ /**
+ * The generated v3 identity string.
+ */
+ private String v3Identity;
+
+ public synchronized String getV3Identity() throws PuppeTorException {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "getV3Identity");
+
+ // wait until either the v3 identity has been determined or an exception
+ // was caught
+ while (v3Identity == null && caughtException == null) {
+
+ try {
+ wait(500);
+ } catch (final InterruptedException e) {
+ // do nothing
+ }
+ }
+
+ if (caughtException != null) {
+ logger.throwing(this.getClass().getName(), "getV3Identity",
+ caughtException);
+ throw caughtException;
+ }
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "getV3Identity", v3Identity);
+ return v3Identity;
+ }
+
+ public synchronized String getDirServerString() throws PuppeTorException {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "getDirServerString");
+
+ // determine fingerprint
+ String fingerprint = getFingerprint();
+
+ // cut off router nickname
+ fingerprint = fingerprint.substring(fingerprint.indexOf(" ") + 1);
+
+ // determine v3 identity
+ final String determinedV3Identity = getV3Identity();
+
+ // put everything together
+ final String dirServerString =
+ "DirServer " + nodeName + " v3ident=" + determinedV3Identity
+ + " orport=" + orPort + " " + serverIpAddress + ":"
+ + dirPort + " " + fingerprint;
+
+ // log exiting and return dir server string
+ logger.exiting(this.getClass().getName(), "getDirServerString",
+ dirServerString);
+ return dirServerString;
+ }
+
+ public void addApprovedRouters(final Set<String> routers) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "addApprovedRouters",
+ routers);
+
+ // check parameter
+ if (routers == null) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "addApprovedRouters", e);
+ throw e;
+ }
+
+ // add the given approved router strings to the sorted set of already
+ // known strings (if any)
+ approvedRouters.addAll(routers);
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "addApprovedRouters");
+ }
+
+ @Override
+ protected synchronized void determineFingerprint() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "determineFingerprint");
+
+ // start a thread to generate the directory's certificate
+ final GenerateCertificateThread certificateThread =
+ new GenerateCertificateThread();
+ certificateThread.setName(nodeName + " Certificate Generator");
+ certificateThread.start();
+
+ // wait (non-blocking) for the v3 identity string
+ try {
+ getV3Identity();
+ } catch (final PuppeTorException e1) {
+ final PuppeTorException ex =
+ new PuppeTorException("Could not read v3 identity string!",
+ e1);
+ caughtException = ex;
+ return;
+ }
+
+ // create an empty approved-routers file to make Tor happy
+ try {
+ new File(workingDir.getAbsolutePath() + File.separator
+ + "approved-routers").createNewFile();
+ } catch (final IOException e) {
+ final PuppeTorException ex =
+ new PuppeTorException(
+ "Could not write empty approved-routers file!", e);
+ caughtException = ex;
+ return;
+ }
+
+ // invoke overwritten method
+ super.determineFingerprint();
+ }
+
+ @Override
+ public synchronized void writeConfiguration() throws PuppeTorException {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "writeConfiguration");
+
+ // write approved-routers file
+ try {
+ final File approvedRoutersFile =
+ new File(workingDir.getAbsolutePath() + File.separator
+ + "approved-routers");
+ final BufferedWriter bw =
+ new BufferedWriter(new FileWriter(approvedRoutersFile));
+ for (final String approvedRouter : approvedRouters) {
+ bw.write(approvedRouter + "\n");
+ }
+ bw.close();
+ } catch (final IOException e) {
+ final PuppeTorException ex = new PuppeTorException(e);
+ logger
+ .throwing(this.getClass().getName(), "writeConfiguration",
+ ex);
+ throw ex;
+ }
+
+ // invoke overridden method
+ super.writeConfiguration();
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "writeConfiguration");
+ }
+
+ /**
+ * Template configuration of directory nodes.
+ */
+ static List<String> templateConfiguration;
+
+ static {
+ templateConfiguration = new ArrayList<String>();
+
+ // configure this node as an authoritative directory
+ templateConfiguration.add("AuthoritativeDirectory 1");
+ templateConfiguration.add("V2AuthoritativeDirectory 1");
+ templateConfiguration.add("V3AuthoritativeDirectory 1");
+ templateConfiguration.add("DirAllowPrivateAddresses 1");
+ templateConfiguration.add("MinUptimeHidServDirectoryV2 0 minutes");
+
+ // TODO This is now contained in proposal 135.
+ // templateConfiguration.add("AuthDirMaxServersPerAddr 0");
+ // templateConfiguration.add("AuthDirMaxServersPerAuthAddr 0");
+ // templateConfiguration.add("V3AuthVotingInterval 5 minutes");
+ // templateConfiguration.add("V3AuthVoteDelay 20 seconds");
+ // templateConfiguration.add("V3AuthDistDelay 20 seconds");
+ }
+}
diff --git a/src/org/torproject/puppetor/impl/EventImpl.java b/src/org/torproject/puppetor/impl/EventImpl.java
new file mode 100644
index 0000000..4c277c5
--- /dev/null
+++ b/src/org/torproject/puppetor/impl/EventImpl.java
@@ -0,0 +1,123 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.impl;
+
+import java.util.Date;
+
+import org.torproject.puppetor.Event;
+import org.torproject.puppetor.EventType;
+
+
+/**
+ * Implementation of <code>Event</code>.
+ *
+ * @author kloesing
+ */
+@SuppressWarnings("serial")
+public class EventImpl implements Event {
+
+ /**
+ * The source of this event.
+ */
+ private final String source;
+
+ /**
+ * The type of this event.
+ */
+ private EventType type;
+
+ /**
+ * Either the log message that led to firing this event, or an internal
+ * message.
+ */
+ private final String message;
+
+ /**
+ * The occurrence time of the event or of the corresponding log statement.
+ */
+ private long occurrenceTime;
+
+ /**
+ * Creates a new <code>EventImpl</code>.
+ *
+ * @param occurrenceTime
+ * The occurrence time of the event or of the corresponding log
+ * statement.
+ * @param source
+ * The source of this event.
+ * @param type
+ * The type of this event.
+ * @param message
+ * Either the log message that led to firing this event, or an
+ * internal message.
+ */
+ EventImpl(final long occurrenceTime, final String source,
+ final EventType type, final String message) {
+ this.occurrenceTime = occurrenceTime;
+ this.source = source;
+ this.type = type;
+ this.message = message;
+ }
+
+ /**
+ * Creates a new <code>EventImpl</code>.
+ *
+ * @param source
+ * The source of this event.
+ * @param message
+ * Either the log message that led to firing this event, or an
+ * internal message.
+ */
+ EventImpl(final String source, final String message) {
+ this.source = source;
+ this.message = message;
+ }
+
+ public String getSource() {
+ return source;
+ }
+
+ public EventType getType() {
+ return type;
+ }
+
+ /**
+ * Sets the event type to the given type.
+ *
+ * @param type
+ * The type of this event.
+ */
+ void setType(final EventType type) {
+ this.type = type;
+ }
+
+ public String getMessage() {
+ return message;
+ }
+
+ public long getOccurrenceTime() {
+ return occurrenceTime;
+ }
+
+ @Override
+ public String toString() {
+ return this.getClass().getSimpleName() + ": occurenceTime="
+ + new Date(occurrenceTime) + ", source=\"" + source
+ + "\", type=" + type.getTypeName() + ", message=\"" + message
+ + "\"";
+ }
+
+ /**
+ * Sets the occurrence time to the given time.
+ *
+ * @param occurrenceTime
+ * The occurrence time of the event or of the corresponding log
+ * statement.
+ */
+ void setOccurenceTime(final long occurrenceTime) {
+ this.occurrenceTime = occurrenceTime;
+ }
+}
diff --git a/src/org/torproject/puppetor/impl/EventManagerImpl.java b/src/org/torproject/puppetor/impl/EventManagerImpl.java
new file mode 100644
index 0000000..163e4d8
--- /dev/null
+++ b/src/org/torproject/puppetor/impl/EventManagerImpl.java
@@ -0,0 +1,730 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.impl;
+
+import java.text.ParsePosition;
+import java.text.SimpleDateFormat;
+import java.util.ArrayList;
+import java.util.Calendar;
+import java.util.Date;
+import java.util.HashMap;
+import java.util.HashSet;
+import java.util.LinkedList;
+import java.util.List;
+import java.util.Locale;
+import java.util.Map;
+import java.util.Set;
+import java.util.Map.Entry;
+import java.util.logging.Level;
+import java.util.logging.Logger;
+import java.util.regex.Matcher;
+import java.util.regex.Pattern;
+
+import org.torproject.puppetor.Event;
+import org.torproject.puppetor.EventListener;
+import org.torproject.puppetor.EventManager;
+import org.torproject.puppetor.EventType;
+import org.torproject.puppetor.HiddenServiceEventType;
+import org.torproject.puppetor.NodeEventType;
+
+
+/**
+ * Implementation of <code>EventManager</code>.
+ *
+ * @author kloesing
+ */
+public class EventManagerImpl implements EventManager {
+
+ /**
+ * Registered event handlers for specific sources.
+ */
+ private final Map<String, Set<EventListener>> eventHandlers;
+
+ /**
+ * Registered event handlers for all sources.
+ */
+ private final Set<EventListener> eventHandlersForAllSources;
+
+ /**
+ * Logger for this event manager which is called "event." plus the name of
+ * the network.
+ */
+ private final Logger logger;
+
+ /**
+ * Events observed so far.
+ */
+ private final Map<String, List<Event>> observedEvents;
+
+ /**
+ * Set of all registered event sources. This is required to ensure that
+ * requests for events from a given source specify valid event sources.
+ */
+ private final Set<String> eventSources;
+
+ /**
+ * Creates a new <code>EventManagerImpl</code> for the network with name
+ * <code>networkName</code> and initializes it.
+ *
+ * @param networkName
+ * Name of this event manager that is used as part of the logger
+ * name.
+ * @throws IllegalArgumentException
+ * Thrown if the given <code>networkName</code> is either
+ * <code>null</code> or a zero-length string.
+ */
+ EventManagerImpl(final String networkName) {
+
+ // check if networkName can be used as logger name
+ if (networkName == null || networkName.length() == 0) {
+ throw new IllegalArgumentException("Invalid networkName: "
+ + networkName);
+ }
+
+ // create logger
+ logger = Logger.getLogger("event." + networkName);
+
+ // log entering
+ logger.entering(this.getClass().getName(), "EventManagerImpl",
+ networkName);
+
+ // create data structures
+ observedEvents = new HashMap<String, List<Event>>();
+ eventHandlers = new HashMap<String, Set<EventListener>>();
+ eventHandlersForAllSources = new HashSet<EventListener>();
+ eventSources = new HashSet<String>();
+
+ // start thread to parse events
+ final Thread eventParseThread = new Thread() {
+ @Override
+ public void run() {
+ while (true) {
+ parseNextEvent();
+ }
+ }
+ };
+ eventParseThread.setDaemon(true);
+ eventParseThread.start();
+
+ // initialize event type patterns
+ initializeEventTypePatterns();
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "EventManagerImpl");
+ }
+
+ public synchronized List<Event> addEventListener(final String source,
+ final EventListener listener) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "addEventListener",
+ new Object[] { source, listener });
+
+ // check parameters
+ if (source == null || listener == null
+ || !eventSources.contains(source)) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "addEventListener", e);
+ throw e;
+ }
+
+ // if necessary, create new event listener set for source
+ if (!eventHandlers.containsKey(source)) {
+ eventHandlers.put(source, new HashSet<EventListener>());
+ }
+
+ // add listener
+ eventHandlers.get(source).add(listener);
+
+ // log change
+ logger.log(Level.FINE, "Added event listener for source " + source);
+
+ // log exiting and return
+ final List<Event> result = getEventHistory(source);
+ logger.exiting(this.getClass().getName(), "addEventListener", result);
+ return result;
+ }
+
+ public synchronized void addEventListener(final EventListener listener) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "addEventListener",
+ new Object[] { listener });
+
+ // check parameters
+ if (listener == null) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "addEventListener", e);
+ throw e;
+ }
+
+ // add listener
+ eventHandlersForAllSources.add(listener);
+
+ // log change
+ logger.log(Level.FINE, "Added event listener for all sources.");
+
+ // log exiting and return
+ logger.exiting(this.getClass().getName(), "addEventListener");
+ return;
+ }
+
+ public synchronized List<Event> getEventHistory(final String source) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "getEventHistory", source);
+
+ // check parameter
+ if (source == null || !eventSources.contains(source)) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "getEventHistory", e);
+ throw e;
+ }
+
+ // prepare result
+ final List<Event> result = new ArrayList<Event>();
+
+ // did we already observe events for this source?
+ if (observedEvents.containsKey(source)) {
+ // yes, add all events to result list
+ result.addAll(observedEvents.get(source));
+ }
+
+ // log exiting and return result
+ logger.exiting(this.getClass().getName(), "getEventHistory", result);
+ return result;
+ }
+
+ public synchronized boolean hasEventOccured(final String source,
+ final EventType type) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "hasEventOccured",
+ new Object[] { source, type });
+
+ // check parameters
+ if (source == null || type == null || !eventSources.contains(source)) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "hasEventOccured", e);
+ throw e;
+ }
+
+ // determine result
+ boolean result = false;
+ if (observedEvents.containsKey(source)) {
+ for (final Event event : observedEvents.get(source)) {
+ if (event.getType() == type) {
+ result = true;
+ break;
+ }
+ }
+ }
+
+ // log exiting and return result
+ logger.exiting(this.getClass().getName(), "hasEventOccured", result);
+ return result;
+ }
+
+ /**
+ * An ordered list of all log statements that are still unparsed.
+ */
+ private final List<EventImpl> unparsedLogStatements =
+ new LinkedList<EventImpl>();
+
+ /**
+ * Stores the occurrence of an unparsed Tor log events that might result in
+ * an event. All such unparsed events are later parsed by a background
+ * thread in invocation order. Then, the occurrence time and the event type
+ * will be parsed from the log message; if the log message does not contain
+ * anything of interest, the event will be discarded.
+ *
+ * @param source
+ * The event source.
+ * @param logMessage
+ * The log message.
+ * @throws IllegalArgumentException
+ * Thrown if the source is unknown.
+ */
+ synchronized void observeUnparsedEvent(final String source,
+ final String logMessage) {
+ unparsedLogStatements.add(new EventImpl(source, logMessage));
+ notifyAll();
+ }
+
+ /**
+ * Add an internal event to the event queue.
+ *
+ * @param occurrenceTime
+ * The occurrence time of the event.
+ * @param source
+ * The event source.
+ * @param type
+ * The event type.
+ * @param message
+ * The event message.
+ * @throws IllegalArgumentException
+ * Thrown if the source is unknown.
+ */
+ public synchronized void observeInternalEvent(final long occurrenceTime,
+ final String source, final EventType type, final String message) {
+
+ if (!eventSources.contains(source)) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "observeInternalEvent",
+ e);
+ throw e;
+ }
+
+ unparsedLogStatements.add(new EventImpl(occurrenceTime, source, type,
+ message));
+ notifyAll();
+ }
+
+ /**
+ * Parses a log statement coming from Tor and decides whether it is
+ * interesting for us.
+ */
+ void parseNextEvent() {
+
+ // wait for the next event in the queue
+ EventImpl event = null;
+ synchronized (this) {
+ while (unparsedLogStatements.isEmpty()) {
+ try {
+ wait();
+ } catch (final InterruptedException e) {}
+ }
+ event = unparsedLogStatements.remove(0);
+ }
+
+ // does the event contain a known source? if not, discard it
+ if (!eventSources.contains(event.getSource())) {
+ logger.log(Level.WARNING,
+ "Unknown event source while parsing an event: "
+ + event.getSource());
+ return;
+ }
+
+ // does the event require parsing? if not, process immediately
+ if (event.getType() != null) {
+ observeEvent(event);
+ } else {
+ final String line = event.getMessage();
+
+ /*
+ * the logging output of Tor does not contain a year component; put
+ * in the year at the time of parsing (which happens approx. 10 ms
+ * after the logging took place) in the good hope that this tool is
+ * not run at midnight on New Year's Eve...
+ */
+ final Calendar c = Calendar.getInstance();
+ final int currentYear = c.get(Calendar.YEAR);
+
+ // try to apply one of the event type patterns
+ for (final Entry<Pattern, EventType> entry : eventTypePatterns
+ .entrySet()) {
+ final Matcher matcher = entry.getKey().matcher(line);
+ if (matcher.find()) {
+ final SimpleDateFormat sdf =
+ new SimpleDateFormat("MMM dd HH:mm:ss.SSS",
+ Locale.US);
+ final Date logTime = sdf.parse(line, new ParsePosition(0));
+ c.setTimeInMillis(logTime.getTime());
+ c.set(Calendar.YEAR, currentYear);
+ final long t = c.getTimeInMillis();
+ event.setOccurenceTime(t);
+ event.setType(entry.getValue());
+ observeEvent(event);
+ break;
+ }
+ }
+ }
+ }
+
+ /**
+ * Map of all patterns, that should be included when parsing log statements
+ * coming from Tor, and the respective event types of the events that should
+ * be fired.
+ */
+ Map<Pattern, EventType> eventTypePatterns;
+
+ public synchronized void registerEventTypePattern(
+ final String patternString, final EventType eventType) {
+ eventTypePatterns.put(Pattern.compile(patternString), eventType);
+ }
+
+ /**
+ * Initializes the parsing engine with the standard log message patterns
+ * that should be included in current Tor. Any further patterns need to be
+ * added by the test application manually.
+ */
+ private void initializeEventTypePatterns() {
+ eventTypePatterns = new HashMap<Pattern, EventType>();
+ registerEventTypePattern("Opening Control listener on .*",
+ NodeEventType.NODE_CONTROL_PORT_OPENED);
+ registerEventTypePattern("Tor has successfully opened a circuit. "
+ + "Looks like client functionality is working.",
+ NodeEventType.NODE_CIRCUIT_OPENED);
+ registerEventTypePattern(
+ "Established circuit .* as introduction "
+ + "point for service .*",
+ HiddenServiceEventType.BOB_BUILT_INTRO_CIRC_SENDING_ESTABLISH_INTRO);
+ registerEventTypePattern("Received INTRO_ESTABLISHED cell on "
+ + "circuit .* for service .*",
+ HiddenServiceEventType.BOB_INTRO_ESTABLISHED_RECEIVED);
+ registerEventTypePattern("Sending publish request for hidden "
+ + "service .*", HiddenServiceEventType.BOB_SENDING_PUBLISH_DESC);
+ registerEventTypePattern("Uploaded rendezvous descriptor",
+ HiddenServiceEventType.BOB_DESC_PUBLISHED_RECEIVED);
+ registerEventTypePattern("Received INTRODUCE2 cell for service .* "
+ + "on circ .*", HiddenServiceEventType.BOB_INTRODUCE2_RECEIVED);
+ registerEventTypePattern("Done building circuit .* to rendezvous "
+ + "with cookie .* for service .*",
+ HiddenServiceEventType.BOB_BUILT_REND_CIRC_SENDING_RENDEZVOUS1);
+ registerEventTypePattern("begin is for rendezvous",
+ HiddenServiceEventType.BOB_APP_CONN_OPENED);
+ registerEventTypePattern("Got a hidden service request for ID '.*'",
+ HiddenServiceEventType.ALICE_ONION_REQUEST_RECEIVED);
+ registerEventTypePattern("Fetching rendezvous descriptor for "
+ + "service .*", HiddenServiceEventType.ALICE_SENDING_FETCH_DESC);
+ registerEventTypePattern("Received rendezvous descriptor",
+ HiddenServiceEventType.ALICE_DESC_FETCHED_RECEIVED);
+ registerEventTypePattern(
+ "Sending an ESTABLISH_RENDEZVOUS cell",
+ HiddenServiceEventType.ALICE_BUILT_REND_CIRC_SENDING_ESTABLISH_RENDEZVOUS);
+ registerEventTypePattern("Got rendezvous ack. This circuit is now "
+ + "ready for rendezvous",
+ HiddenServiceEventType.ALICE_RENDEZVOUS_ESTABLISHED_RECEIVED);
+ registerEventTypePattern("introcirc is open",
+ HiddenServiceEventType.ALICE_BUILT_INTRO_CIRC);
+ registerEventTypePattern("Sending an INTRODUCE1 cell",
+ HiddenServiceEventType.ALICE_SENDING_INTRODUCE1);
+ registerEventTypePattern("Received ack. Telling rend circ",
+ HiddenServiceEventType.ALICE_INTRODUCE_ACK_RECEIVED);
+ registerEventTypePattern(
+ "Got RENDEZVOUS2 cell from hidden service",
+ HiddenServiceEventType.ALICE_RENDEZVOUS2_RECEIVED_APP_CONN_OPENED);
+ registerEventTypePattern("Handling rendezvous descriptor post",
+ HiddenServiceEventType.DIR_PUBLISH_DESC_RECEIVED);
+ registerEventTypePattern("Handling rendezvous descriptor get",
+ HiddenServiceEventType.DIR_FETCH_DESC_RECEIVED);
+ registerEventTypePattern(
+ "Received an ESTABLISH_INTRO request on circuit .*",
+ HiddenServiceEventType.IPO_RECEIVED_ESTABLISH_INTRO_SENDING_INTRO_ESTABLISHED);
+ registerEventTypePattern(
+ "Received an INTRODUCE1 request on circuit .*",
+ HiddenServiceEventType.IPO_RECEIVED_INTRODUCE1_SENDING_INTRODUCE2_AND_INTRODUCE_ACK);
+ registerEventTypePattern(
+ "Received an ESTABLISH_RENDEZVOUS request on circuit .*",
+ HiddenServiceEventType.RPO_RECEIVED_ESTABLISH_RENDEZVOUS_SENDING_RENDEZVOUS_ESTABLISHED);
+ registerEventTypePattern(
+ "Got request for rendezvous from circuit .* to cookie .*",
+ HiddenServiceEventType.RPO_RECEIVING_RENDEZVOUS1_SENDING_RENDEZVOUS2);
+ }
+
+ /**
+ * Stores the given <code>event</code> from <code>source</code> to the
+ * event history and propagates its occurrence to all registered event
+ * handlers.
+ *
+ * @param event
+ * The observed event.
+ */
+ private synchronized void observeEvent(final Event event) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "observeEvent", event);
+
+ final String source = event.getSource();
+ logger.log(Level.FINE, "Observed event " + event + " from source "
+ + source + "!");
+
+ // remember observed event
+ if (!observedEvents.containsKey(event.getSource())) {
+ observedEvents.put(source, new ArrayList<Event>());
+ }
+ observedEvents.get(source).add(event);
+
+ // notify waiting threads
+ notifyAll();
+
+ // inform event listeners
+ if (eventHandlers.containsKey(source)) {
+
+ // make a copy of the event handler set, because some event handlers
+ // might want to remove themselves from this set while handling the
+ // event
+ final Set<EventListener> copyOfEventHandlers =
+ new HashSet<EventListener>(eventHandlers.get(source));
+
+ for (final EventListener eventHandler : copyOfEventHandlers) {
+
+ logger.log(Level.FINE, "Informing event listener "
+ + eventHandler + " about recently observed event "
+ + event + " from source " + source + "!");
+ eventHandler.handleEvent(event);
+
+ }
+ }
+
+ // make a copy of the event handler set for all sources, because some
+ // event handlers might want to remove themselves from this set while
+ // handling the event
+ final Set<EventListener> copyOfEventHandlersForAllSources =
+ new HashSet<EventListener>(eventHandlersForAllSources);
+
+ for (final EventListener eventHandler : copyOfEventHandlersForAllSources) {
+
+ logger.log(Level.FINE, "Informing event listener " + eventHandler
+ + " about recently observed event " + event
+ + " from source " + source + "!");
+ eventHandler.handleEvent(event);
+ }
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "observeEvent");
+ }
+
+ public synchronized void removeEventListener(
+ final EventListener eventListener) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "removeEventListener",
+ eventListener);
+
+ // check parameters
+ if (eventListener == null) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger
+ .throwing(this.getClass().getName(), "removeEventListener",
+ e);
+ throw e;
+ }
+
+ // don't know to which source this listener has been added (may to more
+ // than one), so remove it from all possible sets
+ for (final Set<EventListener> set : eventHandlers.values()) {
+ if (set.remove(eventListener)) {
+ logger.log(Level.FINE, "Removed event listener!");
+ }
+ }
+
+ // remove from event listeners for all sources
+ if (eventHandlersForAllSources.remove(eventListener)) {
+ logger.log(Level.FINE, "Removed event listener!");
+ }
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "removeEventListener");
+ }
+
+ public synchronized void waitForAnyOccurence(final String source,
+ final EventType event) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "waitForAnyOccurence",
+ new Object[] { source, event });
+
+ // check parameters
+ if (source == null || event == null || !eventSources.contains(source)) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger
+ .throwing(this.getClass().getName(), "waitForAnyOccurence",
+ e);
+ throw e;
+ }
+
+ // invoke overloaded method with maximumTimeToWaitInMillis of -1L which
+ // means to wait forever
+ waitForAnyOccurence(source, event, -1L);
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "waitForAnyOccurence");
+
+ }
+
+ public synchronized boolean waitForAnyOccurence(final String source,
+ final EventType event, final long maximumTimeToWaitInMillis) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "waitForAnyOccurence",
+ new Object[] { source, event, maximumTimeToWaitInMillis });
+
+ // check parameters
+ if (source == null || event == null || !eventSources.contains(source)) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger
+ .throwing(this.getClass().getName(), "waitForAnyOccurence",
+ e);
+ throw e;
+ }
+
+ // check if we have already observed the event
+ if (hasEventOccured(source, event)) {
+ logger.log(Level.FINE, "Waiting for any occurence of event "
+ + event + " returned immediately!");
+ logger.exiting(this.getClass().getName(), "waitForAnyOccurence",
+ true);
+ return true;
+ }
+
+ // invoke method that waits for next occurence of the event
+ final boolean result =
+ waitForNextOccurence(source, event, maximumTimeToWaitInMillis);
+
+ // log exiting and return result
+ logger
+ .exiting(this.getClass().getName(), "waitForAnyOccurence",
+ result);
+ return result;
+ }
+
+ public synchronized void waitForNextOccurence(final String source,
+ final EventType event) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "waitForNextOccurence",
+ new Object[] { source, event });
+
+ // check parameters
+ if (source == null || event == null || !eventSources.contains(source)) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "waitForNextOccurence",
+ e);
+ throw e;
+ }
+
+ // invoke overloaded method with maximumTimeToWaitInMillis of -1L which
+ // means to wait forever
+ waitForNextOccurence(source, event, -1L);
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "waitForNextOccurence");
+ }
+
+ public synchronized boolean waitForNextOccurence(final String source,
+ final EventType type, final long maximumTimeToWaitInMillis) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "waitForNextOccurence",
+ new Object[] { source, type, maximumTimeToWaitInMillis });
+
+ // check parameters
+ if (source == null || type == null || !eventSources.contains(source)) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "waitForNextOccurence",
+ e);
+ throw e;
+ }
+
+ // distinguish between negative waiting time (wait forever) and zero or
+ // positive waiting time
+ if (maximumTimeToWaitInMillis < 0) {
+
+ // wait forever
+ while (!hasEventOccured(source, type)) {
+
+ logger.log(Level.FINEST,
+ "We will wait infinetely for the next occurence of "
+ + "event type " + type + " from source "
+ + source + "...");
+ try {
+ wait();
+ } catch (final InterruptedException e) {
+ // don't handle
+ }
+
+ logger.log(Level.FINEST,
+ "We have been notified about an observed event while "
+ + "waiting for events of type " + type
+ + " from source " + source
+ + "; need to check whether the observed event "
+ + "is what we are looking for...");
+ }
+
+ logger.log(Level.FINE, "Waiting for occurence of event type "
+ + type + " succeeded!");
+
+ // log exiting and return result
+ logger.exiting(this.getClass().getName(), "waitForNextOccurence",
+ true);
+ return true;
+
+ }
+
+ // wait for the given time at most
+ final long endOfTime =
+ System.currentTimeMillis() + maximumTimeToWaitInMillis;
+ long timeLeft = 0;
+ while (!hasEventOccured(source, type)
+ && (timeLeft = endOfTime - System.currentTimeMillis()) > 0) {
+
+ logger.log(Level.FINEST, "We will wait for " + timeLeft
+ + " milliseconds for the next occurence of event type "
+ + type + " from source " + source + "...");
+
+ try {
+ wait(timeLeft);
+ } catch (final InterruptedException e) {
+ // don't handle
+ }
+
+ logger.log(Level.FINEST,
+ "We have been notified about an observed event while "
+ + "waiting for events of type " + type
+ + " from source " + source
+ + "; need to check whether the observed event "
+ + "is what we are looking for...");
+ }
+
+ // determine result
+ final boolean result = hasEventOccured(source, type);
+
+ logger.log(Level.FINE, "Waiting for next occurence of event type "
+ + type + " from source " + source
+ + (result ? " succeeded!" : " did not succeed!"));
+
+ // log exiting and return result
+ logger.exiting(this.getClass().getName(), "waitForNextOccurence",
+ result);
+ return result;
+ }
+
+ /**
+ * Adds the given <code>source</code> as possible event source.
+ *
+ * @param source
+ * The name of the node, client, or server to add.
+ * @throws IllegalArgumentException
+ * Thrown if there is already an event source with this name.
+ */
+ void addEventSource(final String source) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "addEventSource", source);
+
+ // check if source name is unique in this network
+ if (eventSources.contains(source)) {
+ final IllegalArgumentException e =
+ new IllegalArgumentException(
+ "There is already an event source with name "
+ + source + " in this network!");
+ logger.throwing(this.getClass().getName(), "addEventSource", e);
+ throw e;
+ }
+
+ // add event source name
+ eventSources.add(source);
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "addEventSource");
+ }
+
+ @Override
+ public String toString() {
+ return this.getClass().getSimpleName();
+ }
+}
diff --git a/src/org/torproject/puppetor/impl/HiddenServiceImpl.java b/src/org/torproject/puppetor/impl/HiddenServiceImpl.java
new file mode 100644
index 0000000..35fe787
--- /dev/null
+++ b/src/org/torproject/puppetor/impl/HiddenServiceImpl.java
@@ -0,0 +1,167 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.impl;
+
+import java.io.BufferedReader;
+import java.io.File;
+import java.io.FileReader;
+import java.io.IOException;
+import java.util.logging.Level;
+import java.util.logging.Logger;
+
+import org.torproject.puppetor.HiddenService;
+import org.torproject.puppetor.PuppeTorException;
+
+
+/**
+ * Implementation of <code>HiddenService</code>.
+ *
+ * @author kloesing
+ */
+public class HiddenServiceImpl implements HiddenService {
+
+ /**
+ * Logger for this node which is called "hidserv." plus the name of this
+ * hidden service.
+ */
+ protected Logger logger;
+
+ /**
+ * The node at which this hidden service is configured.
+ */
+ private final ProxyNodeImpl node;
+
+ /**
+ * Name of the hidden service that will be used as name for the hidden
+ * service directory.
+ */
+ private final String serviceName;
+
+ /**
+ * The TCP port on which the service will be available for requests.
+ */
+ private final int servicePort;
+
+ /**
+ * The virtual TCP port that this hidden service runs on as it is announced
+ * to clients.
+ */
+ private final int virtualPort;
+
+ /**
+ * Adds the entries for a hidden service to the configuration of this node.
+ *
+ * Creates a new <code>HiddenServiceImpl</code>.
+ *
+ * @param node
+ * The node at which this hidden service is configured.
+ * @param serviceName
+ * Name of the hidden service that will be used as name for the
+ * hidden service directory. May neither be <code>null</code>
+ * or a zero-length string.
+ * @param servicePort
+ * The TCP port on which the service will be available for
+ * requests. This can, but need not be different from the virtual
+ * port that is announced to clients. May not be negative or
+ * greater than 65535.
+ * @param virtualPort
+ * The virtual TCP port that this hidden service runs on as it is
+ * announced to clients. May not be negative or greater than
+ * 65535.
+ * @throws IllegalArgumentException
+ * Thrown if an invalid value is given for either of the
+ * parameters.
+ */
+ HiddenServiceImpl(final ProxyNodeImpl node, final String serviceName,
+ final int servicePort, final int virtualPort) {
+
+ // check if networkName can be used as logger name
+ if (serviceName == null || serviceName.length() == 0) {
+ throw new IllegalArgumentException("Invalid serviceName: "
+ + serviceName);
+ }
+
+ // create logger
+ logger = Logger.getLogger("hidserv." + serviceName);
+
+ // log entering
+ logger.entering(this.getClass().getName(), "HiddenServiceImpl",
+ new Object[] { node, serviceName, servicePort, virtualPort });
+
+ // check parameters
+ if (servicePort < 0 || servicePort > 65535 || virtualPort < 0
+ || virtualPort > 65535) {
+ logger.log(Level.SEVERE,
+ "Illegal argument when adding hidden service!");
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "addHiddenService", e);
+ throw e;
+ }
+
+ // store parameter values
+ this.node = node;
+ this.serviceName = serviceName;
+ this.servicePort = servicePort;
+ this.virtualPort = virtualPort;
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "HiddenServiceImpl");
+ }
+
+ public String determineOnionAddress() throws PuppeTorException {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "getOnionAddress");
+
+ // check if hidden service directory exists
+ final File hiddenServiceFile =
+ new File(node.getWorkingDir().getAbsolutePath()
+ + File.separator + serviceName + File.separator
+ + "hostname");
+ if (!hiddenServiceFile.exists()) {
+ logger.log(Level.SEVERE,
+ "Hidden service directory or hostname file does not exist: "
+ + hiddenServiceFile.getAbsolutePath());
+
+ final PuppeTorException e =
+ new PuppeTorException(
+ "Hidden service directory or hostname file does not exist: "
+ + hiddenServiceFile.getAbsolutePath());
+ logger.throwing(this.getClass().getName(), "getOnionAddress", e);
+ throw e;
+ }
+
+ // read hostname from file
+ String address = null;
+ try {
+ final BufferedReader br =
+ new BufferedReader(new FileReader(hiddenServiceFile));
+ address = br.readLine();
+ br.close();
+ } catch (final IOException e) {
+ final PuppeTorException ex =
+ new PuppeTorException("Could not read hostname file!", e);
+ logger.throwing(this.getClass().getName(), "getOnionAddress", ex);
+ throw ex;
+ }
+
+ // log exiting and return address
+ logger.exiting(this.getClass().getName(), "getOnionAddress", address);
+ return address;
+ }
+
+ public String getServiceName() {
+ return serviceName;
+ }
+
+ public int getServicePort() {
+ return servicePort;
+ }
+
+ public int getVirtualPort() {
+ return virtualPort;
+ }
+}
diff --git a/src/org/torproject/puppetor/impl/NetworkImpl.java b/src/org/torproject/puppetor/impl/NetworkImpl.java
new file mode 100644
index 0000000..8bda03d
--- /dev/null
+++ b/src/org/torproject/puppetor/impl/NetworkImpl.java
@@ -0,0 +1,1083 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.impl;
+
+import java.io.File;
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+import java.util.concurrent.ConcurrentHashMap;
+import java.util.logging.Level;
+import java.util.logging.Logger;
+
+import org.torproject.puppetor.ClientApplication;
+import org.torproject.puppetor.DirectoryNode;
+import org.torproject.puppetor.Event;
+import org.torproject.puppetor.EventListener;
+import org.torproject.puppetor.EventManager;
+import org.torproject.puppetor.Network;
+import org.torproject.puppetor.NodeEventType;
+import org.torproject.puppetor.NodeState;
+import org.torproject.puppetor.ProxyNode;
+import org.torproject.puppetor.PuppeTorException;
+import org.torproject.puppetor.RouterNode;
+import org.torproject.puppetor.ServerApplication;
+
+
+/**
+ * Implementation of <code>Network</code>.
+ *
+ * @author kloesing
+ */
+public class NetworkImpl implements Network {
+
+ /**
+ * Internal thread class that is used to start Tor processes in parallel.
+ */
+ private class NodeStarter extends Thread {
+
+ /**
+ * The exception, if one is caught while trying to start the node.
+ */
+ Exception caughtException;
+
+ /**
+ * The maximum time to wait for the Tor process to start in
+ * milliseconds.
+ */
+ private final long maximumTimeToWaitInMillis;
+
+ /**
+ * The node for which the Tor process shall be started.
+ */
+ private final ProxyNode node;
+
+ /**
+ * Flag that denotes whether starting the Tor process was successful.
+ */
+ boolean success = false;
+
+ /**
+ * Creates a new <code>NodeStarter</code> for node <code>node</code>
+ * that will wait for <code>maximumTimeToWaitInMillis</code>
+ * milliseconds to start a Tor process, but that is not started, yet.
+ *
+ * @param node
+ * The node for which the Tor process shall be started.
+ * @param maximumTimeToWaitInMillis
+ * The maximum time to wait for the Tor process to start in
+ * milliseconds.
+ */
+ NodeStarter(final ProxyNode node, final long maximumTimeToWaitInMillis) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "NodeStarter",
+ new Object[] { node, maximumTimeToWaitInMillis });
+
+ // store parameters
+ this.node = node;
+ this.maximumTimeToWaitInMillis = maximumTimeToWaitInMillis;
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "NodeStarter");
+ }
+
+ @Override
+ public void run() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "run");
+
+ try {
+ // try to start node
+ success = node.startNode(maximumTimeToWaitInMillis);
+ } catch (final PuppeTorException e) {
+ logger.log(Level.SEVERE,
+ "Caught an exception while starting node "
+ + node.toString() + "!");
+ caughtException = e;
+ }
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "run");
+ }
+ }
+
+ /**
+ * Event manager to which all events concerning this network are notified.
+ */
+ private final EventManagerImpl eventManager;
+
+ /**
+ * Logger for this network which is called "network." plus the name of this
+ * network.
+ */
+ private final Logger logger;
+
+ /**
+ * Contains the name of this network configuration which is the String
+ * conversion of System.currentTimeMillis() of the network creation time.
+ */
+ private final String networkName;
+
+ /**
+ * All nodes contained in this network.
+ */
+ private final Map<String, ProxyNode> nodes;
+
+ /**
+ * Directory that contains status information of all nodes contained in this
+ * network.
+ */
+ private final File workingDir;
+
+ /**
+ * The counter for automatically assigned port numbers created by this
+ * <code>Network</code>.
+ */
+ private int portCounter = 7000;
+
+ /**
+ * Creates an initially unpopulated Tor network and creates a new working
+ * directory for it at test-env/randomTestID/.
+ *
+ * @param networkName
+ * Name of this network configuration. May neither be
+ * <code>null</code> or a zero-length string.
+ * @param startPort
+ * The initial value for automatically assigned port numbers of
+ * nodes created by this <code>Network</code>; must be a value
+ * between 1024 and 65535.
+ * @throws IllegalArgumentException
+ * Thrown if the given <code>networkName</code> is either
+ * <code>null</code> or a zero-length string, or if an illegal
+ * number is given for <code>startPort</code>.
+ */
+ public NetworkImpl(final String networkName, final int startPort) {
+ // initialize using overloaded constructor
+ this(networkName);
+
+ // check if start port is valid
+ if (startPort < 1024 || startPort > 65535) {
+ throw new IllegalArgumentException("Invalid startPort: "
+ + startPort);
+ }
+
+ // remember parameter
+ portCounter = startPort;
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "NetworkImpl");
+ }
+
+ /**
+ * Creates an initially unpopulated Tor network and creates a new working
+ * directory for it at test-env/randomTestID/.
+ *
+ * @param networkName
+ * Name of this network configuration. May neither be
+ * <code>null</code> or a zero-length string.
+ * @throws IllegalArgumentException
+ * Thrown if the given <code>networkName</code> is either
+ * <code>null</code> or a zero-length string.
+ */
+ public NetworkImpl(final String networkName) {
+
+ // check if networkName can be used as logger name
+ if (networkName == null || networkName.length() == 0) {
+ throw new IllegalArgumentException("Invalid networkName: "
+ + networkName);
+ }
+
+ // create logger
+ logger = Logger.getLogger("network." + networkName);
+
+ // log entering
+ logger.entering(this.getClass().getName(), "NetworkImpl", networkName);
+
+ // TODO is this necessary?!
+ logger.setLevel(Level.ALL);
+
+ // remember parameter
+ this.networkName = networkName;
+
+ // create working directory
+ workingDir = new File("test-env/" + System.currentTimeMillis());
+ workingDir.mkdirs();
+ logger.log(Level.FINE, "Created working directory \""
+ + workingDir.getAbsolutePath() + "\"");
+
+ // TODO if we want to log to file, set this... somehow...
+ // this.logFile = new File(this.workingDir.getAbsolutePath()
+ // + "\\events.log");
+
+ // initialize data structures
+ nodes = new ConcurrentHashMap<String, ProxyNode>();
+
+ // create event manager
+ eventManager = new EventManagerImpl(this.networkName);
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "NetworkImpl");
+ }
+
+ /**
+ * Returns whether all nodes in this network are up, or not.
+ *
+ * @return <code>true</code> if all nodes are up, <code>false</code> if
+ * at least one node is not up.
+ */
+ private boolean allNodesUp() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "allNodesUp");
+
+ // fail on first node that is not up
+ for (final ProxyNode node : nodes.values()) {
+ if (!eventManager.hasEventOccured(node.getNodeName(),
+ NodeEventType.NODE_CIRCUIT_OPENED)) {
+
+ // log exiting and return false
+ logger.exiting(this.getClass().getName(), "allNodesUp");
+ return false;
+ }
+ }
+
+ // log exiting and return true
+ logger.exiting(this.getClass().getName(), "allNodesUp");
+ return true;
+ }
+
+ public void configureAsPrivateNetwork() throws PuppeTorException {
+ // log entering
+ logger.entering(this.getClass().getName(), "configureAsPrivateNetwork");
+
+ // read DirServer strings for all directories
+ final List<String> authorizedDirectoriesFingerprints =
+ new ArrayList<String>();
+ for (final ProxyNode node : nodes.values()) {
+ if (node instanceof DirectoryNode) {
+ final DirectoryNode dirNode = (DirectoryNode) node;
+ authorizedDirectoriesFingerprints.add(dirNode
+ .getDirServerString());
+ }
+ }
+
+ this.configureAsPartOfPrivateNetwork(authorizedDirectoriesFingerprints);
+
+ // read fingerprints for all directories and routers
+ final HashSet<String> approvedRoutersFingerprints =
+ new HashSet<String>();
+ for (final ProxyNode node : nodes.values()) {
+ if (node instanceof RouterNode) {
+ final RouterNode routerOrDirNode = (RouterNode) node;
+ approvedRoutersFingerprints.add(routerOrDirNode
+ .getFingerprint());
+ }
+ }
+
+ // write fingerprints for all directories and routers to the
+ // approved-routers file
+ for (final ProxyNode node : nodes.values()) {
+ if (node instanceof DirectoryNode) {
+ final DirectoryNode dirNode = (DirectoryNode) node;
+ dirNode.addApprovedRouters(approvedRoutersFingerprints);
+ }
+ }
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "configureAsPrivateNetwork");
+ }
+
+ public void configureAsPartOfPrivateNetwork(
+ List<String> authorizedDirectoriesFingerprints) {
+ //throws PuppeTorException {
+ // log entering
+ logger.entering(this.getClass().getName(), "configureAsPartOfPrivateNetwork");
+
+ for (final ProxyNode node : nodes.values()) {
+ if (node.getNodeState() == NodeState.CONFIGURING ||
+ node.getNodeState() == NodeState.CONFIGURATION_WRITTEN) {
+ // add to configuration
+ node.addConfiguration("TestingTorNetwork 1");
+ }
+ }
+
+ // configure nodes
+ for (final ProxyNode node : nodes.values()) {
+ if (node.getNodeState() == NodeState.CONFIGURING ||
+ node.getNodeState() == NodeState.CONFIGURATION_WRITTEN) {
+ // add to configuration
+ node.addConfigurations(authorizedDirectoriesFingerprints);
+ }
+ }
+
+/* // read fingerprints for all directories and routers
+ final HashSet<String> approvedRoutersFingerprints =
+ new HashSet<String>();
+ for (final ProxyNode node : nodes.values()) {
+ if (node instanceof RouterNode) {
+ final RouterNode routerOrDirNode = (RouterNode) node;
+ approvedRoutersFingerprints.add(routerOrDirNode
+ .getFingerprint());
+ }
+ }
+
+ // write fingerprints for all directories and routers to the
+ // approved-routers file
+ for (final ProxyNode node : nodes.values()) {
+ if (node instanceof DirectoryNode) {
+ final DirectoryNode dirNode = (DirectoryNode) node;
+ dirNode.addApprovedRouters(approvedRoutersFingerprints);
+ }
+ }
+*/
+ // log exiting
+ logger.exiting(this.getClass().getName(), "configureAsPartOfPrivateNetwork");
+ }
+
+ public ClientApplication createClient(final String clientApplicationName,
+ final String targetAddress, final int targetPort,
+ final int socksPort) {
+ // log entering
+ logger.entering(this.getClass().getName(), "createClient",
+ new Object[] { clientApplicationName, targetAddress,
+ targetPort, socksPort });
+
+ // create client; parameter checking is done in constructor
+ final ClientApplicationImpl client =
+ new ClientApplicationImpl(this, clientApplicationName,
+ targetAddress, targetPort, socksPort);
+
+ // add name to event manager as event source
+ eventManager.addEventSource(clientApplicationName);
+
+ // log exiting and return client
+ logger.exiting(this.getClass().getName(), "createClient", client);
+ return client;
+ }
+
+ public DirectoryNode createDirectory(final String nodeName,
+ final int controlPort, final int socksPort, final int orPort,
+ final int dirPort, final String serverIpAddress) {
+ // log entering
+ logger.entering(this.getClass().getName(), "createDirectory",
+ new Object[] { nodeName, controlPort, socksPort, orPort,
+ dirPort, serverIpAddress });
+
+ // create directory node; parameter checking is done in constructor
+ final DirectoryNode dir =
+ new DirectoryNodeImpl(this, nodeName, controlPort, socksPort,
+ orPort, dirPort, serverIpAddress);
+
+ // add new directory node to nodes collection
+ nodes.put(nodeName, dir);
+
+ // add name to event manager as event source
+ eventManager.addEventSource(nodeName);
+
+ // log exiting and return directory node
+ logger.exiting(this.getClass().getName(), "createDirectory", dir);
+ return dir;
+ }
+
+ public DirectoryNode createDirectory(final String nodeName,
+ final int controlPort, final int socksPort, final int orPort,
+ final int dirPort) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "createDirectory",
+ new Object[] { nodeName, controlPort, socksPort, orPort,
+ dirPort });
+
+ // invoke overloaded method
+ final DirectoryNode dir =
+ this.createDirectory(nodeName, controlPort, socksPort, orPort,
+ dirPort, "127.0.0.1");
+
+ // log exiting and return directory node
+ logger.exiting(this.getClass().getName(), "createDirectory", dir);
+ return dir;
+ }
+
+ public DirectoryNode createDirectory(final String nodeName,
+ final String serverIpAddress) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "createDirectory",
+ new Object[] { nodeName, serverIpAddress });
+
+ // invoke overloaded method
+ final DirectoryNode dir =
+ this.createDirectory(nodeName, portCounter++, portCounter++,
+ portCounter++, portCounter++, serverIpAddress);
+
+ // log exiting and return directory node
+ logger.exiting(this.getClass().getName(), "createDirectory", dir);
+ return dir;
+ }
+
+ public DirectoryNode createDirectory(final String nodeName) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "createDirectory", nodeName);
+
+ // invoke overloaded method
+ final DirectoryNode dir =
+ this.createDirectory(nodeName, portCounter++, portCounter++,
+ portCounter++, portCounter++, "127.0.0.1");
+
+ // log exiting and return directory node
+ logger.exiting(this.getClass().getName(), "createDirectory", dir);
+ return dir;
+ }
+
+ public ProxyNode createProxy(final String nodeName, final int controlPort,
+ final int socksPort) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "createProxy", new Object[] {
+ nodeName, controlPort, socksPort });
+
+ // create proxy node; parameter checking is done in constructor
+ final ProxyNode proxy =
+ new ProxyNodeImpl(this, nodeName, controlPort, socksPort);
+
+ // add new proxy node to nodes collection
+ nodes.put(nodeName, proxy);
+
+ // add name to event manager as event source
+ eventManager.addEventSource(nodeName);
+
+ // log exiting and return proxy node
+ logger.exiting(this.getClass().getName(), "createProxy", proxy);
+ return proxy;
+ }
+
+ public ProxyNode createProxy(final String nodeName) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "createProxy", nodeName);
+
+ // invoke overloaded method
+ final ProxyNode proxy =
+ this.createProxy(nodeName, portCounter++, portCounter++);
+
+ // log exiting and return proxy node
+ logger.exiting(this.getClass().getName(), "createProxy", proxy);
+ return proxy;
+ }
+
+ public RouterNode createRouter(final String nodeName,
+ final int controlPort, final int socksPort, final int orPort,
+ final int dirPort, final String serverIpAddress) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "createRouter",
+ new Object[] { nodeName, controlPort, socksPort, orPort,
+ dirPort, serverIpAddress });
+
+ // create router node; parameter checking is done in constructor
+ final RouterNode router =
+ new RouterNodeImpl(this, nodeName, controlPort, socksPort,
+ orPort, dirPort, serverIpAddress);
+
+ // add new router node to nodes collection
+ nodes.put(nodeName, router);
+
+ // add name to event manager as event source
+ eventManager.addEventSource(nodeName);
+
+ // log exiting and return router node
+ logger.exiting(this.getClass().getName(), "createRouter", router);
+ return router;
+ }
+
+ public RouterNode createRouter(final String nodeName,
+ final int controlPort, final int socksPort, final int orPort,
+ final int dirPort) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "createRouter",
+ new Object[] { nodeName, controlPort, socksPort, orPort,
+ dirPort });
+
+ // invoke overloaded method
+ final DirectoryNode dir =
+ this.createDirectory(nodeName, controlPort, socksPort, orPort,
+ dirPort, "127.0.0.1");
+
+ // log exiting and return directory node
+ logger.exiting(this.getClass().getName(), "createRouter", dir);
+ return dir;
+ }
+
+ public RouterNode createRouter(final String nodeName,
+ final String serverIpAddress) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "createRouter",
+ new Object[] { nodeName, serverIpAddress });
+
+ // invoke overloaded method
+ final RouterNode dir =
+ this.createRouter(nodeName, portCounter++, portCounter++,
+ portCounter++, portCounter++, serverIpAddress);
+
+ // log exiting and return directory node
+ logger.exiting(this.getClass().getName(), "createRouter", dir);
+ return dir;
+ }
+
+ public RouterNode createRouter(final String nodeName) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "createRouter", nodeName);
+
+ // invoke overloaded method
+ final RouterNode router =
+ this.createRouter(nodeName, portCounter++, portCounter++,
+ portCounter++, portCounter++, "127.0.0.1");
+
+ // log exiting and return router node
+ logger.exiting(this.getClass().getName(), "createRouter", router);
+ return router;
+ }
+
+ public ServerApplication createServer(final String serverApplicationName,
+ final int serverPort) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "createServer",
+ new Object[] { serverApplicationName, serverPort });
+
+ // create server; parameter checking is done in constructor
+ final ServerApplicationImpl server =
+ new ServerApplicationImpl(this, serverApplicationName,
+ serverPort);
+
+ // add name to event manager as event source
+ eventManager.addEventSource(serverApplicationName);
+
+ // log exiting and return server
+ logger.exiting(this.getClass().getName(), "createServer", server);
+ return server;
+ }
+
+ public ServerApplication createServer(final String serverApplicationName) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "createServer",
+ serverApplicationName);
+
+ // invoke overloaded method
+ final ServerApplication server =
+ this.createServer(serverApplicationName, portCounter++);
+
+ // log exiting and return server
+ logger.exiting(this.getClass().getName(), "createServer", server);
+ return server;
+ }
+
+ public EventManager getEventManager() {
+ return eventManager;
+ }
+
+ /**
+ * Returns the implementation instance of the event manager of this network.
+ *
+ * @return The implementation instance of the event manager of this network.
+ */
+ public EventManagerImpl getEventManagerImpl() {
+ return eventManager;
+ }
+
+ public File getWorkingDirectory() {
+ return workingDir;
+ }
+
+ public ProxyNode getNode(final String nodeName) {
+ return nodes.get(nodeName);
+ }
+
+ public Map<String, ProxyNode> getAllProxyNodes() {
+ final Map<String, ProxyNode> result = new HashMap<String, ProxyNode>();
+ for (final String nodeName : nodes.keySet()) {
+ final ProxyNode node = nodes.get(nodeName);
+ if (!(node instanceof RouterNode)) {
+ result.put(nodeName, node);
+ }
+ }
+ return result;
+ }
+
+ public Map<String, RouterNode> getAllRouterNodes() {
+ final Map<String, RouterNode> result =
+ new HashMap<String, RouterNode>();
+ for (final String nodeName : nodes.keySet()) {
+ final ProxyNode node = nodes.get(nodeName);
+ if (node instanceof RouterNode && !(node instanceof DirectoryNode)) {
+ result.put(nodeName, (RouterNode) node);
+ }
+ }
+ return result;
+ }
+
+ public Map<String, DirectoryNode> getAllDirectoryNodes() {
+ final Map<String, DirectoryNode> result =
+ new HashMap<String, DirectoryNode>();
+ for (final String nodeName : nodes.keySet()) {
+ final ProxyNode node = nodes.get(nodeName);
+ if (node instanceof DirectoryNode) {
+ result.put(nodeName, (DirectoryNode) node);
+ }
+ }
+ return result;
+ }
+
+ public Map<String, ProxyNode> getAllNodes() {
+ return new HashMap<String, ProxyNode>(nodes);
+ }
+
+ public boolean hupUntilUp(final int tries, final long hupInterval)
+ throws PuppeTorException {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "hupUntilUp", new Object[] {
+ tries, hupInterval });
+
+ // check if all nodes are running
+ for (final ProxyNode node : nodes.values()) {
+ if (node.getNodeState() != NodeState.RUNNING) {
+ final IllegalStateException e =
+ new IllegalStateException(
+ "All nodes must be running before sending them HUP "
+ + "commands!");
+ logger.throwing(this.getClass().getName(), "hupUntilUp", e);
+ throw e;
+ }
+ }
+
+ // check if nodes are already up; if so, return immediately
+ if (allNodesUp()) {
+
+ // log exiting and return true
+ logger.exiting(this.getClass().getName(), "hupUntilUp", true);
+ return true;
+ }
+
+ final Object hupLock = new Object();
+ // create and register a new event handler for each node
+ for (final ProxyNode node : nodes.values()) {
+ eventManager.addEventListener(node.getNodeName(),
+ new EventListener() {
+ public void handleEvent(final Event event) {
+ if (event.getType() == NodeEventType.NODE_CIRCUIT_OPENED) {
+ synchronized(hupLock) {
+ hupLock.notify();
+ }
+ eventManager.removeEventListener(this);
+ }
+ }
+ });
+ }
+
+ // walk through wait-check-hup loop until there are no tries left
+ for (int i = 0; i < tries; i++) {
+
+ // determine how long to try waiting for the hangup
+ final long endOfSleeping = System.currentTimeMillis() + hupInterval;
+
+ // unless all nodes have reported to be up, wait for the given
+ // maximum time
+ while (System.currentTimeMillis() < endOfSleeping) {
+
+ synchronized(hupLock) {
+ // check if nodes are up now
+ if (allNodesUp()) {
+
+ // log exiting and return true
+ logger.exiting(this.getClass().getName(), "hupUntilUp",
+ true);
+ return true;
+ }
+ // sleep
+ try {
+ hupLock.wait(hupInterval);
+ } catch (final InterruptedException e) {
+ // do nothing about it
+ }
+ }
+ }
+
+ logger.log(Level.FINE, "Sending HUP to nodes");
+ // send a HUP signal to all nodes
+ for (final ProxyNode node : nodes.values()) {
+ logger
+ .log(Level.FINE, "Sending HUP to node "
+ + node.toString());
+ node.hup();
+ }
+
+ // continue in loop
+ }
+
+ // no retries left and not all nodes are up; log exiting and return
+ // failure
+ logger.exiting(this.getClass().getName(), "hupUntilUp", false);
+ return false;
+ }
+
+ public void hupAllNodes() throws PuppeTorException {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "hupAllNodes");
+
+ // check if all nodes are running
+ for (final ProxyNode node : nodes.values()) {
+ if (node.getNodeState() != NodeState.RUNNING) {
+ final IllegalStateException e =
+ new IllegalStateException(
+ "All nodes must be running before sending them HUP "
+ + "commands!");
+ logger.throwing(this.getClass().getName(), "hupAllNodes", e);
+ throw e;
+ }
+ }
+
+ // send a HUP signal to all nodes
+ for (final ProxyNode node : nodes.values()) {
+ logger.log(Level.FINE, "Sending HUP to node " + node.toString());
+ node.hup();
+ }
+
+ // no retries left and not all nodes are up; log exiting and return
+ // failure
+ logger.exiting(this.getClass().getName(), "hupAllNodes");
+ }
+
+ public void hupAllDirectories() throws PuppeTorException {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "hupAllDirectories");
+
+ // check if all directory nodes are running
+ for (final ProxyNode node : nodes.values()) {
+ if (node instanceof DirectoryNode
+ && node.getNodeState() != NodeState.RUNNING) {
+ final IllegalStateException e =
+ new IllegalStateException(
+ "All directory nodes must be running before sending "
+ + "them HUP commands!");
+ logger.throwing(this.getClass().getName(), "hupAllDirectories",
+ e);
+ throw e;
+ }
+ }
+
+ // send a HUP signal to all nodes
+ for (final ProxyNode node : nodes.values()) {
+ if (node instanceof DirectoryNode) {
+ logger
+ .log(Level.FINE, "Sending HUP to node "
+ + node.toString());
+ node.hup();
+ }
+ }
+
+ // no retries left and not all nodes are up; log exiting and return
+ // failure
+ logger.exiting(this.getClass().getName(), "hupAllDirectories");
+ }
+
+ public void shutdownNodes() throws PuppeTorException {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "shutdownNodes");
+
+ // iteratively shut down all running nodes; if an exception is caught,
+ // continue shutting down the other nodes and throw the first exception
+ // subsequently
+ PuppeTorException firstCaughtException = null;
+ for (final ProxyNode node : nodes.values()) {
+ if (node.getNodeState() == NodeState.RUNNING) {
+ try {
+ node.shutdown();
+ } catch (final PuppeTorException e) {
+ if (firstCaughtException == null) {
+ firstCaughtException = e;
+ }
+ }
+ }
+ }
+
+ // if an exception was caught during shutting down nodes, throw the
+ // first caught exception
+ if (firstCaughtException != null) {
+ logger.throwing(this.getClass().getName(), "shutdownNodes",
+ firstCaughtException);
+ throw firstCaughtException;
+ }
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "shutdownNodes");
+ }
+
+ public boolean startNodes(final long maximumTimeToWaitInMillis)
+ throws PuppeTorException {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "startNodes",
+ maximumTimeToWaitInMillis);
+
+ // check node states
+ for (final ProxyNode node : nodes.values()) {
+ if (node.getNodeState() != NodeState.CONFIGURATION_WRITTEN) {
+ final IllegalStateException e =
+ new IllegalStateException(
+ "All configurations must be written before starting "
+ + "nodes!");
+ logger.throwing(this.getClass().getName(), "startNodes", e);
+ throw e;
+ }
+ }
+
+ // check parameter
+ if (maximumTimeToWaitInMillis < 0) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "startNodes", e);
+ throw e;
+ }
+
+ // remember time when we begin starting the nodes
+ final long before = System.currentTimeMillis();
+
+ // start nodes in parallel
+ final Set<NodeStarter> allNodeStarters = new HashSet<NodeStarter>();
+ for (final ProxyNode node : nodes.values()) {
+ final NodeStarter nodeStarter =
+ new NodeStarter(node, maximumTimeToWaitInMillis);
+ allNodeStarters.add(nodeStarter);
+ nodeStarter.start();
+ }
+
+ // wait for all node starts to complete
+ for (final NodeStarter nodeStarter : allNodeStarters) {
+
+ // join node starts one after the other
+ try {
+ nodeStarter.join();
+ } catch (final InterruptedException e) {
+ // this happens?! we have some kind of problem here!
+ logger.log(Level.WARNING,
+ "Interrupt while joining node starter!");
+
+ // log exiting and return false
+ logger.exiting(this.getClass().getName(), "startNodes", false);
+ return false;
+ }
+
+ // if any thread has caught an exception, throw that exception now
+ final Exception caughtException = nodeStarter.caughtException;
+ if (caughtException != null) {
+ final PuppeTorException ex =
+ new PuppeTorException("Exception while starting node "
+ + nodeStarter.node.getNodeName(),
+ caughtException);
+ logger.throwing(this.getClass().getName(), "startNodes", ex);
+ throw ex;
+ }
+
+ // if node start did not succeed in the given time, fail
+ if (!nodeStarter.success) {
+ logger.log(Level.WARNING,
+ "Starting nodes was not successful in "
+ + maximumTimeToWaitInMillis / 1000
+ + " seconds.", networkName);
+
+ // log exiting and return false
+ logger.exiting(this.getClass().getName(), "startNodes", false);
+ return false;
+ }
+ }
+
+ // determine how long we took to start all nodes
+ final long after = System.currentTimeMillis();
+ logger.log(Level.FINE, "Starting nodes was successful and took "
+ + (after - before) / 1000 + " seconds.", networkName);
+
+ // log exiting and return true
+ logger.exiting(this.getClass().getName(), "startNodes", true);
+ return true;
+ }
+
+ @Override
+ public String toString() {
+ return this.getClass().getSimpleName() + ": networkName=\""
+ + networkName;
+ }
+
+ public String getNetworkName() {
+ return networkName;
+ }
+
+ public void writeConfigurations() throws PuppeTorException {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "writeConfigurations");
+
+ // write configurations for all nodes
+ for (final ProxyNode node : nodes.values()) {
+ node.writeConfiguration();
+ }
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "writeConfigurations");
+ }
+
+ public void addTemplateConfiguration(
+ final Class<? extends ProxyNode> nodeClass,
+ final String templateConfigurationString) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "addTemplateConfiguration",
+ new Object[] { nodeClass, templateConfigurationString });
+
+ // check parameters
+ if (nodeClass == null || templateConfigurationString == null
+ || templateConfigurationString.length() < 1
+ || !templateConfigurationString.contains(" ")) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(),
+ "addTemplateConfiguration", e);
+ throw e;
+ }
+
+ // add template string to appropriate template configuration
+ if (nodeClass == ProxyNode.class) {
+ ProxyNodeImpl.templateConfiguration
+ .add(templateConfigurationString);
+ } else if (nodeClass == RouterNode.class) {
+ RouterNodeImpl.templateConfiguration
+ .add(templateConfigurationString);
+ } else if (nodeClass == DirectoryNode.class) {
+ DirectoryNodeImpl.templateConfiguration
+ .add(templateConfigurationString);
+ } else {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(),
+ "addTemplateConfiguration", e);
+ throw e;
+ }
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "addTemplateConfiguration");
+ }
+
+ public List<String> getTemplateConfiguration(
+ final Class<? extends ProxyNode> nodeClass) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "getTemplateConfiguration",
+ nodeClass);
+
+ // check parameter
+ if (nodeClass == null) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(),
+ "getTemplateConfiguration", e);
+ throw e;
+ }
+
+ // obtain reference on appropriate template configuration
+ List<String> result = null;
+ if (nodeClass == ProxyNode.class) {
+ result = new ArrayList<String>(ProxyNodeImpl.templateConfiguration);
+ } else if (nodeClass == RouterNode.class) {
+ result =
+ new ArrayList<String>(RouterNodeImpl.templateConfiguration);
+ } else if (nodeClass == DirectoryNode.class) {
+ result =
+ new ArrayList<String>(
+ DirectoryNodeImpl.templateConfiguration);
+ } else {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(),
+ "getTemplateConfiguration", e);
+ throw e;
+ }
+
+ // log exiting and return result
+ logger.exiting(this.getClass().getName(), "getTemplateConfiguration",
+ result);
+ return result;
+ }
+
+ public void removeTemplateConfiguration(
+ final Class<? extends ProxyNode> nodeClass,
+ final String templateConfigurationKey) {
+
+ // log entering
+ logger.entering(this.getClass().getName(),
+ "removeTemplateConfiguration", new Object[] { nodeClass,
+ templateConfigurationKey });
+
+ // check parameters
+ if (nodeClass == null || templateConfigurationKey == null
+ || templateConfigurationKey.length() < 1) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(),
+ "removeTemplateConfiguration", e);
+ throw e;
+ }
+
+ // obtain reference on appropriate template configuration
+ List<String> templateConfig = null;
+ if (nodeClass == ProxyNode.class) {
+ templateConfig = ProxyNodeImpl.templateConfiguration;
+ } else if (nodeClass == RouterNode.class) {
+ templateConfig = RouterNodeImpl.templateConfiguration;
+ } else if (nodeClass == DirectoryNode.class) {
+ templateConfig = DirectoryNodeImpl.templateConfiguration;
+ } else {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(),
+ "removeTemplateConfiguration", e);
+ throw e;
+ }
+
+ // iterate over existing template configuration strings and remove all
+ // configuration strings that have the given configuration key
+ final List<String> configurationStringsToRemove =
+ new ArrayList<String>();
+ for (final String currentConfigurationString : templateConfig) {
+ final String currentConfigurationKey =
+ currentConfigurationString.substring(0,
+ currentConfigurationString.indexOf(" "));
+ if (currentConfigurationKey.equals(templateConfigurationKey)) {
+ configurationStringsToRemove.add(currentConfigurationString);
+ }
+ }
+ templateConfig.removeAll(configurationStringsToRemove);
+
+ // log exiting
+ logger
+ .exiting(this.getClass().getName(),
+ "removeTemplateConfiguration");
+ }
+
+ /**
+ * Returns the current port number and increments it afterwards.
+ *
+ * @return The current port number.
+ */
+ int getNextPortNumber() {
+ return portCounter++;
+ }
+}
diff --git a/src/org/torproject/puppetor/impl/ProxyNodeImpl.java b/src/org/torproject/puppetor/impl/ProxyNodeImpl.java
new file mode 100644
index 0000000..5b65479
--- /dev/null
+++ b/src/org/torproject/puppetor/impl/ProxyNodeImpl.java
@@ -0,0 +1,735 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.impl;
+
+import java.io.BufferedReader;
+import java.io.BufferedWriter;
+import java.io.File;
+import java.io.FileWriter;
+import java.io.IOException;
+import java.io.InputStreamReader;
+import java.net.Socket;
+import java.util.ArrayList;
+import java.util.Iterator;
+import java.util.List;
+import java.util.concurrent.atomic.AtomicInteger;
+import java.util.logging.Level;
+import java.util.logging.Logger;
+
+import org.torproject.puppetor.HiddenService;
+import org.torproject.puppetor.NodeEventType;
+import org.torproject.puppetor.NodeState;
+import org.torproject.puppetor.ProxyNode;
+import org.torproject.puppetor.PuppeTorException;
+
+import net.freehaven.tor.control.TorControlConnection;
+
+/**
+ * Implementation of <code>ProxyNode</code>.
+ *
+ * @author kloesing
+ */
+public class ProxyNodeImpl implements ProxyNode {
+
+ /**
+ * Executable file containing Tor.
+ *
+ * TODO make this configurable!
+ */
+ protected static final File torExecutable = new File("tor");
+
+ /**
+ * The <code>torrc</code> configuration file of this Tor node.
+ */
+ protected File configFile;
+
+ /**
+ * Collects all configuration strings for this node during the configuration
+ * phase in the order they are added.
+ */
+ protected List<String> configuration;
+
+ /**
+ * Connection via Tor controller.
+ */
+ protected TorControlConnection conn;
+
+ /**
+ * Port on which the Tor node will be listening for us as its controller.
+ */
+ protected int controlPort;
+
+ /**
+ * Event manager to which all events concerning this node are notified.
+ */
+ private final EventManagerImpl eventManager;
+
+ /**
+ * Logger for this node which is called "node." plus the name of this node.
+ */
+ protected Logger logger;
+
+ /**
+ * Network to which this node belongs.
+ */
+ protected NetworkImpl network;
+
+ /**
+ * Name of this node that is used as part of the working directory, as
+ * logger name of this node, and as event source.
+ */
+ protected String nodeName;
+
+ /**
+ * The state of this node.
+ */
+ protected NodeState nodeState = NodeState.CONFIGURING;
+
+ /**
+ * Port on which the Tor node will be listening for SOCKS connection
+ * requests.
+ */
+ protected int socksPort;
+
+ /**
+ * The running Tor process that belongs to this node.
+ */
+ protected Process torProcess;
+
+ /**
+ * Directory in which all information concerning this node is stored.
+ */
+ protected File workingDir;
+
+ /**
+ * Returns this node's working directory.
+ *
+ * @return This node's working directory.
+ */
+ File getWorkingDir() {
+ return workingDir;
+ }
+
+ /**
+ * Creates a new <code>ProxyNodeImpl</code> and adds it to the given
+ * <code>network</code>, but does not yet write its configuration to disk
+ * or start the corresponding Tor process.
+ *
+ * @param network
+ * Network configuration to which this node belongs.
+ * @param nodeName
+ * The name of the new node which may only consist of between 1
+ * and 19 alpha-numeric characters.
+ * @param controlPort
+ * Port on which the Tor node will be listening for us as its
+ * controller. May not be negative or greater than 65535.
+ * @param socksPort
+ * Port on which the Tor node will be listening for SOCKS
+ * connection requests. May not be negative or greater than
+ * 65535.
+ * @throws IllegalArgumentException
+ * If at least one of the parameters is <code>null</code> or
+ * has an invalid value.
+ */
+ ProxyNodeImpl(final NetworkImpl network, final String nodeName,
+ final int controlPort, final int socksPort) {
+
+ // make sure that nodeName is a valid logger name
+ if (nodeName == null || nodeName.length() < 1 || nodeName.length() > 19
+ || !nodeName.matches("[a-zA-Z0-9]*")) {
+ final String reason =
+ "\"" + nodeName + "\" is not a valid node name!";
+ final IllegalArgumentException e =
+ new IllegalArgumentException(reason);
+ throw e;
+ }
+
+ // create logger
+ logger = Logger.getLogger(nodeName + "." + this.getClass().getName());
+
+ logger.setLevel(Level.ALL);
+
+ // log entering
+ logger.entering(this.getClass().getName(), "ProxyNodeImpl",
+ new Object[] { network, nodeName, controlPort, socksPort });
+
+ // check remaining parameters
+ if (network == null || controlPort < 0 || controlPort > 65535
+ || socksPort < 0 || socksPort > 65535) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "ProxyNodeImpl", e);
+ throw e;
+ }
+
+ // store parameter values
+ this.network = network;
+ this.nodeName = nodeName;
+ this.controlPort = controlPort;
+ this.socksPort = socksPort;
+
+ // obtain reference on event manager from network
+ eventManager = network.getEventManagerImpl();
+
+ // create working directory
+ workingDir =
+ new File(this.network.getWorkingDirectory().getAbsolutePath()
+ + File.separator + nodeName + File.separator);
+ workingDir.mkdirs();
+ logger.log(Level.FINE, "Created working directory \""
+ + workingDir.getAbsolutePath() + "\"");
+
+ // create reference on config file
+ configFile =
+ new File(workingDir.getAbsolutePath() + File.separator
+ + "torrc");
+
+ // initialize configuration
+ configuration = new ArrayList<String>(templateConfiguration);
+ configuration.add("ControlPort " + controlPort);
+ configuration.add("SocksPort " + socksPort);
+
+ // initialize state
+ nodeState = NodeState.CONFIGURING;
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "ProxyNodeImpl");
+ }
+
+ public void addConfiguration(final String configurationString) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "addConfiguration",
+ configurationString);
+
+ // check parameter
+ if (configurationString == null || configurationString.length() < 1
+ || !configurationString.contains(" ")) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "addConfiguration", e);
+ throw e;
+ }
+
+ // add configuration string
+ configuration.add(configurationString);
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "addConfiguration");
+ }
+
+ public void addConfigurations(final List<String> configurationStrings) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "addConfigurations",
+ configurationStrings);
+
+ // check parameter
+ if (configurationStrings == null) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "addConfigurations", e);
+ throw e;
+ }
+
+ // add configuration strings one by one
+ for (final String conf : configurationStrings) {
+ addConfiguration(conf);
+ }
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "addConfigurations");
+ }
+
+ public void replaceConfiguration(final String configurationString) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "replaceConfiguration",
+ configurationString);
+
+ // check parameter
+ if (configurationString == null || configurationString.length() < 1
+ || !configurationString.contains(" ")) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "replaceConfiguration",
+ e);
+ throw e;
+ }
+
+ // extract configuration key
+ final String configurationKey =
+ configurationString.substring(0, configurationString
+ .indexOf(" "));
+
+ // iterate over existing configuration strings and replace the first
+ // occurrence of configuration key with new configuration string
+ final Iterator<String> it = configuration.listIterator();
+ boolean replaced = false;
+ for (int counter = 0; !replaced && it.hasNext(); counter++) {
+ final String currentConfigurationString = it.next();
+ final String currentConfigurationKey =
+ currentConfigurationString.substring(0,
+ currentConfigurationString.indexOf(" "));
+ if (currentConfigurationKey.equals(configurationKey)) {
+ configuration.set(counter, configurationString);
+ replaced = true;
+ }
+ }
+
+ // if no such configuration key was found, append the configuration
+ // string
+ if (!replaced) {
+ configuration.add(configurationString);
+ }
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "replaceConfiguration");
+ }
+
+ public void removeConfiguration(final String configurationKey) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "deleteConfiguration",
+ configurationKey);
+
+ // check parameter
+ if (configurationKey == null || configurationKey.length() < 1) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger
+ .throwing(this.getClass().getName(), "deleteConfiguration",
+ e);
+ throw e;
+ }
+
+ // iterate over existing configuration strings and remove all
+ // configuration strings that have the given configuration key
+ final List<String> configurationStringsToRemove =
+ new ArrayList<String>();
+ for (final String currentConfigurationString : configuration) {
+ final String currentConfigurationKey =
+ currentConfigurationString.substring(0,
+ currentConfigurationString.indexOf(" "));
+ if (currentConfigurationKey.equals(configurationKey)) {
+ configurationStringsToRemove.add(currentConfigurationString);
+ }
+ }
+ configuration.removeAll(configurationStringsToRemove);
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "deleteConfiguration");
+ }
+
+ public synchronized HiddenService addHiddenService(
+ final String serviceName, final int servicePort,
+ final int virtualPort) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "addHiddenService",
+ new Object[] { serviceName, servicePort, virtualPort });
+
+ // create hidden service object; parameter checking is done in
+ // constructor
+ final HiddenService result =
+ new HiddenServiceImpl(this, serviceName, servicePort,
+ virtualPort);
+
+ // add hidden service using Tor controller
+ configuration.add("HiddenServiceDir " + workingDir.getAbsolutePath()
+ + File.separator + serviceName + "\nHiddenServicePort "
+ + virtualPort + " 127.0.0.1:" + servicePort);
+
+ // log exiting and return hidden service object
+ logger.exiting(this.getClass().getName(), "addHiddenService", result);
+ return result;
+ }
+
+ public synchronized HiddenService addHiddenService(
+ final String serviceName, final int servicePort) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "addHiddenService",
+ new Object[] { serviceName, servicePort });
+
+ // invoke overloaded method
+ final HiddenService result =
+ this.addHiddenService(serviceName, servicePort, 80);
+
+ // log exiting and return hidden service
+ logger.exiting(this.getClass().getName(), "addHiddenService", result);
+ return result;
+ }
+
+ public synchronized HiddenService addHiddenService(final String serviceName) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "addHiddenService",
+ serviceName);
+
+ // invoke overloaded method
+ final HiddenService result =
+ this.addHiddenService(serviceName, network.getNextPortNumber(),
+ 80);
+
+ // log exiting and return hidden service
+ logger.exiting(this.getClass().getName(), "addHiddenService", result);
+ return result;
+ }
+
+ public String getNodeName() {
+ return nodeName;
+ }
+
+ public synchronized NodeState getNodeState() {
+ return nodeState;
+ }
+
+ public synchronized void hup() throws PuppeTorException {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "hup");
+
+ // check state
+ if (nodeState != NodeState.RUNNING || conn == null) {
+ final IllegalStateException e =
+ new IllegalStateException(
+ "Cannot hup a process when it's not running or there is "
+ + "no connection to its control port!");
+ logger.throwing(this.getClass().getName(), "hup", e);
+ throw e;
+ }
+
+ // send HUP signal to Tor process
+ try {
+ conn.signal("HUP");
+ } catch (final IOException e) {
+ final PuppeTorException ex =
+ new PuppeTorException(
+ "Could not send the command HUP to the Tor process!",
+ e);
+ logger.throwing(this.getClass().getName(), "hup", ex);
+ throw ex;
+ }
+ // log exiting
+ logger.exiting(this.getClass().getName(), "hup");
+ }
+
+ public synchronized void shutdown() throws PuppeTorException {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "shutdown");
+
+ // check state
+ if (nodeState != NodeState.RUNNING) {
+ final IllegalStateException e = new IllegalStateException();
+ logger.throwing(this.getClass().getName(), "shutdown", e);
+ throw e;
+ }
+
+ // we cannot simply kill the Tor process, because we have established a
+ // controller connection to it which would interpret a closed socket as
+ // failure and throw a RuntimeException
+ try {
+ conn.shutdownTor("SHUTDOWN");
+ } catch (final IOException e) {
+ final PuppeTorException ex =
+ new PuppeTorException(
+ "Could not send shutdown command to Tor process!",
+ e);
+ logger.throwing(this.getClass().getName(), "shutdown", ex);
+ throw ex;
+ }
+
+ // change state
+ nodeState = NodeState.SHUT_DOWN;
+
+ // fire event
+ eventManager.observeInternalEvent(System.currentTimeMillis(),
+ getNodeName(), NodeEventType.NODE_STOPPED, "Node stopped.");
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "shutdown");
+ }
+
+ /**
+ * Helper thread that waits for a given time for a given process to
+ * potentially terminate in order to find out if there are problems. If
+ * either the process terminates cleanly within this timeout, or does not
+ * terminate, the exit value will be 0; otherwise it will contain the exit
+ * code of the terminated process. This functionality is added, because it
+ * is not provided by Process. XXX Some stuff in here looks still dodgy.
+ * What happens if we get an InterruptedException in run and thus don't set
+ * exitValue?-SH
+ */
+ private static class ProcessWaiter extends Thread {
+
+ /** The process to wait for. */
+ private final Process process;
+
+ /** The exit value or 0 if the process is still running. */
+ private final AtomicInteger exitValue;
+
+ /**
+ * Creates a new <code>ProcessWaiter</code> for process
+ * <code>process</code>, but does not start it, yet.
+ *
+ * @param process
+ * The process to wait for.
+ */
+ ProcessWaiter(final Process process) {
+ this.process = process;
+ exitValue = new AtomicInteger(0);
+ }
+
+ @Override
+ public void run() {
+ try {
+ exitValue.set(process.waitFor());
+ } catch (final InterruptedException e) {}
+ }
+
+ /**
+ * Causes the current thread to wait until the process has terminated or
+ * the <code>timeoutInMillis</code> has expired. This method returns
+ * immediately if the subprocess has already terminated.
+ *
+ * @param timeoutInMillis
+ * The maximum time to wait for the process to terminate.
+ * @return The exit value of the terminated process or 0 if the process
+ * is still running.
+ */
+ public int waitFor(final long timeoutInMillis) {
+ try {
+ sleep(timeoutInMillis);
+ } catch (final InterruptedException e) {}
+ interrupt();
+ return exitValue.get();
+ }
+ }
+
+ public synchronized boolean startNode(final long maximumTimeToWaitInMillis)
+ throws PuppeTorException {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "startNode",
+ maximumTimeToWaitInMillis);
+
+ // check state
+ if (nodeState != NodeState.CONFIGURATION_WRITTEN) {
+ final String reason =
+ "Node is not in state "
+ + "NodeState.CONFIGURATION_WRITTEN!";
+ final IllegalStateException e = new IllegalStateException(reason);
+ logger.throwing(this.getClass().getName(), "startNode", e);
+ throw e;
+ }
+
+ // start process
+ final ProcessBuilder processBuilder =
+ new ProcessBuilder(torExecutable.getPath(), "-f", "torrc");
+ processBuilder.directory(workingDir);
+ processBuilder.redirectErrorStream(true);
+ try {
+ torProcess = processBuilder.start();
+ logger.log(Level.FINE, "Started Tor process successfully!");
+ } catch (final IOException e) {
+ final String reason = "Could not start Tor process!";
+ final PuppeTorException ex = new PuppeTorException(reason, e);
+ logger.throwing(this.getClass().getName(), "startNode", ex);
+ throw ex;
+ }
+
+ // start thread to parse output
+ final BufferedReader br =
+ new BufferedReader(new InputStreamReader(torProcess
+ .getInputStream()));
+ final Thread outputThread = new Thread() {
+ @Override
+ public void run() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "run");
+
+ // read output from Tor to parse it
+ String line = null;
+ try {
+ while ((line = br.readLine()) != null) {
+ eventManager.observeUnparsedEvent(ProxyNodeImpl.this
+ .getNodeName(), line);
+ }
+ } catch (IOException e) {
+
+ // only print out a warning for this exception if this node
+ // is running; otherwise, silently ignore it...
+ if (getNodeState() == NodeState.RUNNING) {
+ String reason =
+ "IOException when reading output from Tor "
+ + "process "
+ + ProxyNodeImpl.this.getNodeName()
+ + "!";
+ logger.log(Level.WARNING, reason, e);
+ }
+ }
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "run");
+ }
+ };
+ outputThread.setDaemon(true);
+ outputThread.setName(nodeName + " Output Parser");
+ outputThread.start();
+ logger.log(Level.FINE, "Started thread to parse output!");
+
+ // add shutdown hook that kills the process on JVM exit
+ final Process p = torProcess;
+ Runtime.getRuntime().addShutdownHook(new Thread() {
+ @Override
+ public void run() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "run");
+
+ // destroy Tor process
+ p.destroy();
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "run");
+ }
+ });
+ logger.log(Level.FINER,
+ "Started shutdown hook that will destroy the Tor process on "
+ + "JVM exit!");
+
+ // wait to see if the process is started or exited immediately; wait for
+ // one second to be sure that Tor terminates if there is an error,
+ // especially if the computer is very busy and many nodes are created
+ final ProcessWaiter waiter = new ProcessWaiter(torProcess);
+ waiter.start();
+ final int exitValue = waiter.waitFor(1000);
+ if (exitValue != 0) {
+ // Tor did not manage to start correctly
+ logger.log(Level.WARNING, "Could not start Tor process! Tor "
+ + "exited with exit value " + exitValue
+ + "! Please go check the config options in " + configFile
+ + " manually!");
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "startNode", false);
+ return false;
+ }
+
+ // wait for Tor to open the control port
+ logger.log(Level.FINER, "Waiting for Tor to open its control port...");
+ if (!eventManager.waitForAnyOccurence(nodeName,
+ NodeEventType.NODE_CONTROL_PORT_OPENED,
+ maximumTimeToWaitInMillis)) {
+
+ // Tor did not open its control port
+ logger.log(Level.WARNING, "Tor node " + nodeName
+ + " did not manage to open its control port within "
+ + maximumTimeToWaitInMillis + " milliseconds!");
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "startNode", false);
+ return false;
+ }
+ logger.log(Level.FINE,
+ "Tor has successfully opened its control port and told us "
+ + "about that!");
+
+ // connect to the controller
+ logger.log(Level.FINER, "Connecting to control port...");
+ try {
+ final Socket controlSocket =
+ new java.net.Socket("127.0.0.1", controlPort);
+ conn = TorControlConnection.getConnection(controlSocket);
+ conn.authenticate(new byte[0]);
+ } catch (final IOException e) {
+ final String reason =
+ "Could not connect to control port " + controlPort + "!";
+ final PuppeTorException ex = new PuppeTorException(reason, e);
+ logger.throwing(this.getClass().getName(), "startNode", ex);
+ throw ex;
+ }
+ logger.log(Level.FINE, "Connected to control port successfully!");
+
+ // set state to RUNNING
+ nodeState = NodeState.RUNNING;
+
+ // fire event
+ eventManager.observeInternalEvent(System.currentTimeMillis(),
+ getNodeName(), NodeEventType.NODE_STARTED, "Node started.");
+
+ // log exiting and return with success
+ logger.exiting(this.getClass().getName(), "startNode", true);
+ return true;
+ }
+
+ @Override
+ public String toString() {
+ return this.getClass().getSimpleName() + ": nodeName=\"" + nodeName
+ + "\", controlPort=" + controlPort + ", socksPort=" + socksPort;
+ }
+
+ public synchronized void writeConfiguration() throws PuppeTorException {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "writeConfiguration");
+
+ // write config file
+ try {
+ final BufferedWriter bw =
+ new BufferedWriter(new FileWriter(configFile));
+ for (final String c : configuration) {
+ bw.write(c + "\n");
+ }
+ bw.close();
+ } catch (final IOException e) {
+ final PuppeTorException ex =
+ new PuppeTorException(
+ "Could not write configuration file!", e);
+ logger.throwing(this.getClass().getName(),
+ "writeConfigurationFile", ex);
+ throw ex;
+ }
+
+ // change state, if necessary
+ if (nodeState == NodeState.CONFIGURING) {
+ nodeState = NodeState.CONFIGURATION_WRITTEN;
+ }
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "writeConfiguration");
+ }
+
+ public int getSocksPort() {
+ return socksPort;
+ }
+
+ public int getControlPort() {
+ return controlPort;
+ }
+
+ public List<String> getConfiguration() {
+ return new ArrayList<String>(configuration);
+ }
+
+ /**
+ * Template configuration of proxy nodes.
+ */
+ static List<String> templateConfiguration;
+
+ static {
+ templateConfiguration = new ArrayList<String>();
+
+ templateConfiguration.add("DataDirectory .");
+ templateConfiguration.add("SafeLogging 0");
+ templateConfiguration.add("UseEntryGuards 0");
+
+ templateConfiguration.add("Log info stdout");
+ templateConfiguration.add("Log info file log");
+
+ // TODO This is now contained in proposal 135.
+ // templateConfiguration.add("EnforceDistinctSubnets 0");
+ // templateConfiguration.add("ClientDNSRejectInternalAddresses 0");
+ }
+}
diff --git a/src/org/torproject/puppetor/impl/RouterNodeImpl.java b/src/org/torproject/puppetor/impl/RouterNodeImpl.java
new file mode 100644
index 0000000..d291452
--- /dev/null
+++ b/src/org/torproject/puppetor/impl/RouterNodeImpl.java
@@ -0,0 +1,403 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.impl;
+
+import java.io.BufferedReader;
+import java.io.BufferedWriter;
+import java.io.File;
+import java.io.FileReader;
+import java.io.FileWriter;
+import java.io.IOException;
+import java.util.ArrayList;
+import java.util.List;
+import java.util.logging.Level;
+import java.util.regex.Pattern;
+
+import org.torproject.puppetor.PuppeTorException;
+import org.torproject.puppetor.RouterNode;
+
+
+/**
+ * Implementation of <code>RouterNode</code>.
+ *
+ * @author kloesing
+ */
+public class RouterNodeImpl extends ProxyNodeImpl implements RouterNode {
+
+ /**
+ * Internal thread class that is used to determine fingerprints in parallel,
+ * which can take a few seconds.
+ */
+ private class FingerprintThread extends Thread {
+
+ @Override
+ public void run() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "run");
+
+ // create file reference for temporary config file
+ final File tempConfigFile =
+ new File(workingDir.getAbsolutePath() + File.separator
+ + "torrc.tmp");
+
+ // compose a modified config file, including a DirServer option with
+ // false fingerprint; this is necessary, because otherwise Tor
+ // would not accept that this router node might have a private IP
+ // address, but connects to the public directory servers
+ final List<String> copyOfConfig =
+ new ArrayList<String>(configuration);
+ final String fakeDirServerString =
+ "DirServer "
+ + nodeName
+ + " 127.0.0.1:"
+ + dirPort
+ + " 0000 0000 0000 0000 0000 0000 0000 0000 0000 0000";
+ copyOfConfig.add(fakeDirServerString);
+
+ // write config file
+ try {
+ final BufferedWriter bw =
+ new BufferedWriter(new FileWriter(tempConfigFile));
+ for (final String c : copyOfConfig) {
+ bw.write(c + "\n");
+ }
+ bw.close();
+ } catch (final IOException e) {
+ final PuppeTorException ex =
+ new PuppeTorException(
+ "Could not write configuration file!", e);
+ logger.log(Level.WARNING, "Could not start Tor process!", ex);
+ setCaughtException(ex);
+ return;
+ }
+
+ // start process with option --list-fingerprint
+ final ProcessBuilder processBuilder =
+ new ProcessBuilder(torExecutable.getPath(),
+ "--list-fingerprint", "-f", "torrc.tmp");
+ processBuilder.directory(workingDir);
+ processBuilder.redirectErrorStream(true);
+ Process tmpProcess = null;
+ try {
+ tmpProcess = processBuilder.start();
+ } catch (final IOException e) {
+ final PuppeTorException ex =
+ new PuppeTorException(
+ "Could not start Tor process temporarily with "
+ + "--list-fingerprint option!", e);
+ logger.log(Level.WARNING, "Could not start Tor process!", ex);
+ setCaughtException(ex);
+ return;
+ }
+
+ // wait for process to terminate
+ int exitValue = 0;
+ try {
+ exitValue = tmpProcess.waitFor();
+ } catch (final InterruptedException e) {
+ final PuppeTorException ex =
+ new PuppeTorException(
+ "Interrupted while waiting for Tor process to exit!",
+ e);
+ logger.log(Level.WARNING,
+ "Temporary Tor process was interrupted!", ex);
+ setCaughtException(ex);
+ return;
+ }
+
+ if (exitValue != 0) {
+ final PuppeTorException ex =
+ new PuppeTorException(
+ "Could not start Tor process temporarily with "
+ + "--list-fingerprint option! Tor exited with "
+ + "exit value "
+ + exitValue
+ + "! Please go check the config options in "
+ + tempConfigFile + " manually!");
+ logger.log(Level.WARNING, "Could not start Tor process!", ex);
+ setCaughtException(ex);
+ return;
+ }
+
+ // read fingerprint from file
+ final File fingerprintFile =
+ new File(workingDir.getAbsolutePath() + File.separator
+ + "fingerprint");
+ try {
+ final BufferedReader br2 =
+ new BufferedReader(new FileReader(fingerprintFile));
+ setFingerprint(br2.readLine());
+ br2.close();
+ } catch (final IOException e) {
+ final PuppeTorException ex =
+ new PuppeTorException(
+ "Could not read fingerprint from file!", e);
+ logger.log(Level.WARNING, "Could not read fingerprint file!",
+ ex);
+ setCaughtException(ex);
+ return;
+ }
+
+ // delete temporary config file
+ tempConfigFile.delete();
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "run");
+ }
+ }
+
+ /**
+ * Invoked by the fingerprint thread: sets the determined fingerprint string
+ *
+ * @param fingerprint
+ * The determined fingerprint string.
+ */
+ private synchronized void setFingerprint(final String fingerprint) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "setFingerprint",
+ fingerprint);
+
+ // remember fingerprint and notify all waiting threads
+ this.fingerprint = fingerprint;
+ notifyAll();
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "setFingerprint");
+ }
+
+ /**
+ * Invoked by the fingerprint thread: sets the exception that occurred when
+ * trying to determine the fingerprint.
+ *
+ * @param caughtException
+ * The exception that occurred when trying to determine the
+ * fingerprint.
+ */
+ protected synchronized void setCaughtException(
+ final PuppeTorException caughtException) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "setCaughtException",
+ caughtException);
+
+ // remember caught exception and notify all waiting threads
+ this.caughtException = caughtException;
+ notifyAll();
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "setCaughtException");
+ }
+
+ /**
+ * Port on which the Tor node will be listening for directory requests from
+ * other Tor nodes.
+ */
+ protected int dirPort;
+
+ /**
+ * The IP v4 address on which the node will listen in dotted decimal
+ * notation.
+ */
+ protected String serverIpAddress;
+
+ /**
+ * The fingerprint of this node that is determined as hash value of its
+ * onion key. It is initialized with <code>null</code> and set by the
+ * fingerprint thread as soon as it is determined.
+ */
+ private String fingerprint;
+
+ /**
+ * The exception that was caught when determining the fingerprint of this
+ * node, if any.
+ */
+ protected PuppeTorException caughtException;
+
+ /**
+ * The pattern for valid IP v4 addresses in dotted decimal notation.
+ */
+ private static final Pattern validIpAddressPattern =
+ Pattern
+ .compile("([0-9]|[1-9][0-9]|1[0-9][0-9]|2[0-4][0-9]|25[0-5])"
+ + "(\\.([0-9]|[1-9][0-9]|1[0-9][0-9]|2[0-4][0-9]|25[0-5])){1,3}");
+
+ /**
+ * Port on which the Tor node will be listening for onion requests by other
+ * Tor nodes.
+ */
+ protected int orPort;
+
+ /**
+ * Creates a new <code>RouterNodeImpl</code> and adds it to the given
+ * <code>network</code>, but does not yet write its configuration to disk
+ * or start the corresponding Tor process.
+ *
+ * @param network
+ * Network configuration to which this node belongs.
+ * @param nodeName
+ * The name of the new node which may only consist of between 1
+ * and 19 alpha-numeric characters.
+ * @param controlPort
+ * Port on which the Tor node will be listening for us as its
+ * controller. May not be negative or greater than 65535.
+ * @param socksPort
+ * Port on which the Tor node will be listening for SOCKS
+ * connection requests. May not be negative or greater than
+ * 65535.
+ * @param orPort
+ * Port on which the Tor node will be listening for onion
+ * requests by other Tor nodes. May not be negative or greater
+ * than 65535.
+ * @param dirPort
+ * Port on which the Tor node will be listening for directory
+ * requests from other Tor nodes. May not be negative or greater
+ * than 65535.
+ * @param serverIpAddress
+ * The IP address on which the node will listen. Must be a valid
+ * IP v4 address in dotted decimal notation. May not be
+ * <code>null</code>.
+ * @throws IllegalArgumentException
+ * If at least one of the parameters is <code>null</code> or
+ * has an invalid value.
+ */
+ RouterNodeImpl(final NetworkImpl network, final String nodeName,
+ final int controlPort, final int socksPort, final int orPort,
+ final int dirPort, final String serverIpAddress) {
+
+ // create superclass instance; parameter checking is done in super
+ // constructor
+ super(network, nodeName, controlPort, socksPort);
+
+ // log entering
+ logger.entering(this.getClass().getName(), "RouterNodeImpl",
+ new Object[] { network, nodeName, controlPort, socksPort,
+ orPort, dirPort, serverIpAddress });
+
+ // check parameters
+ if (orPort < 0 || orPort > 65535 || dirPort < 0 || dirPort > 65535
+ || serverIpAddress == null
+ || !validIpAddressPattern.matcher(serverIpAddress).matches()) {
+ final IllegalArgumentException e =
+ new IllegalArgumentException("nodeName=" + nodeName
+ + ", controlPort=" + controlPort + ", socksPort="
+ + socksPort + ", orPort=" + orPort + ", dirPort="
+ + dirPort + ", serverIpAddress='" + serverIpAddress
+ + "'");
+ logger.throwing(this.getClass().getName(), "RouterNodeImpl", e);
+ throw e;
+ }
+
+ // remember parameters
+ this.orPort = orPort;
+ this.dirPort = dirPort;
+ this.serverIpAddress = serverIpAddress;
+
+ // extend configuration by template configuration of router nodes
+ configuration.addAll(templateConfiguration);
+
+ // add further configuration to make this node a router node
+ configuration.add("ORPort " + orPort);
+ configuration.add("Nickname " + nodeName);
+
+ // all routers mirror the directory
+ configuration.add("DirPort " + dirPort);
+
+ // the address of this node should be manually specified and not guessed
+ // by Tor
+ configuration.add("Address " + serverIpAddress);
+
+ // the OR port may only be contacted locally
+ configuration.add("ORListenAddress " + serverIpAddress);
+
+ // offer directory only locally (either by being an authority, or by
+ // mirroring it)
+ configuration.add("DirListenAddress " + serverIpAddress);
+
+ // start a thread to determine the node's fingerprint in the background
+ determineFingerprint();
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "RouterNodeImpl");
+ }
+
+ public synchronized String getFingerprint() throws PuppeTorException {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "getFingerprint");
+
+ // wait until either the fingerprint has been determined or an exception
+ // was caught
+ while (fingerprint == null && caughtException == null) {
+ try {
+ wait();
+ } catch (final InterruptedException e) {
+ // do nothing
+ }
+ }
+
+ if (caughtException != null) {
+ logger.throwing(this.getClass().getName(), "getFingerprint",
+ caughtException);
+ throw caughtException;
+ }
+
+ // log exiting
+ logger
+ .exiting(this.getClass().getName(), "getFingerprint",
+ fingerprint);
+ return fingerprint;
+ }
+
+ @Override
+ public String toString() {
+ return super.toString() + ", orPort=" + orPort + ", dirPort=" + dirPort;
+ }
+
+ public int getDirPort() {
+ return dirPort;
+ }
+
+ public int getOrPort() {
+ return orPort;
+ }
+
+ /**
+ * Determines the fingerprint of this node by starting a background thread
+ * that performs this operation.
+ */
+ protected synchronized void determineFingerprint() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "determineFingerprint");
+
+ // start a thread to determine this node's fingerprint
+ final FingerprintThread fingerprintThread = new FingerprintThread();
+ fingerprintThread.setName(nodeName + " Fingerprint Resolver");
+ fingerprintThread.start();
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "determineFingerprint");
+ }
+
+ /**
+ * Template configuration of router nodes.
+ */
+ static List<String> templateConfiguration;
+
+ static {
+ templateConfiguration = new ArrayList<String>();
+
+ templateConfiguration.add("ContactInfo wont@xxxxxxxxx");
+ templateConfiguration.add("HidServDirectoryV2 1");
+
+ // TODO This is now contained in proposal 135.
+ // templateConfiguration.add("ExitPolicyRejectPrivate 0");
+ // templateConfiguration.add("AssumeReachable 1");
+ // templateConfiguration.add("ServerDNSAllowBrokenResolvConf 1");
+ }
+}
diff --git a/src/org/torproject/puppetor/impl/ServerApplicationImpl.java b/src/org/torproject/puppetor/impl/ServerApplicationImpl.java
new file mode 100644
index 0000000..95e1536
--- /dev/null
+++ b/src/org/torproject/puppetor/impl/ServerApplicationImpl.java
@@ -0,0 +1,352 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.impl;
+
+import java.io.BufferedReader;
+import java.io.IOException;
+import java.io.InputStreamReader;
+import java.io.PrintStream;
+import java.net.ServerSocket;
+import java.net.Socket;
+import java.util.logging.Level;
+import java.util.logging.Logger;
+
+import org.torproject.puppetor.ServerApplication;
+import org.torproject.puppetor.ServerEventType;
+
+
+/**
+ * Implementation of <code>ServerApplication</code>.
+ *
+ * @author kloesing
+ */
+public class ServerApplicationImpl implements ServerApplication {
+
+ /**
+ * Internal thread class that is used to process an incoming request.
+ */
+ private class HandlerThread extends Thread {
+
+ /**
+ * Accepted socket on which the request came in.
+ */
+ private Socket handleSocket = null;
+
+ /**
+ * Creates a new thread to handle the request coming in on
+ * <code>handleSocket</code>, but does not start handling it.
+ *
+ * @param handleSocket
+ * Accepted socket on which the request came in.
+ */
+ public HandlerThread(final Socket handleSocket) {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "HandlerThread",
+ handleSocket);
+
+ // remember parameter
+ this.handleSocket = handleSocket;
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "HandlerThread");
+ }
+
+ @Override
+ public void run() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "run");
+
+ try {
+
+ // wait for request (don't mind the content)
+ final BufferedReader in =
+ new BufferedReader(new InputStreamReader(handleSocket
+ .getInputStream()));
+ in.read();
+
+ // send event to event manager
+ eventManager.observeInternalEvent(System.currentTimeMillis(),
+ getServerApplicationName(),
+ ServerEventType.SERVER_RECEIVING_REQUEST_SENDING_REPLY,
+ "Receiving request.");
+
+ // write response
+ final PrintStream out =
+ new PrintStream(handleSocket.getOutputStream());
+ out.print("HTTP/1.0 200 OK\r\n");
+
+ } catch (final IOException e) {
+ logger.log(Level.SEVERE,
+ "I/O exception while handling incoming request!");
+ // we can't do more, because nobody takes notice of this thread.
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "run");
+ return;
+ // TODO do we need more?
+ }
+
+ // close socket
+ try {
+ handleSocket.close();
+ } catch (final IOException e) {
+ logger
+ .log(Level.WARNING,
+ "I/O exception while closing socket!");
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "run");
+ return;
+ }
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "run");
+ }
+ }
+
+ /**
+ * Internal thread class that is used to listen for requests.
+ */
+ private class ListenThread extends Thread {
+
+ /**
+ * Flag to remember whether this thread listens for requests at the
+ * moment (<code>true</code>), or has been stopped (<code>false</code>).
+ */
+ private boolean connected;
+
+ /**
+ * Creates a new thread to listen for requests, but does not start
+ * listening, yet.
+ */
+ ListenThread() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "ListenThread");
+
+ // start connected
+ connected = true;
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "ListenThread");
+ }
+
+ @Override
+ public void run() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "run");
+
+ try {
+
+ // create server socket
+ ServerSocket serverSocket = null;
+ try {
+ serverSocket = new ServerSocket(serverPort);
+ } catch (final IOException ioe) {
+ logger.log(Level.SEVERE,
+ "Can't open server socket on port " + serverPort
+ + "!");
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "run");
+ return;
+ }
+
+ // as long as we are connected, accept incoming requests
+ logger.log(Level.FINE, "Listening on port " + serverPort
+ + "...");
+ while (connected) {
+ Socket incomingConnection = null;
+ try {
+ incomingConnection = serverSocket.accept();
+ } catch (final Exception e) {
+ logger
+ .log(
+ Level.SEVERE,
+ "Exception while accepting socket requests! Stopping listening!",
+ e);
+ break;
+ }
+ new HandlerThread(incomingConnection).start();
+ }
+
+ } catch (final Exception e) {
+
+ // log that we have been interrupted
+ logger.log(Level.WARNING, "Server has been interrupted!", e);
+ }
+
+ // mark as disconnected
+ connected = false;
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "run");
+ }
+
+ /**
+ * Stops listening on server socket.
+ */
+ public void stopListening() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "stopListening");
+
+ // change connected state to false and interrupt thread
+ connected = false;
+ interrupt();
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "stopListening");
+ }
+ }
+
+ /**
+ * Event manager that handles all events concerning this server application.
+ */
+ private final EventManagerImpl eventManager;
+
+ /**
+ * Logger for this server which is called "server." plus the name of this
+ * server application.
+ */
+ private final Logger logger;
+
+ /**
+ * Name of this server application that is used as logger name of this node.
+ */
+ private final String serverApplicationName;
+
+ /**
+ * Port on which this server will listen for incoming requests.
+ */
+ private final int serverPort;
+
+ /**
+ * Thread that listens for requests in the background.
+ */
+ private Thread serverThread;
+
+ /**
+ * Creates a new HTTP server application within this JVM, but does not yet
+ * listen for incoming requests.
+ *
+ * @param network
+ * Network to which this HTTP server belongs; at the moment this
+ * is only used to determine the event manager instance.
+ * @param serverApplicationName
+ * Name of this server that is used as part of the logger name.
+ * @param serverPort
+ * Port on which this server will listen for incoming requests.
+ * @throws IllegalArgumentException
+ * If at least one of the parameters is <code>null</code> or
+ * has an invalid value.
+ */
+ ServerApplicationImpl(final NetworkImpl network,
+ final String serverApplicationName, final int serverPort) {
+
+ // check if serverApplicationName can be used as logger name
+ if (serverApplicationName == null
+ || serverApplicationName.length() == 0) {
+ throw new IllegalArgumentException(
+ "Invalid serverApplicationName: " + serverApplicationName);
+ }
+
+ // create logger
+ logger = Logger.getLogger("server." + serverApplicationName);
+
+ // log entering
+ logger.entering(this.getClass().getName(), "ServerApplicationImpl",
+ new Object[] { network, serverApplicationName, serverPort });
+
+ // check parameters
+ if (network == null || serverPort < 0 || serverPort > 65535) {
+ final IllegalArgumentException e = new IllegalArgumentException();
+ logger.throwing(this.getClass().getName(), "ServerApplicationImpl",
+ e);
+ throw e;
+ }
+
+ // remember parameters
+ this.serverApplicationName = serverApplicationName;
+ this.serverPort = serverPort;
+
+ // obtain reference on event manager
+ eventManager = network.getEventManagerImpl();
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "ServerApplicationImpl");
+ }
+
+ public synchronized void startListening() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "listen");
+
+ // check if we are already listening
+ if (serverThread != null) {
+ final IllegalStateException e =
+ new IllegalStateException("We are already listening!");
+ logger.throwing(this.getClass().getName(), "listen", e);
+ throw e;
+ }
+
+ // create a thread that listens in the background
+ serverThread = new ListenThread();
+ serverThread.setName("Reply Thread");
+ serverThread.setDaemon(true);
+ serverThread.start();
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "listen");
+ }
+
+ public synchronized void stopListening() {
+
+ // log entering
+ logger.entering(this.getClass().getName(), "stopListening");
+
+ // check if we are listening
+ if (serverThread == null) {
+ final IllegalStateException e =
+ new IllegalStateException("We are not listening!");
+ logger.throwing(this.getClass().getName(), "stopListening", e);
+ throw e;
+ }
+
+ // log this event
+ logger.log(Level.FINE, "Shutting down server");
+
+ // interrupt thread
+ serverThread.interrupt();
+
+ // unset listening thread
+ serverThread = null;
+
+ // log exiting
+ logger.exiting(this.getClass().getName(), "stopListening");
+ }
+
+ public synchronized boolean isListening() {
+ return serverThread != null;
+ }
+
+ @Override
+ public String toString() {
+ return this.getClass().getSimpleName() + ": serverApplicationName=\""
+ + serverApplicationName + "\", serverPort=" + serverPort;
+ }
+
+ public String getServerApplicationName() {
+ return serverApplicationName;
+ }
+
+ public int getServerPort() {
+ return serverPort;
+ }
+}
diff --git a/src/org/torproject/puppetor/rmi/AbstractMasterFactory.java b/src/org/torproject/puppetor/rmi/AbstractMasterFactory.java
new file mode 100644
index 0000000..85508e3
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/AbstractMasterFactory.java
@@ -0,0 +1,79 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi;
+
+import java.rmi.RemoteException;
+
+import org.torproject.puppetor.rmi.tests.TestRegistration;
+
+
+/**
+ * Factory to create all objects needed by the master side of a PuppeTor
+ * network.
+ *
+ * @author Sebastian Hahn
+ */
+public abstract class AbstractMasterFactory {
+
+ /**
+ * Hold the concrete RemotePuppeTorFactory that will be used to create the
+ * implementation classes.
+ */
+ private static AbstractMasterFactory factory;
+
+ /**
+ * @return a new concrete
+ * <code>AbstractRemotePuppeTorFactory<code>subclass as
+ * specified by the initialization
+ */
+ final public synchronized static AbstractMasterFactory getInstance() {
+ return factory;
+ }
+
+ /**
+ * @param factory
+ * save this as the factory if this hasn't been called before.
+ */
+ final public synchronized static void initialize(
+ final AbstractMasterFactory factory) {
+ if (AbstractMasterFactory.factory != null) {
+ throw new RuntimeException("Don't call initialize twice!");
+ }
+ AbstractMasterFactory.factory = factory;
+ }
+
+ /**
+ * Override this to create a subclass of <code>RemotePuppeTor</code>
+ *
+ * @param slave
+ * Create the server represantation for this connected slave
+ * @return The new <code>RemotePuppeTor</code> instance
+ */
+ public abstract Master createMaster(Slave slave) throws RemoteException;
+
+ /**
+ * Override this to create a subclass of <code>TestExecutor</code>.
+ *
+ * @return The <code>TestExecutor</code> instance.
+ */
+ public abstract TestExecutor getTestExecutorInstance();
+
+ /**
+ * Override this to create a subclass of <code>TestRegistration</code>.
+ *
+ * @return The <code>TestRegistration</code> instance.
+ */
+ public abstract TestRegistration getTestRegistrationInstance();
+
+ /**
+ * XXX just for now so I don't forget.-SH
+ */
+ public abstract NetworkDescription createNetworkDescription();
+
+
+
+// public abstract Network createNetwork(String name);
+}
diff --git a/src/org/torproject/puppetor/rmi/AbstractSlaveFactory.java b/src/org/torproject/puppetor/rmi/AbstractSlaveFactory.java
new file mode 100644
index 0000000..ac5b2d7
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/AbstractSlaveFactory.java
@@ -0,0 +1,89 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi;
+
+import java.util.Set;
+
+import org.torproject.puppetor.rmi.Slave.OS;
+
+
+/**
+ * Create a new representation of the server. Initialize this once with a
+ * concrete subclass of itself to create the actual <code>RemotePuppeTor</code>
+ * instance.
+ *
+ * @author Sebastian Hahn
+ */
+public abstract class AbstractSlaveFactory {
+
+ /**
+ * Hold the concrete RemotePuppeTorFactory that will be used to create
+ * <code>Slave</code> instances. Uses
+ * <code>AbstractSlaveFactory.class</code> for locking.
+ */
+ private static AbstractSlaveFactory factory;
+
+ /**
+ * Hold the remote master object. We only have one per slave. XXX Maybe this
+ * entire thing is more like a SlaveManager, not a factory. Oh well. Rename
+ * later, I guess -SH
+ */
+ private static RemoteMaster master;
+
+ /**
+ * @return a new concrete <code>AbstractSlaveFactory<code>subclass as
+ * specified by the initialization
+ */
+ final public synchronized static AbstractSlaveFactory getInstance() {
+ return factory;
+ }
+
+ /**
+ * @param factory
+ * save this as the factory if this hasn't been called before.
+ * XXX - We could throw an exception here, too. I will think
+ * about that.-SH
+ */
+ final public synchronized static void initialize(
+ final AbstractSlaveFactory factory) {
+ if (AbstractSlaveFactory.factory == null) {
+ AbstractSlaveFactory.factory = factory;
+ }
+ }
+
+ /**
+ * Register a new master. Note that calling this twice for different master
+ * doesn't indicate an error, but trying to to set the same master again is
+ * a bug. Setting this to null means the connection to the master was
+ * closed.
+ *
+ * @param master
+ * The master that should be used from now on.
+ */
+ final public synchronized static void setMaster(final RemoteMaster master) {
+ AbstractSlaveFactory.master = master;
+ }
+
+ /**
+ * @return the RemoteMaster currently registered.
+ */
+ final public synchronized static RemoteMaster getMaster() {
+ if (AbstractSlaveFactory.master == null) {
+ throw new IllegalStateException("No master set yet!");
+ }
+ return AbstractSlaveFactory.master;
+ }
+
+ /**
+ * Override this to create a subclass of <code>Slave</code>
+ *
+ * @return The new <code>Slave</code> instance
+ */
+ public abstract Slave createSlave(final String slaveName, final OS os,
+ final String ip, final Set<Integer> ports,
+ final Set<Integer> availableRevisions) throws NullPointerException;
+
+}
diff --git a/src/org/torproject/puppetor/rmi/LocalMaster.java b/src/org/torproject/puppetor/rmi/LocalMaster.java
new file mode 100644
index 0000000..50dbacb
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/LocalMaster.java
@@ -0,0 +1,36 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi;
+
+/**
+ * Represent a connection between the locally running PuppeTor network Master
+ * and a slave. This is the master part of the <code>Master</code> interface.
+ *
+ * @author Sebastian Hahn
+ */
+public interface LocalMaster {
+ /**
+ * @return the slave instance this connection refers to.
+ */
+ public Slave getSlave();
+
+ /**
+ * Add a new AbstractTaskImpl for the connected slave.
+ *
+ * @param abstractTaskImpl
+ */
+ public void addTask(Task task);
+
+ /**
+ * @param taskId
+ * The task id that we want to get
+ * @return Get the TaskResult for the Task with the specified taskId.
+ * @throws InterruptedException
+ * if interrupted while waiting for the next TestResult
+ */
+ public TaskResult getTaskResult(int taskId) throws InterruptedException;
+
+}
diff --git a/src/org/torproject/puppetor/rmi/Master.java b/src/org/torproject/puppetor/rmi/Master.java
new file mode 100644
index 0000000..20673c8
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/Master.java
@@ -0,0 +1,15 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi;
+
+/**
+ * Any subclass implementing this must also implement <code>RemoteMaster</code>
+ * and <code>LocalMaster</code>. XXX This isn't a good way to do it. Think
+ * about it more later-SH & Karsten
+ *
+ * @author Sebastian Hahn
+ */
+public interface Master {}
diff --git a/src/org/torproject/puppetor/rmi/MasterConnector.java b/src/org/torproject/puppetor/rmi/MasterConnector.java
new file mode 100644
index 0000000..44ea315
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/MasterConnector.java
@@ -0,0 +1,31 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi;
+
+import java.rmi.Remote;
+import java.rmi.RemoteException;
+
+/**
+ * Handle the initial handshake with the slaves and pass them back an object
+ * that the slave uses to refer to the master. The master will create exactly
+ * one <code>MasterConnector</code>, slaves don't ever need one.
+ *
+ * @author Sebastian Hahn
+ */
+public interface MasterConnector extends Remote {
+
+ /**
+ * Called once by every connecting client so that the master knows about it
+ * and has a chance to pass back a representation of itself.
+ *
+ * @param slave
+ * A slave representation as passed from the connecting slave
+ * @return The representation of the master
+ * @throws RemoteException
+ */
+ public RemoteMaster registerClient(final Slave slave)
+ throws RemoteException;
+}
diff --git a/src/org/torproject/puppetor/rmi/Network.java b/src/org/torproject/puppetor/rmi/Network.java
new file mode 100644
index 0000000..4f4f6c2
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/Network.java
@@ -0,0 +1,79 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi;
+
+import java.io.Serializable;
+import java.util.List;
+import java.util.Set;
+
+/**
+ * This class describes all Tor instances on the slaves that are used to form a
+ * private Tor network, or the Tor instances that are added to the public
+ * network and are used in a set of tests.
+ *
+ * @author Sebastian Hahn
+ */
+public interface Network extends Serializable {
+
+ /**
+ * @param torInstance
+ * add this <code>TorInstance</code> to the network.
+ */
+ public void addTorInstance(TorInstance torInstance);
+
+ /**
+ * @return this network's name, mainly to identify it in logs.
+ */
+ public String getName();
+
+ /**
+ * @param test
+ * add this <code>test</code> to the network.
+ */
+ public void addTest(Test test);
+
+ /**
+ * @return a link for each slave that is part of this network.
+ */
+ public Set<LocalMaster> getMasters();
+
+ /**
+ * @return all Tor instances that are authorities
+ */
+ public Set<TorInstance> getDirectoryAuthorities();
+
+ /**
+ * @return all Tor instances that are routers, but not authorities
+ */
+ public Set<TorInstance> getRoutersOnly();
+
+ /**
+ * @return all Tor instances that are routers (including authorities)
+ */
+ public Set<TorInstance> getRouters();
+
+ /**
+ * @return all Tor instances that are proxis (but not authorities or
+ * routers)
+ */
+ public Set<TorInstance> getProxiesOnly();
+
+ /**
+ * @return all Tor instances that are proxies (including authorities and
+ * routers)
+ */
+ public Set<TorInstance> getProxies();
+
+ /**
+ * @return all Tests that can run on this network -SH
+ */
+ public List<Test> getTests();
+
+ /**
+ * Is this a private network or part of the public one?
+ */
+ public boolean isPrivateNetwork();
+}
diff --git a/src/org/torproject/puppetor/rmi/NetworkDescription.java b/src/org/torproject/puppetor/rmi/NetworkDescription.java
new file mode 100644
index 0000000..1636566
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/NetworkDescription.java
@@ -0,0 +1,16 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi;
+
+import java.io.Serializable;
+
+public interface NetworkDescription extends Serializable {
+ public int necessaryProxies();
+
+ public int necessaryRouters();
+
+ public int necessaryDirectoryAuthorities();
+}
diff --git a/src/org/torproject/puppetor/rmi/RemoteMaster.java b/src/org/torproject/puppetor/rmi/RemoteMaster.java
new file mode 100644
index 0000000..07a1b87
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/RemoteMaster.java
@@ -0,0 +1,47 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi;
+
+import java.rmi.Remote;
+import java.rmi.RemoteException;
+
+/**
+ * <code>RemotePuppeTor</code> is the main interface that drives the
+ * interaction between the interconnected master and its slave instances. It is
+ * published by the master's instance, and represents the view of the master a
+ * slave has. It provides ways for the slave to register a <code>Slave</code>
+ * to provide information about its capabilities, as well as a way for the slave
+ * to poll the master for new work and then report back results. XXX We will
+ * need a way to allow the master to access this, as well as a way for clients
+ * with open ports to register for callback support instead of polling for new
+ * tasks =SH
+ *
+ * @author Sebastian Hahn
+ */
+public interface RemoteMaster extends Remote {
+
+ /**
+ * Slaves that must poll because they are not reachable can use this method
+ * to ask the master for new work.
+ *
+ * @throws RemoteException
+ * RMI...
+ * @throws InterruptedException
+ * when interrupted while waiting for the next job to arrive.
+ */
+ public Task getNewTask() throws RemoteException, InterruptedException;
+
+ /**
+ * Report back the result of the task to the master. If possible, add an
+ * error message. XXX Maybe down the road an error object is
+ * needed/wanted-SH
+ *
+ * @throws RemoteException
+ * RMI
+ */
+ public void reportTaskResult(TaskResult task) throws RemoteException,
+ InterruptedException;
+}
diff --git a/src/org/torproject/puppetor/rmi/Slave.java b/src/org/torproject/puppetor/rmi/Slave.java
new file mode 100644
index 0000000..2f037a1
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/Slave.java
@@ -0,0 +1,51 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi;
+
+import java.io.Serializable;
+import java.net.InetAddress;
+import java.util.Set;
+
+/**
+ * A <code>Slave</code> represents a slave instance ready for tests. It
+ * carries information about the testing capabilities of a given slave. A slave
+ * may not provide information except for its name, which is always mandatory.
+ * The slave will not be used for non-client traffic, though.
+ *
+ * @author Sebastian Hahn
+ */
+public interface Slave extends Serializable {
+
+ public enum OS {
+ UNDEFINED, LINUX_I86_32, LINUX_I86_64, MAC_OSX_PPC, MAC_OSX_I86,
+ WINDOWS_XP_32, WINDOWS_XP_64, WINDOWS_VISTA_32, WINDOWS_VISTA_64
+ }
+
+ /**
+ * @return the slave's name.
+ */
+ public String getName();
+
+ /**
+ * @return the slave's operating system as defined in <code>Slave.OS</code>.
+ */
+ public OS getOS();
+
+ /**
+ * @return the slave's IP address. XXX Test this with IPv6-SH
+ */
+ public InetAddress getIP();
+
+ /**
+ * @return the ports this slave can open
+ */
+ public Set<Integer> openablePorts();
+
+ /**
+ * @return Tor revisions available on the slave
+ */
+ public Set<Integer> availableRevisions();
+}
diff --git a/src/org/torproject/puppetor/rmi/Task.java b/src/org/torproject/puppetor/rmi/Task.java
new file mode 100644
index 0000000..bb466e8
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/Task.java
@@ -0,0 +1,38 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi;
+
+import java.io.Serializable;
+import java.rmi.RemoteException;
+import java.util.Collection;
+
+public interface Task extends Serializable {
+
+ /**
+ * Execute the task. Must be called by the slave.
+ *
+ * @throws InterruptedException
+ * if interrupted while the task is running.
+ * @throws RemoteException
+ * RMI
+ */
+ public void execute() throws InterruptedException, RemoteException;
+
+ /**
+ * Returns this tasks id. Task ids must be globally unique and larger than 0
+ * or 0 to indicate a task that was sent out by the slave, not the master.
+ */
+ public int getId();
+
+ /**
+ * This method will be called by the <code>TestExecutor</code> to allow
+ * tasks to collect additional information if they need it.
+ * @param additionalInformation
+ * offer this information.
+ */
+ public void addAdditionalInformation(
+ Collection<String> additionalInformation);
+}
diff --git a/src/org/torproject/puppetor/rmi/TaskExecutionNotSuccessfulException.java b/src/org/torproject/puppetor/rmi/TaskExecutionNotSuccessfulException.java
new file mode 100644
index 0000000..a37bb9a
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/TaskExecutionNotSuccessfulException.java
@@ -0,0 +1,23 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi;
+
+import java.util.Set;
+
+public class TaskExecutionNotSuccessfulException extends Exception {
+ private static final long serialVersionUID = 1L;
+ final private Set<TaskResult> taskResults;
+
+ public TaskExecutionNotSuccessfulException(final Set<TaskResult> taskResults) {
+ this.taskResults = taskResults;
+ }
+
+ public Set<TaskResult> getTaskResults() {
+ return taskResults;
+ }
+
+ // XXX Add some logging capabilities -SH
+}
diff --git a/src/org/torproject/puppetor/rmi/TaskResult.java b/src/org/torproject/puppetor/rmi/TaskResult.java
new file mode 100644
index 0000000..7cd9f68
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/TaskResult.java
@@ -0,0 +1,39 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi;
+
+import java.io.Serializable;
+
+/**
+ * After the completion of a <code>Task</code>, a <code>TaskResult</code> is
+ * created to report the result of the task as well as logging information to
+ * the Master.
+ *
+ * @author Sebastian Hahn
+ */
+public interface TaskResult extends Serializable {
+
+ /**
+ * @return
+ * the task that we have a result for
+ */
+ public Task getTask();
+
+ /**
+ * @return
+ * <code>true</code> if the task execution was successful,
+ * false otherwise.
+ */
+ public boolean wasSuccessful();
+
+ /**
+ * A <code>Task</code> may optionally pass back a result that can be used
+ * by the Master to manage further Task execution.
+ * @return
+ * the result of this task or null if no result is set.
+ */
+ public String getTaskResult();
+}
\ No newline at end of file
diff --git a/src/org/torproject/puppetor/rmi/Test.java b/src/org/torproject/puppetor/rmi/Test.java
new file mode 100644
index 0000000..c6a400f
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/Test.java
@@ -0,0 +1,37 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi;
+
+import java.io.Serializable;
+
+/**
+ * <code>Test</code> describes a test and the kind of slave it can run on, as
+ * well as other network settings that are required for the test to work. A test
+ * should be able to tell whether it was successfully run or not when passed a
+ * TestResult.
+ *
+ * @author Sebastian Hahn
+ */
+public interface Test extends Serializable {
+ /**
+ * @return
+ * a description of what this test is useful for.
+ */
+ public String getDescription();
+
+ /**
+ * @return
+ * a description of the network that this test requires to run, so the
+ * </code>TestExecutor</code> can pick or set up one.
+ */
+ public NetworkDescription getNetworkDescription();
+
+ /**
+ * Call this when the network has been set up and you want to run the test
+ * with the current network.
+ */
+ public void doTest();
+}
diff --git a/src/org/torproject/puppetor/rmi/TestExecutor.java b/src/org/torproject/puppetor/rmi/TestExecutor.java
new file mode 100644
index 0000000..f5ed5c6
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/TestExecutor.java
@@ -0,0 +1,114 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi;
+
+import java.util.Collection;
+import java.util.Map;
+import java.util.Set;
+
+/**
+ * A master uses the <code>TestExecutor</code> instance to load all registered
+ * tests and check whether the currently running PuppeTor network can satisfy
+ * the precondition necessary to run the test. If yes, it sets up the testing
+ * environment and passes <code>PuppeTorJob</code>s to the slaves as the test
+ * instructs it to do. There must be only a single instance of
+ * <code>TestExecutor</code> at any time.
+ *
+ * @author Sebastian Hahn
+ */
+public interface TestExecutor {
+
+ /**
+ * Register a test with the executor so it will consider it for execution.
+ * Accept all submitted tests even though some may not be executable at the
+ * moment of submission.
+ */
+ public void registerTest(Test test);
+
+ /**
+ * Start executing tests. Callers should make sure that the PuppeTor network
+ * that is supposed to be used is ready.
+ */
+ public void startTesting();
+
+ /**
+ * Return an immutable Map of the nodes that have been run successfully
+ * along with the results of the tests.
+ */
+ public Map<Test, TestResult> getSuccessfulTests();
+
+ /**
+ * Return an immutable Map of the nodes that have been run but didn't
+ * complete successfully along with the results of the tests.
+ */
+ public Map<Test, TestResult> getFailedTests();
+
+ /**
+ * Return all tests that could not be executed because the PuppeTorNetwork
+ * could not match a test's preconditions.
+ */
+ public Set<Test> getCannotRunTests();
+
+ /**
+ * Register a newly connected PuppeTor slave via its
+ * <code>LocalMaster</code> connection object.
+ */
+ public void registerLocalMaster(LocalMaster master);
+
+ /**
+ * Unregister a PuppeTor slave. Called when a slave disconnects.
+ */
+ public void unregisterLocalMaster(LocalMaster master);
+
+ /**
+ * @return
+ * the currently running network.
+ */
+ public Network getCurrentNetwork();
+
+ /**
+ * Execute the passed <code>Task</code> once per passed
+ * <code>LocalMaster</code>. If a <code>LocalMaster</code> is passed
+ * more than once, the task will be executed multiple times.
+ *
+ * @param masters
+ * execute the task for those masters.
+ * @param taskClass
+ * create a task of this class and execute it.
+ * @param taskResults
+ * place the results of the task in this <code>Set</code> unless
+ * <code>null</code> is passed.
+ * @param additionalInformation TODO
+ * @throws TaskExecutionNotSuccessfulException
+ * Indicates a failure during task execution
+ */
+ public void executeTaskForMasters(final Collection<LocalMaster> masters,
+ final Class<? extends Task> taskClass, final Set<TaskResult> taskResults,
+ Collection<String> additionalInformation)
+ throws TaskExecutionNotSuccessfulException;
+
+ /**
+ * Execute the passed <code>Task</code> once per passed
+ * <code>TorInstance</code>. If a <code>TorInstance</code> is passed
+ * more than once, the task will be executed multiple times.
+ * @param taskClass
+ * create a task of this class and execute it.
+ * @param taskResults
+ * place the results of the task in this <code>Set</code> unless
+ * <code>null</code> is passed.
+ * @param additionalInformation
+ * call the task's addAdditionalInformation method with this
+ * parameter.
+ * @param torInstances
+ * execute the task for those masters.
+ * @throws TaskExecutionNotSuccessfulException
+ * Indicates a failure during task execution
+ */
+ public void executeTaskForTorInstances(Collection<TorInstance> instances,
+ Class<? extends Task> taskClass, Set<TaskResult> taskResults,
+ Collection<String> additionalInformation)
+ throws TaskExecutionNotSuccessfulException;
+}
diff --git a/src/org/torproject/puppetor/rmi/TestResult.java b/src/org/torproject/puppetor/rmi/TestResult.java
new file mode 100644
index 0000000..32788c1
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/TestResult.java
@@ -0,0 +1,15 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi;
+
+import java.io.Serializable;
+
+/**
+ * XXX unused for now. this will hold the result of a /<code>Test</code> -SH
+ */
+public interface TestResult extends Serializable {
+ public Test getJob();
+}
diff --git a/src/org/torproject/puppetor/rmi/TorInstance.java b/src/org/torproject/puppetor/rmi/TorInstance.java
new file mode 100644
index 0000000..fd9295e
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/TorInstance.java
@@ -0,0 +1,57 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi;
+
+import java.util.List;
+
+/**
+ * Describe each tor instance at the client with a <code>TorInstance</code>.
+ *
+ * @author Sebastian Hahn
+ */
+public interface TorInstance {
+ /**
+ * Additional configuration options that a test wants to be set for a tor
+ * instance. <code>TorInstance.shouldReplaceConfiguration()</code>
+ * determines whether this configuration should be added to the default tor
+ * configuration or replace it.
+ *
+ * @return An immutable list of extra configuration settings.
+ */
+ public List<String> getConfiguration();
+
+ /**
+ * @return the name by which this Tor instance will be identified at the
+ * slaves. See org.torproject.puppetor.Network.getNode().
+ */
+ public String getName();
+
+ /**
+ * Whether the caller should use the result of
+ * <code>TorInstance.getExtraConfiguration()</code> to append tor's
+ * default configuration or replace it.
+ *
+ * @return true if configuration should be replaced, false if it should be
+ * appended.
+ */
+ public boolean shouldReplaceConfiguration();
+
+ public boolean isDirectoryAuthority();
+
+ public boolean isBridgeAuthority();
+
+ public boolean isRouter();
+
+ public boolean isExit();
+
+ public boolean isOnPrivateNetwork();
+
+ public int getRevision();
+
+ public boolean isBridge();
+
+ public LocalMaster getMaster();
+}
\ No newline at end of file
diff --git a/src/org/torproject/puppetor/rmi/execute/PuppeTorMasterProgram.java b/src/org/torproject/puppetor/rmi/execute/PuppeTorMasterProgram.java
new file mode 100644
index 0000000..e05593a
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/execute/PuppeTorMasterProgram.java
@@ -0,0 +1,194 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.execute;
+
+import java.rmi.AccessException;
+import java.rmi.NoSuchObjectException;
+import java.rmi.NotBoundException;
+import java.rmi.RemoteException;
+import java.rmi.registry.LocateRegistry;
+import java.rmi.registry.Registry;
+import java.rmi.server.UnicastRemoteObject;
+
+import javax.rmi.ssl.SslRMIClientSocketFactory;
+import javax.rmi.ssl.SslRMIServerSocketFactory;
+
+import org.torproject.puppetor.rmi.AbstractMasterFactory;
+import org.torproject.puppetor.rmi.LocalMaster;
+import org.torproject.puppetor.rmi.Master;
+import org.torproject.puppetor.rmi.MasterConnector;
+import org.torproject.puppetor.rmi.RemoteMaster;
+import org.torproject.puppetor.rmi.Slave;
+import org.torproject.puppetor.rmi.TestExecutor;
+import org.torproject.puppetor.rmi.impl.MasterImplFactory;
+import org.torproject.puppetor.rmi.tests.TestRegistration;
+
+
+/**
+ * The <code>PuppeTorMasterProgram</code> contains the main method for the
+ * master instance of a distributed PuppeTor testing infrastructure. Currently,
+ * configuration of the the master's network settings are implemented as
+ * instance variables of <code>PuppeTorMasterProgram</code>. XXX eventually,
+ * using configuration files would be better -SH
+ *
+ * @author Sebastian Hahn
+ */
+public class PuppeTorMasterProgram {
+
+ /**
+ * The RMI registry.
+ */
+ private Registry registry;
+
+ /**
+ * The port the server is supposed to listen on. This port must not be
+ * firewalled and must be forwarded to the machine if necessary.
+ */
+ final private static int serverport = 2050;
+
+ /**
+ * The address this server should bind on. Use 127.0.0.1 for local testing.
+ * XXX Someone should check if this works with IPv6 -SH
+ */
+ final private static String serveraddress = "127.0.0.1";
+
+ /**
+ * Master application entry point
+ *
+ * @param args
+ * Command-line arguments - ignored for now.
+ * @throws NotBoundException
+ * @throws RemoteException
+ * @throws AccessException
+ */
+ public static void main(final String[] args) throws AccessException,
+ RemoteException, NotBoundException {
+ final PuppeTorMasterProgram server = new PuppeTorMasterProgram();
+
+ final TestExecutor exec =
+ AbstractMasterFactory.getInstance().getTestExecutorInstance();
+ exec.startTesting();
+ server.unregisterRMIRegistry();
+ // Suggest a garbage collection, may increase shutdown.
+ System.gc();
+ }
+
+ /**
+ * Private constructor to set up RMI-related Java-internal variables and
+ * create a <code>RemotePuppeTorImpl</code>.
+ *
+ * @throws NotBoundException
+ * @throws RemoteException
+ * @throws AccessException
+ */
+ private PuppeTorMasterProgram() throws AccessException, RemoteException,
+ NotBoundException {
+ // Set the location of the keystore where your private certificate is.
+ System.setProperty("javax.net.ssl.keyStore", "res/keystore");
+ // Set the password for your keystore.
+ System.setProperty("javax.net.ssl.keyStorePassword", "pleasechange");
+
+ /*
+ * Set the location of the truststore which contains the exported
+ * certificates of your slaves.
+ */
+ System.setProperty("javax.net.ssl.trustStore", "res/truststore");
+ // Use a timeout of 45 seconds to detect disconnected clients.
+ System.setProperty("java.rmi.dgc.leaseValue", "45000");
+
+ // Tell the RMI system the location of the master
+ System.setProperty("java.rmi.server.hostname", serveraddress);
+
+ // Set up the factories we want to use
+ AbstractMasterFactory.initialize(new MasterImplFactory());
+
+ registerTests();
+ final InnerMasterConnector master = InnerMasterConnector.getInstance();
+
+ try {
+ registry =
+ LocateRegistry.createRegistry(serverport,
+ new SslRMIClientSocketFactory(),
+ new SslRMIServerSocketFactory(null, null, true));
+ registry.bind("Master", master);
+ } catch (final Throwable th) {
+ th.printStackTrace();
+ throw new RuntimeException(
+ "Could not create the registry or bind"
+ + "an object in it. Please check your system's configuration. "
+ + th);
+ }
+
+ System.out.println("waiting for slaves.");
+ try {
+ Thread.sleep(1000 * 60 * 2);
+ } catch (final InterruptedException e) {
+ e.printStackTrace();
+ throw new RuntimeException("This is the main thread. We don't "
+ + "want to be interrupted.");
+ }
+ registry.unbind("Master");
+ // Only unbinding is not enough, the object will still be exported.
+ UnicastRemoteObject.unexportObject(master, true);
+ }
+
+ /**
+ * Register all the tests you want to execute here.
+ */
+ private void registerTests() {
+ final AbstractMasterFactory fact = AbstractMasterFactory.getInstance();
+ final TestExecutor exec = fact.getTestExecutorInstance();
+ final TestRegistration registration = fact.getTestRegistrationInstance();
+ registration.registerTests(exec);
+ }
+
+ private void unregisterRMIRegistry() throws NoSuchObjectException {
+ UnicastRemoteObject.unexportObject(registry, true);
+ registry = null;
+ }
+
+ static class InnerMasterConnector extends UnicastRemoteObject implements
+ MasterConnector {
+ /**
+ * Required for serialization. Needs to change for new released
+ * versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ private static InnerMasterConnector instance;
+
+ static {
+ try {
+ instance = new InnerMasterConnector();
+ } catch (final RemoteException e) {
+ e.printStackTrace();
+ throw new ExceptionInInitializerError(e);
+ }
+ }
+
+ /**
+ * private constructor, Singleton
+ *
+ * @throws RemoteException
+ * RMI
+ */
+ private InnerMasterConnector() throws RemoteException {}
+
+ public static InnerMasterConnector getInstance() {
+ return instance;
+ }
+
+ public RemoteMaster registerClient(final Slave slave)
+ throws RemoteException {
+ final AbstractMasterFactory fact =
+ AbstractMasterFactory.getInstance();
+ final Master master = fact.createMaster(slave);
+ AbstractMasterFactory.getInstance().getTestExecutorInstance()
+ .registerLocalMaster((LocalMaster) master);
+ return (RemoteMaster) master;
+ }
+ }
+}
diff --git a/src/org/torproject/puppetor/rmi/execute/PuppeTorSlaveProgram.java b/src/org/torproject/puppetor/rmi/execute/PuppeTorSlaveProgram.java
new file mode 100644
index 0000000..5605ed3
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/execute/PuppeTorSlaveProgram.java
@@ -0,0 +1,216 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.execute;
+
+import java.rmi.NoSuchObjectException;
+import java.rmi.NotBoundException;
+import java.rmi.RemoteException;
+import java.rmi.registry.LocateRegistry;
+import java.rmi.registry.Registry;
+import java.util.HashSet;
+import java.util.Set;
+import java.util.concurrent.BlockingQueue;
+import java.util.concurrent.CopyOnWriteArraySet;
+import java.util.concurrent.LinkedBlockingQueue;
+
+import javax.rmi.ssl.SslRMIClientSocketFactory;
+
+import org.torproject.puppetor.rmi.AbstractSlaveFactory;
+import org.torproject.puppetor.rmi.MasterConnector;
+import org.torproject.puppetor.rmi.Slave;
+import org.torproject.puppetor.rmi.Task;
+import org.torproject.puppetor.rmi.impl.SlaveImplFactory;
+import org.torproject.puppetor.rmi.tasks.TerminateTask;
+
+
+/**
+ * The <code>PuppeTorSlaveProgram</code> contains the main method for the
+ * slave instances of a distributed PuppeTor testing infrastructure. Currently,
+ * configuration of the the master's network settings and some slave-specific
+ * options are implemented as instance variables of
+ * <code>PuppeTorSlaveProgram</code>. XXX eventually, using configuration
+ * files would be better -SH
+ *
+ * @author Sebastian Hahn
+ */
+public class PuppeTorSlaveProgram {
+
+ /**
+ * The port the master server is supposed to listen on. This port must not
+ * be firewalled and must be forwarded to the machine if necessary. Use the
+ * same value as you used for the master's configuration.
+ */
+ final private static int serverport = 2050;
+
+ /**
+ * The address master server listens on. Use the same value as you used for
+ * the master's configuration. XXX Someone should check if this works with
+ * IPv6 -SH
+ */
+ final private static String serveraddress = "127.0.0.1";//78.47.18.109";
+
+ /**
+ * The address this slave listens on. Use 127.0.0.1 for slaves that do not
+ * open any ports for the outside world and thus can act as Tor clients
+ * only.
+ */
+ final private static String myaddress = "127.0.0.1";
+
+ /**
+ * Each slave is identified by its unique <code>slaveName</code>. Be
+ * careful to prevent naming collisions between slaves.
+ */
+ final private static String slaveName = "slave1";
+
+ /**
+ * A representation of this slave node to be sent to the master.
+ */
+ final private Slave slave;
+
+ final protected static BlockingQueue<Task> tasks =
+ new LinkedBlockingQueue<Task>();
+
+ /**
+ * Slave application entry point
+ *
+ * @param args
+ * Command-line arguments
+ * @throws NoSuchObjectException
+ */
+ public static void main(final String[] args) {
+ final PuppeTorSlaveProgram slave = new PuppeTorSlaveProgram();
+ while(true) {
+ while (slave.connectToMaster() == false) {
+ try {
+ Thread.sleep(60000);
+ } catch (final InterruptedException e) {
+ throw new RuntimeException("Bug: Nothing should " +
+ "interrupt us here!");
+ }
+ }
+
+ final Thread taskGetter = new Thread(new Runnable() {
+ public void run() {
+ while (true) {
+ try {
+ try {
+ // XXX I dislike the following lines, but
+ // interruption doesn't work while we wait for the
+ // master. Maybe there is a better solution that
+ // doesn't eat the poison pill twice. -SH
+ Task task = AbstractSlaveFactory.getMaster()
+ .getNewTask();
+ tasks.put(task);
+ if (task instanceof TerminateTask) {
+ break;
+ }
+ } catch (final RemoteException e) {
+ // Empty the task queue - we cannot send any results
+ // back, anyways.
+ tasks.clear();
+ // A taskid of 0 indicates a local Task
+ tasks.put(new TerminateTask(0,
+ "local termination"));
+ Thread.currentThread().interrupt();
+ }
+ } catch (final InterruptedException e) {
+ Thread.currentThread().interrupt();
+ }
+ if (Thread.interrupted()) {
+ break;
+ }
+
+ }
+ }
+ });
+
+ taskGetter.start();
+ // XXX We use just a single Thread here. That isn't the smartest move.
+ // -SH
+ try {
+ while (true) {
+ final Task newTask = tasks.take();
+ newTask.execute();
+ if (Thread.interrupted()) {
+ break;
+ }
+ }
+ } catch (final InterruptedException ignored) {} catch (final RemoteException ignored) {}
+ taskGetter.interrupt();
+ AbstractSlaveFactory.setMaster(null);
+ // We don't shut down, but want the Master to see we disconnected,
+ // so hopefully a gc will happen soon.
+ System.gc();
+ }
+
+ }
+
+ /**
+ * Private constructor to set up RMI-related Java-internal variables.
+ */
+ private PuppeTorSlaveProgram() {
+ /*
+ * Set the location of the keystore where your private certificate is.
+ */
+ System.setProperty("javax.net.ssl.keyStore", "res/keystore");
+ /*
+ * Set the password for your keystore.
+ */
+ System.setProperty("javax.net.ssl.keyStorePassword", "pleasechange");
+ /*
+ * Set the location of the truststore which contains the exported
+ * certificates of your slaves
+ */
+ System.setProperty("javax.net.ssl.trustStore", "res/truststore");
+ // Set up the factories we want to use
+ AbstractSlaveFactory.initialize(new SlaveImplFactory());
+ final AbstractSlaveFactory fac = AbstractSlaveFactory.getInstance();
+
+ /*
+ * Create a new slave with the properties that are relevant for the
+ * computer and networking environment it runs on.
+ * XXX unused for now -SH
+ */
+ final Set<Integer> ports = new HashSet<Integer>();
+ /*for (int i = 10025; i < 20000; i++) {
+ ports.add(i);
+ }*/
+ final Set<Integer> availableRevisions =
+ new CopyOnWriteArraySet<Integer>();
+ slave = fac.createSlave(slaveName, Slave.OS.UNDEFINED, myaddress,
+ ports, availableRevisions);
+
+ }
+
+ /**
+ * Attempt to make a connection to the master through RMI. If the master is
+ * down, that is not a fatal condition, we want to keep trying until we can
+ * reach it. Don't call this except from main().
+ *
+ * @return true if the connection was successful, false otherwise.
+ */
+ private boolean connectToMaster() {
+ try {
+ final Registry registry =
+ LocateRegistry.getRegistry(serveraddress, serverport,
+ new SslRMIClientSocketFactory());
+ AbstractSlaveFactory.setMaster(((MasterConnector) registry
+ .lookup("Master")).registerClient(slave));
+ } catch (final NotBoundException e) {
+ e.printStackTrace();
+ System.out.println("We could connect, but the master has not "
+ + "exported the Master Object yet." + e);
+ return false;
+ } catch (final RemoteException e) {
+ return false;
+ } catch (final IllegalArgumentException e) {
+ throw new RuntimeException(
+ "A slave with this name is already connected "
+ + "to the server! " + e);
+ }
+ return true;
+ }
+}
diff --git a/src/org/torproject/puppetor/rmi/impl/AbstractTaskImpl.java b/src/org/torproject/puppetor/rmi/impl/AbstractTaskImpl.java
new file mode 100644
index 0000000..e63eb14
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/impl/AbstractTaskImpl.java
@@ -0,0 +1,62 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.impl;
+
+import java.rmi.RemoteException;
+import java.util.Collection;
+
+import org.torproject.puppetor.rmi.AbstractSlaveFactory;
+import org.torproject.puppetor.rmi.Task;
+import org.torproject.puppetor.rmi.TaskResult;
+
+
+public abstract class AbstractTaskImpl implements Task {
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ /**
+ * A human-readable name for the task (to be included in log-messages, etc).
+ */
+ final protected String name;
+
+ /**
+ * The globally unique id for this task.
+ */
+ final protected int id;
+
+ public AbstractTaskImpl(final int id, final String name) {
+ this.name = name;
+ this.id = id;
+ }
+
+ public int getId() {
+ return id;
+ }
+
+ public abstract void execute() throws InterruptedException, RemoteException;
+
+ protected void reportResult(final boolean success)
+ throws InterruptedException, RemoteException {
+ reportResult(new TaskResultImpl(success, this, null));
+ }
+
+ protected void reportResult(final boolean success, final String result)
+ throws InterruptedException, RemoteException {
+ reportResult(new TaskResultImpl(success, this, result));
+ }
+
+ // XXX protect against double-reporting of results -SH
+ protected void reportResult( final TaskResult taskResult)
+ throws InterruptedException, RemoteException {
+ AbstractSlaveFactory.getMaster().reportTaskResult(taskResult);
+ }
+
+ public void addAdditionalInformation(
+ Collection<String> additionalInformation) {}
+
+}
diff --git a/src/org/torproject/puppetor/rmi/impl/AbstractTestImpl.java b/src/org/torproject/puppetor/rmi/impl/AbstractTestImpl.java
new file mode 100644
index 0000000..018fdc0
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/impl/AbstractTestImpl.java
@@ -0,0 +1,52 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.impl;
+
+import org.torproject.puppetor.rmi.AbstractMasterFactory;
+import org.torproject.puppetor.rmi.NetworkDescription;
+import org.torproject.puppetor.rmi.Test;
+import org.torproject.puppetor.rmi.TestExecutor;
+
+/**
+ * Extend <code>AbstractTestImpl</code> for your own tests. Make sure you also add them
+ * to the TestRegistration.registerTest method.
+ *
+ * @author Sebastian Hahn
+ */
+abstract public class AbstractTestImpl implements Test {
+
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ /**
+ * A test description for log messages.
+ */
+ final private String description;
+
+ /**
+ * The network that must exist for the test to be executed.
+ */
+ final private NetworkDescription networkDescription;
+
+ final protected TestExecutor exec;
+
+ public AbstractTestImpl(final String description, final NetworkDescription network) {
+ this.description = description;
+ networkDescription = network;
+ exec = AbstractMasterFactory.getInstance().
+ getTestExecutorInstance();
+ }
+
+ public final NetworkDescription getNetworkDescription() {
+ return networkDescription;
+ }
+
+ public String getDescription() {
+ return description;
+ }
+}
diff --git a/src/org/torproject/puppetor/rmi/impl/MasterImpl.java b/src/org/torproject/puppetor/rmi/impl/MasterImpl.java
new file mode 100644
index 0000000..5550486
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/impl/MasterImpl.java
@@ -0,0 +1,105 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.impl;
+
+import java.rmi.RemoteException;
+import java.rmi.server.UnicastRemoteObject;
+import java.rmi.server.Unreferenced;
+import java.util.concurrent.BlockingQueue;
+import java.util.concurrent.LinkedBlockingQueue;
+
+import org.torproject.puppetor.rmi.AbstractMasterFactory;
+import org.torproject.puppetor.rmi.LocalMaster;
+import org.torproject.puppetor.rmi.Master;
+import org.torproject.puppetor.rmi.RemoteMaster;
+import org.torproject.puppetor.rmi.Slave;
+import org.torproject.puppetor.rmi.Task;
+import org.torproject.puppetor.rmi.TaskResult;
+
+
+/**
+ * Represent the master at the slave, and give the master a way to interact with
+ * the respective slave.
+ *
+ * @author Sebastian Hahn
+ */
+public class MasterImpl extends UnicastRemoteObject implements LocalMaster,
+ RemoteMaster, Master, Unreferenced {
+
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ /**
+ * Store the reference to the connected slave
+ */
+ private final Slave slave;
+
+ private final BlockingQueue<Task> tasks;
+
+ private final BlockingQueue<TaskResult> completedTasks;
+
+ private final Object completedTasksLock = new Object();
+
+ public Slave getSlave() {
+ return slave;
+ }
+
+ /**
+ * Check the slave object for validity and save it for further reference
+ *
+ * @throws RemoteException
+ * RMI
+ */
+ public MasterImpl(final Slave slave) throws RemoteException {
+ if (slave.getName() == null) {
+ throw new NullPointerException("The slave's name must not be null!");
+ }
+ this.slave = slave;
+ tasks = new LinkedBlockingQueue<Task>();
+ completedTasks = new LinkedBlockingQueue<TaskResult>();
+ }
+
+ public Task getNewTask() throws RemoteException, InterruptedException {
+ return tasks.take();
+ }
+
+ public void reportTaskResult(final TaskResult task) throws RemoteException,
+ InterruptedException {
+ synchronized (completedTasksLock) {
+ completedTasks.put(task);
+ completedTasksLock.notifyAll();
+ }
+ }
+
+ public TaskResult getTaskResult(final int taskId)
+ throws InterruptedException {
+
+ synchronized (completedTasksLock) {
+ TaskResult res;
+ while ((res = completedTasks.peek()) == null
+ || res.getTask().getId() != taskId) {
+ completedTasksLock.wait();
+ }
+ return completedTasks.take();
+ }
+ }
+
+ /**
+ * Remove the slave from the Map of slaves and destroy its workqueues.
+ * Inform the TestExecutor that all still-running tests failed because the
+ * slave went away.
+ */
+ public void unreferenced() {
+ AbstractMasterFactory.getInstance().getTestExecutorInstance()
+ .unregisterLocalMaster(this);
+ }
+
+ public void addTask(final Task task) {
+ tasks.add(task);
+ }
+}
diff --git a/src/org/torproject/puppetor/rmi/impl/MasterImplFactory.java b/src/org/torproject/puppetor/rmi/impl/MasterImplFactory.java
new file mode 100644
index 0000000..f260982
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/impl/MasterImplFactory.java
@@ -0,0 +1,56 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.impl;
+
+import java.rmi.RemoteException;
+
+import org.torproject.puppetor.rmi.AbstractMasterFactory;
+import org.torproject.puppetor.rmi.Master;
+import org.torproject.puppetor.rmi.NetworkDescription;
+import org.torproject.puppetor.rmi.Slave;
+import org.torproject.puppetor.rmi.TestExecutor;
+import org.torproject.puppetor.rmi.tests.TestRegistration;
+
+/**
+ * Create <code>RemotePuppeTorImpl</code> instances
+ *
+ * @author Sebastian Hahn
+ */
+public class MasterImplFactory extends AbstractMasterFactory {
+
+ @Override
+ public Master createMaster(final Slave slave) throws RemoteException {
+ return new MasterImpl(slave);
+ }
+
+ @Override
+ public TestExecutor getTestExecutorInstance() {
+ return TestExecutorImpl.getInstance();
+ }
+
+ @Override
+ public TestRegistration getTestRegistrationInstance() {
+ return TestRegistration.getInstance();
+ }
+
+/* @Override
+ public Test createPublicNetworkTestForClientsOnly(final String description) {
+ return null;
+ return new AbstractTestImpl(description, new NetworkDescriptionImpl(1, 0, 0));
+ }*/
+
+ /*@Override
+ public Network createNetwork(final String name) {
+ return new NetworkImpl(name);
+ }*/
+
+ @Override
+ public NetworkDescription createNetworkDescription() {
+ // XXX implement -SH
+ return null;
+ }
+
+}
diff --git a/src/org/torproject/puppetor/rmi/impl/NetworkDescriptionImpl.java b/src/org/torproject/puppetor/rmi/impl/NetworkDescriptionImpl.java
new file mode 100644
index 0000000..e13614e
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/impl/NetworkDescriptionImpl.java
@@ -0,0 +1,44 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.impl;
+
+import java.util.concurrent.atomic.AtomicInteger;
+
+import org.torproject.puppetor.rmi.NetworkDescription;
+
+
+public class NetworkDescriptionImpl implements NetworkDescription {
+
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ private final AtomicInteger numProxies;
+ private final AtomicInteger numRouters;
+ private final AtomicInteger numDirectoryAuthorities;
+
+ public int necessaryProxies() {
+ return numProxies.get();
+ }
+
+ public int necessaryRouters() {
+ return numRouters.get();
+ }
+
+ public int necessaryDirectoryAuthorities() {
+ return numDirectoryAuthorities.get();
+ }
+
+ public NetworkDescriptionImpl(final int numProxies, final int numRouters,
+ final int numDirectoryAuthorities) {
+ this.numProxies = new AtomicInteger(numProxies);
+ this.numRouters = new AtomicInteger(numRouters);
+ this.numDirectoryAuthorities =
+ new AtomicInteger(numDirectoryAuthorities);
+ }
+
+}
diff --git a/src/org/torproject/puppetor/rmi/impl/NetworkImpl.java b/src/org/torproject/puppetor/rmi/impl/NetworkImpl.java
new file mode 100644
index 0000000..aa8fd01
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/impl/NetworkImpl.java
@@ -0,0 +1,126 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.impl;
+
+import java.util.Collections;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Set;
+import java.util.concurrent.CopyOnWriteArrayList;
+import java.util.concurrent.CopyOnWriteArraySet;
+
+import org.torproject.puppetor.rmi.LocalMaster;
+import org.torproject.puppetor.rmi.Network;
+import org.torproject.puppetor.rmi.Test;
+import org.torproject.puppetor.rmi.TorInstance;
+
+
+public class NetworkImpl implements Network {
+
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ /**
+ * The name of the network, so we can identify it in logs etc.
+ */
+ final private String name;
+
+ /**
+ * All the tests that will run with this network. The individual Tor
+ * instances will have the tests they need to execute broken down further.
+ */
+ private final List<Test> tests;
+
+ /**
+ * All the Tor instances running at the slaves that are linked to this
+ * network.
+ */
+ private final Set<TorInstance> torInstances;
+
+ private final boolean privateNetwork;
+
+ public NetworkImpl(final String name, boolean privateNetwork) {
+ this.name = name;
+ tests = new CopyOnWriteArrayList<Test>();
+ torInstances = new CopyOnWriteArraySet<TorInstance>();
+ this.privateNetwork = privateNetwork;
+ }
+
+ public void addTorInstance(final TorInstance torInstance) {
+ torInstances.add(torInstance);
+ }
+
+ public String getName() {
+ return name;
+ }
+
+ public void addTest(final Test test) {
+ tests.add(test);
+ }
+
+ public Set<LocalMaster> getMasters() {
+ final Set<LocalMaster> masters = new HashSet<LocalMaster>();
+ for (final TorInstance torInstance : torInstances) {
+ masters.add(torInstance.getMaster());
+ }
+ return Collections.unmodifiableSet(masters);
+ }
+
+ public Set<TorInstance> getDirectoryAuthorities() {
+ final Set<TorInstance> authorities = new HashSet<TorInstance>();
+ for (final TorInstance torInstance : torInstances) {
+ if (torInstance.isDirectoryAuthority()) {
+ authorities.add(torInstance);
+ }
+ }
+ return Collections.unmodifiableSet(authorities);
+ }
+
+ public Set<TorInstance> getProxies() {
+ return Collections.unmodifiableSet(torInstances);
+ }
+
+ public Set<TorInstance> getProxiesOnly() {
+ final Set<TorInstance> proxies = new HashSet<TorInstance>();
+ for (final TorInstance torInstance : torInstances) {
+ if (!torInstance.isRouter()) {
+ proxies.add(torInstance);
+ }
+ }
+ return Collections.unmodifiableSet(proxies);
+ }
+
+ public Set<TorInstance> getRouters() {
+ final Set<TorInstance> routers = new HashSet<TorInstance>();
+ for (final TorInstance torInstance : torInstances) {
+ if (!torInstance.isRouter()) {
+ routers.add(torInstance);
+ }
+ }
+ return Collections.unmodifiableSet(routers);
+ }
+
+ public Set<TorInstance> getRoutersOnly() {
+ final Set<TorInstance> routers = new HashSet<TorInstance>();
+ for (final TorInstance torInstance : torInstances) {
+ if (torInstance.isRouter() && !torInstance.isDirectoryAuthority()) {
+ routers.add(torInstance);
+ }
+ }
+ return Collections.unmodifiableSet(routers);
+ }
+
+ public List<Test> getTests() {
+ return Collections.unmodifiableList(tests);
+ }
+
+ public boolean isPrivateNetwork() {
+ return privateNetwork;
+ }
+
+}
diff --git a/src/org/torproject/puppetor/rmi/impl/SlaveImpl.java b/src/org/torproject/puppetor/rmi/impl/SlaveImpl.java
new file mode 100644
index 0000000..ca532b7
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/impl/SlaveImpl.java
@@ -0,0 +1,132 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.impl;
+
+import java.net.InetAddress;
+import java.net.UnknownHostException;
+import java.util.Collections;
+import java.util.Set;
+import java.util.concurrent.CopyOnWriteArraySet;
+
+import org.torproject.puppetor.rmi.Slave;
+
+
+/**
+ * Describe a client including its capabilities and network location to the
+ * master. Also allow the master to query that information. A slave is not
+ * required to give more than its name, but it can only be used for very limited
+ * tests.
+ *
+ * @author Sebastian Hahn
+ */
+public final class SlaveImpl implements Slave {
+
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ /**
+ * The slave's name
+ */
+ protected final String name;
+
+ /**
+ * The slave's ip address
+ */
+ protected final InetAddress ip;
+
+ /**
+ * The operating system of the slave. See <code>Slave.OS</code>.
+ */
+ protected final OS os;
+
+ /**
+ * The ports this slave can open. Currently, we choose to use 256 as a
+ * maximum that "should be" enough so we don't transfer so much data when
+ * serializing. See the constructor. Make sure this is immutable.
+ */
+ protected final Set<Integer> ports;
+
+ /**
+ * The different revisions of Tor clients this slave can execute.
+ */
+ protected final Set<Integer> availableRevisions;
+
+ /**
+ * Create a new <code>Slave</code> with the metrics we currently use,
+ * meaning it's operating system, it's IP, and the ports it can open. We may
+ * in the future include other things like Tor executable version numbers
+ * and the ability to run certain extra scripts.
+ *
+ * @param slaveName
+ * the slave's name. Each slave must have it's own unique name in
+ * a given PuppeTor network.
+ * @param os
+ * the slave's operating system.
+ * @param ip
+ * A String representation of the slave's IP (v4 or v6).
+ * @param ports
+ * All the ports this slave can open. Currently, we pick 256 out
+ * of those at random.
+ * @throws NullPointerException
+ * if the slavename is null
+ */
+ public SlaveImpl(final String slaveName, final OS os, final String ip,
+ final Set<Integer> ports, final Set<Integer> availableRevisions)
+ throws NullPointerException {
+ if (slaveName == null) {
+ throw new NullPointerException("slaveName must not be null");
+ }
+ name = slaveName;
+ this.os = os;
+ try {
+ this.ip = InetAddress.getByName(ip);
+ } catch (final UnknownHostException e) {
+ e.printStackTrace();
+ throw new RuntimeException("You specified an invalid IP address!");
+ }
+
+ if (ports.size() > 256) {
+ final Set<Integer> tmp = new CopyOnWriteArraySet<Integer>();
+ int i = 0;
+ for (final Integer j : tmp) {
+ if (i++ == 256) {
+ break;
+ }
+ tmp.add(j);
+ }
+ this.ports = Collections.unmodifiableSet(tmp);
+ } else {
+ this.ports =
+ Collections
+ .unmodifiableSet(new CopyOnWriteArraySet<Integer>(
+ ports));
+ }
+ this.availableRevisions =
+ Collections.unmodifiableSet(availableRevisions);
+ }
+
+ public String getName() {
+ return name;
+ }
+
+ public OS getOS() {
+ return os;
+ }
+
+ public InetAddress getIP() {
+ return ip;
+ }
+
+ public Set<Integer> openablePorts() {
+ return ports;
+ }
+
+ public Set<Integer> availableRevisions() {
+ return availableRevisions;
+ }
+}
diff --git a/src/org/torproject/puppetor/rmi/impl/SlaveImplFactory.java b/src/org/torproject/puppetor/rmi/impl/SlaveImplFactory.java
new file mode 100644
index 0000000..0448d69
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/impl/SlaveImplFactory.java
@@ -0,0 +1,32 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.impl;
+
+import java.util.Set;
+
+import org.torproject.puppetor.rmi.AbstractSlaveFactory;
+import org.torproject.puppetor.rmi.Slave;
+import org.torproject.puppetor.rmi.Slave.OS;
+
+
+/**
+ * Create <code>RemotePuppeTorImpl</code> instances
+ *
+ * @author Sebastian Hahn
+ */
+public class SlaveImplFactory extends AbstractSlaveFactory {
+
+ /**
+ * @return a new <code>RemotePuppeTorImpl</code> instance.
+ */
+ @Override
+ public Slave createSlave(final String slaveName, final OS os,
+ final String ip, final Set<Integer> ports,
+ final Set<Integer> availableRevisions) throws NullPointerException {
+ return new SlaveImpl(slaveName, os, ip, ports, availableRevisions);
+ }
+
+}
diff --git a/src/org/torproject/puppetor/rmi/impl/TaskResultImpl.java b/src/org/torproject/puppetor/rmi/impl/TaskResultImpl.java
new file mode 100644
index 0000000..2c66f9f
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/impl/TaskResultImpl.java
@@ -0,0 +1,51 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.impl;
+
+import org.torproject.puppetor.rmi.Task;
+import org.torproject.puppetor.rmi.TaskResult;
+
+public class TaskResultImpl implements TaskResult {
+
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ /**
+ * The task this result is used for.
+ */
+ final private Task task;
+
+ /**
+ * Store the result of the <code>task</code>, or <code>null</code> if no
+ * result is necessary.
+ */
+ final private String result;
+
+ /**
+ * Whether the task execution was successful.
+ */
+ final private boolean wasSuccessful;
+
+ public TaskResultImpl(final boolean success, final Task task, final String result) {
+ wasSuccessful = success;
+ this.task = task;
+ this.result = result;
+ }
+
+ public Task getTask() {
+ return task;
+ }
+
+ public boolean wasSuccessful() {
+ return wasSuccessful;
+ }
+
+ public String getTaskResult() {
+ return result;
+ }
+}
diff --git a/src/org/torproject/puppetor/rmi/impl/TestExecutorImpl.java b/src/org/torproject/puppetor/rmi/impl/TestExecutorImpl.java
new file mode 100644
index 0000000..ddf7065
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/impl/TestExecutorImpl.java
@@ -0,0 +1,555 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.impl;
+
+import java.lang.reflect.Constructor;
+import java.lang.reflect.InvocationTargetException;
+import java.rmi.NoSuchObjectException;
+import java.rmi.server.UnicastRemoteObject;
+import java.util.Collection;
+import java.util.Collections;
+import java.util.HashSet;
+import java.util.LinkedList;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+import java.util.Vector;
+import java.util.concurrent.Callable;
+import java.util.concurrent.ConcurrentHashMap;
+import java.util.concurrent.ConcurrentMap;
+import java.util.concurrent.CopyOnWriteArraySet;
+import java.util.concurrent.ExecutionException;
+import java.util.concurrent.FutureTask;
+import java.util.concurrent.atomic.AtomicBoolean;
+import java.util.concurrent.atomic.AtomicInteger;
+
+import org.torproject.puppetor.rmi.LocalMaster;
+import org.torproject.puppetor.rmi.Network;
+import org.torproject.puppetor.rmi.NetworkDescription;
+import org.torproject.puppetor.rmi.RemoteMaster;
+import org.torproject.puppetor.rmi.Task;
+import org.torproject.puppetor.rmi.TaskExecutionNotSuccessfulException;
+import org.torproject.puppetor.rmi.TaskResult;
+import org.torproject.puppetor.rmi.Test;
+import org.torproject.puppetor.rmi.TestExecutor;
+import org.torproject.puppetor.rmi.TestResult;
+import org.torproject.puppetor.rmi.TorInstance;
+import org.torproject.puppetor.rmi.tasks.BuildCircuitsTask;
+import org.torproject.puppetor.rmi.tasks.ConfigureAsPrivateTask;
+import org.torproject.puppetor.rmi.tasks.CreateDirectoryAuthorityTask;
+import org.torproject.puppetor.rmi.tasks.CreateNetworkTask;
+import org.torproject.puppetor.rmi.tasks.CreateProxyTask;
+import org.torproject.puppetor.rmi.tasks.CreateRouterTask;
+import org.torproject.puppetor.rmi.tasks.ShutdownNodesTask;
+import org.torproject.puppetor.rmi.tasks.StartNodesTask;
+import org.torproject.puppetor.rmi.tasks.TerminateTask;
+
+
+/**
+ * @author Sebastian Hahn
+ */
+public final class TestExecutorImpl implements TestExecutor {
+
+ /**
+ * The TestExecutor instance that we'll use
+ */
+ private static final TestExecutor instance = new TestExecutorImpl();
+
+ /**
+ * Store all local representations of connections to slaves. Use the
+ * slave's name as key.
+ */
+ private static final ConcurrentMap<String, LocalMaster> masters =
+ new ConcurrentHashMap<String, LocalMaster>();
+
+ /**
+ * Store all registered tests until they are started.
+ */
+ private static final List<Test> newTests = new Vector<Test>();
+
+ /**
+ * Store all currently running tests.
+ */
+ // private static final List<Test> runningTests = new Vector<Test>();
+ /**
+ * Store all tests that cannot run because the slave's cannot build such a
+ * network.
+ */
+ private static final Set<Test> cannotRunTests =
+ new CopyOnWriteArraySet<Test>();
+
+ /**
+ * Store the networks that are created out of the NetworkDescriptions.
+ */
+ private static final Set<Network> networks =
+ new CopyOnWriteArraySet<Network>();
+
+ /**
+ * Store the currently active network when executing a test. Null indicates
+ * that either testing hasn't started yet, or is already finished.
+ */
+ private static Network currentNetwork = null;
+
+ /**
+ * Store all failed tests.
+ */
+ private static final Map<Test, TestResult> failedTests =
+ new ConcurrentHashMap<Test, TestResult>();
+
+ /**
+ * Store all successfully completed tests.
+ */
+ private static final Map<Test, TestResult> successfulTests =
+ new ConcurrentHashMap<Test, TestResult>();
+
+ /**
+ * Whether testing has started. This cannot be reset.
+ */
+ private static AtomicBoolean startedTesting = new AtomicBoolean(false);
+
+ /**
+ * Store the id for the next task. Every task gets its own unique id > 0. A
+ * task id equal to 0 indicates that the task was created by the slave to
+ * stop execution.
+ */
+ private static AtomicInteger nextTaskId = new AtomicInteger(1);
+
+ /**
+ * Singleton
+ */
+ private TestExecutorImpl() {}
+
+ public static TestExecutor getInstance() {
+ return instance;
+ }
+
+ public void registerLocalMaster(final LocalMaster master) {
+ if (startedTesting.get() == true) {
+ throw new RuntimeException("Bug: Testing has started already.");
+ }
+ System.out.println(master.getSlave().getName() + " registered");
+ if (null != masters.putIfAbsent(master.getSlave().getName(), master)) {
+ throw new IllegalArgumentException(master.getSlave().getName()
+ + " has already registered with this Server");
+ }
+ }
+
+ public void unregisterLocalMaster(final LocalMaster master) {
+ // XXX Don't forget to mark all tests as failed that depended on this
+ // slave if we have already started with testing-SH
+ try {
+ UnicastRemoteObject.unexportObject((RemoteMaster) master, true);
+ } catch (final NoSuchObjectException e) {
+ throw new RuntimeException(
+ "This is a bug. The master should have been exported.");
+ }
+ if (masters.remove(master.getSlave().getName()) == null) {
+ throw new RuntimeException(
+ "This is a bug. unregisterLocalMaster must never be " +
+ "called twice for the same master.");
+ }
+ }
+
+ /**
+ * Evaluate which tests can be run with the currently connected slaves.
+ * Cancel all currently running tests and report them as failed if they
+ * cannot be restarted with the new (reduced) set of connected slaves,
+ * restart those tests otherwise. XXX This should be merged with
+ * makeNetworks -SH
+ */
+ protected void evaluateNetwork() {
+ if (startedTesting.get() == false) {
+ throw new RuntimeException("Bug: Testing has not been started.");
+ // XXX We need to actually check the network.-SH
+ }
+ }
+
+ public void startTesting() {
+ if (startedTesting.get() == true) {
+ throw new RuntimeException("Bug: Testing has started already.");
+ }
+ startedTesting.set(true);
+ // check which tests we can actually execute
+ evaluateNetwork();
+ for (final Test test : newTests) {
+ makeNetworks(test);
+ }
+ // XXX cancelled networks will still appear here...=SH
+ for (final Network nw : networks) {
+ synchronized(TestExecutorImpl.class) {
+ currentNetwork = nw;
+ }
+ try {
+ executeTaskForMasters(nw.getMasters(),
+ CreateNetworkTask.class, null, null);
+ // The network has been created at the slaves. Set up
+ // authorities, routers and proxies.
+
+ Set<TaskResult> authorityResults =
+ new CopyOnWriteArraySet<TaskResult>();
+ executeTaskForTorInstances(nw.getDirectoryAuthorities(),
+ CreateDirectoryAuthorityTask.class, authorityResults,
+ null);
+ executeTaskForTorInstances(nw.getRoutersOnly(),
+ CreateRouterTask.class, null, null);
+ executeTaskForTorInstances(nw.getProxiesOnly(),
+ CreateProxyTask.class, null, null);
+ if(nw.isPrivateNetwork()) {
+ List<String> authorities = new LinkedList<String>();
+ for(TaskResult result : authorityResults) {
+ authorities.add(result.getTaskResult());
+ }
+ executeTaskForMasters(nw.getMasters(),
+ ConfigureAsPrivateTask.class, null, authorities);
+ }
+ // Start the nodes ...
+ executeTaskForMasters(nw.getMasters(), StartNodesTask.class,
+ null, null);
+ // ... and build circuits. This may take some time.
+ executeTaskForMasters(nw.getMasters(), BuildCircuitsTask.class,
+ null, null);
+
+ // Start using the individual tests that were set up for this
+ // network.
+ for (final Test test : nw.getTests()) {
+ test.doTest();
+ }
+
+ // We don't need the network anymore, make sure it's shut down
+ // properly.
+ executeTaskForMasters(nw.getMasters(), ShutdownNodesTask.class,
+ null, null);
+
+ } catch (final TaskExecutionNotSuccessfulException e) {
+ e.printStackTrace();
+ // XXX Log this properly. -SH
+ }
+
+ }
+ // The terminate Task interrupts the slave's main process to
+ // indicate that it should disconnect.
+ synchronized(TestExecutorImpl.class) {
+ currentNetwork = null;
+ }
+ try {
+ executeTaskForMasters(masters.values(), TerminateTask.class, null,
+ null);
+ } catch (TaskExecutionNotSuccessfulException e) {
+ e.printStackTrace();
+ // XXX Log this properly. -SH
+ }
+ }
+
+ //XXX This method and the next look very similar. See what we can do. -SH
+ public void executeTaskForMasters(final Collection<LocalMaster> masters,
+ final Class<? extends Task> taskClass,
+ final Set<TaskResult> taskResults,
+ Collection<String> additionalInformation)
+ throws TaskExecutionNotSuccessfulException {
+
+ if (masters.isEmpty()) {
+ return;
+ }
+ final Set<MasterTask> mTasks = new HashSet<MasterTask>();
+ Constructor<? extends Task> ctor = null;
+ try {
+ ctor = taskClass.getConstructor(int.class, String.class);
+ String networkName = "";
+ synchronized(TestExecutorImpl.class) {
+ if(currentNetwork == null) {
+ networkName = "";
+ } else {
+ networkName = currentNetwork.getName();
+ }
+ }
+ for (final LocalMaster master : masters) {
+ Task task = ctor.newInstance(
+ nextTaskId.getAndIncrement(), networkName);
+ task.addAdditionalInformation(additionalInformation);
+ mTasks.add(new MasterTask(master, task));
+ }
+
+ final TaskManager man = new TaskManager(mTasks);
+ final FutureTask<Boolean> futureTask =
+ new FutureTask<Boolean>(man);
+ new Thread(futureTask).start();
+ try {
+ if (futureTask.get() == false) {
+ throw new TaskExecutionNotSuccessfulException(man
+ .getTaskResults());
+ }
+ } catch (final InterruptedException e) {
+ throw new TaskExecutionNotSuccessfulException(null);
+ } catch (final ExecutionException e) {
+ // XXX Log this with the cause. Should not happen. -SH
+ e.printStackTrace();
+ throw new TaskExecutionNotSuccessfulException(null);
+ }
+ if(taskResults != null) {
+ taskResults.addAll(man.getTaskResults());
+ }
+ } catch (final SecurityException e) {
+ throw new RuntimeException(
+ "BUG: We're not using a security manager, so where " +
+ "does this exception come from?");
+ } catch (final NoSuchMethodException e) {
+ e.printStackTrace();
+ throw new RuntimeException("BUG: Expected constructor undefined!");
+ } catch (final IllegalArgumentException e) {
+ e.printStackTrace();
+ throw new RuntimeException("BUG: Expected constructor undefined!");
+ } catch (final InstantiationException e) {
+ e.printStackTrace();
+ throw new RuntimeException("BUG: InstantiationException.");
+ } catch (final IllegalAccessException e) {
+ e.printStackTrace();
+ throw new RuntimeException("BUG: Expected constructor undefined!");
+ } catch (final InvocationTargetException e) {
+ e.printStackTrace();
+ throw new RuntimeException("BUG: Constructor threw an exception.");
+ }
+ }
+
+ public Network getCurrentNetwork() {
+ synchronized(TestExecutorImpl.class) {
+ if(currentNetwork == null) {
+ throw new RuntimeException("There is no network currently!");
+ }
+ return currentNetwork;
+ }
+ }
+
+ public void executeTaskForTorInstances(
+ final Collection<TorInstance> torInstances,
+ final Class<? extends Task> taskClass,
+ final Set<TaskResult> taskResults,
+ Collection<String> additionalInformation)
+ throws TaskExecutionNotSuccessfulException {
+
+ if (torInstances.isEmpty()) {
+ return;
+ }
+ final Set<MasterTask> mTasks = new HashSet<MasterTask>();
+ Constructor<? extends Task> ctor = null;
+ try {
+ ctor = taskClass.getConstructor(int.class, String.class,
+ String.class);
+ String networkName = "";
+ synchronized(TestExecutorImpl.class) {
+ if(currentNetwork == null) {
+ networkName = "";
+ } else {
+ networkName = currentNetwork.getName();
+ }
+ }
+ for (final TorInstance torInstance : torInstances) {
+ Task task = ctor.newInstance(nextTaskId.getAndIncrement(),
+ networkName, torInstance.getName());
+ task.addAdditionalInformation(additionalInformation);
+ mTasks.add(new MasterTask(torInstance.getMaster(), task));
+ }
+
+ final TaskManager man = new TaskManager(mTasks);
+ final FutureTask<Boolean> futureTask =
+ new FutureTask<Boolean>(man);
+ new Thread(futureTask).start();
+ try {
+ if (futureTask.get() == false) {
+ throw new TaskExecutionNotSuccessfulException(man
+ .getTaskResults());
+ }
+ } catch (final InterruptedException e) {
+ throw new TaskExecutionNotSuccessfulException(null);
+ } catch (final ExecutionException e) {
+ // XXX Log this with the cause. Should not happen. -SH
+ e.printStackTrace();
+ throw new TaskExecutionNotSuccessfulException(null);
+ }
+ if(taskResults != null) {
+ taskResults.addAll(man.getTaskResults());
+ }
+ } catch (final SecurityException e) {
+ throw new RuntimeException(
+ "BUG: We're not using a security manager, " +
+ "so where does this exception come from?");
+ } catch (final NoSuchMethodException e) {
+ e.printStackTrace();
+ throw new RuntimeException("BUG: Expected constructor undefined!");
+ } catch (final IllegalArgumentException e) {
+ e.printStackTrace();
+ throw new RuntimeException("BUG: Expected constructor undefined!");
+ } catch (final InstantiationException e) {
+ e.printStackTrace();
+ throw new RuntimeException("BUG: InstantiationException.");
+ } catch (final IllegalAccessException e) {
+ e.printStackTrace();
+ throw new RuntimeException("BUG: Expected constructor undefined!");
+ } catch (final InvocationTargetException e) {
+ e.printStackTrace();
+ throw new RuntimeException("BUG: Constructor threw an exception.");
+ }
+ }
+
+ protected void makeNetworks(final Test test) {
+ final NetworkDescription nd = test.getNetworkDescription();
+ // XXX we want to do something much smarter here.-SH
+
+ // XXX obviously, the following code adds the specified amount of
+ // authorities, routers, and proxies for each slave, regardless of
+ // capabilities. That isn't good. -SH
+ for (final LocalMaster master : masters.values()) {
+ int necessaryDirectoryAuthorities =
+ nd.necessaryDirectoryAuthorities();
+ boolean onPrivateNetwork = (necessaryDirectoryAuthorities > 0);
+ // XXX the name choosing doesn't make sense. -SH
+ final Network nw =
+ new NetworkImpl(test.getDescription() + " slavename:"
+ + master.getSlave().getName(), onPrivateNetwork);
+ while (necessaryDirectoryAuthorities >= 1) {
+ TorInstanceImpl authority = new TorInstanceImpl(master,
+ "Authority" + necessaryDirectoryAuthorities);
+ authority.setDirectoryAuthority(true);
+ authority.setOnPrivateNetwork(onPrivateNetwork);
+ nw.addTorInstance(authority);
+ necessaryDirectoryAuthorities--;
+ }
+ int necessaryRouters = nd.necessaryRouters();
+ while (necessaryRouters >= 1) {
+ TorInstanceImpl router = new TorInstanceImpl(master, "Router" +
+ necessaryRouters);
+ router.setRouter(true);
+ router.setOnPrivateNetwork(onPrivateNetwork);
+ nw.addTorInstance(router);
+ necessaryRouters--;
+ }
+ int necessaryProxies = nd.necessaryProxies();
+ while (necessaryProxies >= 1) {
+ TorInstanceImpl proxy = new TorInstanceImpl(master, "Proxy" +
+ necessaryProxies);
+ proxy.setOnPrivateNetwork(onPrivateNetwork);
+ nw.addTorInstance(proxy);
+ necessaryProxies--;
+ }
+ nw.addTest(test);
+ networks.add(nw);
+ }
+ }
+
+ public Map<Test, TestResult> getFailedTests() {
+ return failedTests;
+ }
+
+ public Set<Test> getCannotRunTests() {
+ return cannotRunTests;
+ }
+
+ public Map<Test, TestResult> getSuccessfulTests() {
+ return successfulTests;
+ }
+
+ public void registerTest(final Test test) {
+ if (startedTesting.get() == true) {
+ throw new RuntimeException("Bug: Testing has started already.");
+ }
+ if (test == null) {
+ throw new NullPointerException("tests must not be null");
+ }
+ newTests.add(test);
+ }
+
+ private static class TaskManager implements Callable<Boolean> {
+
+ final private Set<TaskResult> taskResults =
+ new CopyOnWriteArraySet<TaskResult>();
+
+ final private Object taskResultsLock = new Object();
+
+ final private Set<MasterTask> tasks;
+
+ TaskManager(final Set<MasterTask> tasks) {
+ if (tasks.isEmpty()) {
+ throw new IllegalArgumentException("No tasks passed!");
+ }
+ this.tasks =
+ Collections.synchronizedSet(
+ new HashSet<MasterTask>(tasks));
+ }
+
+ public Boolean call() {
+ for (final MasterTask task : tasks) {
+ task.getMaster().addTask(task.getTask());
+ // XXX Asking for trouble here. task could throw an exception!
+ // -SH
+ new Thread(new TaskCollector(task)).start();
+ }
+ try {
+ synchronized (taskResultsLock) {
+ // XXX See above. This needs more thinking -SH
+ while (tasks.size() != taskResults.size()) {
+ taskResultsLock.wait();
+ }
+ // XXX We want some logic here.-SH
+ for (final TaskResult taskResult : taskResults) {
+ if (!taskResult.wasSuccessful()) {
+ return false;
+ }
+ }
+ return true;
+ }
+ } catch (final InterruptedException e) {
+ // XXX figure out a better solution here -SH
+ e.printStackTrace();
+ }
+ return false;
+ }
+
+ public Set<TaskResult> getTaskResults() {
+ return taskResults;
+ }
+
+ private class TaskCollector implements Runnable {
+ private final MasterTask task;
+
+ public void run() {
+ try {
+ final TaskResult taskResult =
+ task.getMaster().getTaskResult(
+ task.getTask().getId());
+ synchronized (taskResultsLock) {
+ taskResults.add(taskResult);
+ taskResultsLock.notifyAll();
+ }
+ } catch (final InterruptedException e) {
+ e.printStackTrace();
+ Thread.currentThread().interrupt();
+ }
+
+ }
+
+ protected TaskCollector(final MasterTask task) {
+ this.task = task;
+ }
+ }
+ }
+
+ private static class MasterTask {
+ final private LocalMaster master;
+ final private Task task;
+
+ MasterTask(final LocalMaster master, final Task task) {
+ this.master = master;
+ this.task = task;
+ }
+
+ public LocalMaster getMaster() {
+ return master;
+ }
+
+ public Task getTask() {
+ return task;
+ }
+ }
+}
diff --git a/src/org/torproject/puppetor/rmi/impl/TorInstanceImpl.java b/src/org/torproject/puppetor/rmi/impl/TorInstanceImpl.java
new file mode 100644
index 0000000..eba41fa
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/impl/TorInstanceImpl.java
@@ -0,0 +1,175 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.impl;
+
+import java.util.Collections;
+import java.util.LinkedList;
+import java.util.List;
+import java.util.concurrent.atomic.AtomicBoolean;
+import java.util.concurrent.atomic.AtomicInteger;
+
+import org.torproject.puppetor.rmi.LocalMaster;
+import org.torproject.puppetor.rmi.TorInstance;
+
+
+/**
+ * Implements a Tor node. Threadsafe.
+ *
+ * @author Sebastian Hahn
+ */
+public class TorInstanceImpl implements TorInstance {
+
+ /**
+ * Saves configuration options in an "option value" format.
+ */
+ private final List<String> configuration = new LinkedList<String>();
+
+ /**
+ * The name of this node.
+ */
+ private final String name;
+
+ /**
+ * The svn revision of the Tor build. 0 means no specified revision.
+ */
+ private final AtomicInteger revision = new AtomicInteger(0);
+
+ /**
+ * Whether we are a bridge authority.
+ */
+ private final AtomicBoolean bridgeAuthority = new AtomicBoolean(false);
+
+ /**
+ * Whether we are a directory authority.
+ */
+ final private AtomicBoolean directoryAuthority = new AtomicBoolean(false);
+
+ /**
+ * Whether we are an exit.
+ */
+ private final AtomicBoolean exit = new AtomicBoolean(false);
+
+ /**
+ * Whether we are a relay.
+ */
+ private final AtomicBoolean router = new AtomicBoolean(false);
+
+ /**
+ * Whether we are a bridge.
+ */
+ private final AtomicBoolean bridge = new AtomicBoolean(false);
+
+ /**
+ * Whether we are on a private network.
+ */
+ private AtomicBoolean privateNetwork = new AtomicBoolean(false);
+
+ /**
+ * Whether the configuration of the tor instance should be replaced by the
+ * config returned from <code>this.getConfiguration()</code>.
+ */
+ private AtomicBoolean replaceConfiguration = new AtomicBoolean(false);
+
+ /**
+ * Provide a link to the slave where this torInstance will run.
+ */
+ private final LocalMaster master;
+
+ /**
+ * @param master
+ * the <code>LocalMaster</code> that represents the slave on which
+ * this Tor instance is running.
+ * @param name
+ * A name to identify the node.
+ * XXX think about duplicates, we need a unique name to identify the
+ * node, but is it smart to use that name as the Tor proxy's name as
+ * well? -SH
+ */
+ TorInstanceImpl(final LocalMaster master, final String name) {
+ this.master = master;
+ this.name = name;
+ }
+
+ public synchronized List<String> getConfiguration() {
+ return Collections.unmodifiableList(configuration);
+ }
+
+ public synchronized void addConfiguration(List<String> newConfig) {
+ configuration.addAll(newConfig);
+ }
+
+ public int getRevision() {
+ return revision.get();
+ }
+
+ public void setRevision(final int revision) {
+ this.revision.set(revision);
+ }
+
+ public boolean isBridgeAuthority() {
+ return bridgeAuthority.get();
+ }
+
+ public void setBridgeAuthority(boolean isBridgeAuthority) {
+ bridgeAuthority.set(isBridgeAuthority);
+ }
+
+ public boolean isDirectoryAuthority() {
+ return directoryAuthority.get();
+ }
+
+ public void setDirectoryAuthority(boolean isDirectoryAuthority) {
+ if(isDirectoryAuthority) {
+ setRouter(true);
+ }
+ directoryAuthority.set(isDirectoryAuthority);
+ }
+
+ public boolean isExit() {
+ return exit.get();
+ }
+
+ public boolean isRouter() {
+ return router.get();
+ }
+
+ public void setRouter(boolean isRouter) {
+ if(!isRouter) {
+ setDirectoryAuthority(false);
+ }
+ router.set(isRouter);
+ }
+
+ public boolean isOnPrivateNetwork() {
+ return privateNetwork.get();
+ }
+
+ public void setOnPrivateNetwork(boolean onPrivateNetwork) {
+ privateNetwork.set(onPrivateNetwork);
+ }
+
+ public boolean shouldReplaceConfiguration() {
+ return replaceConfiguration.get();
+ }
+
+ public void setShouldReplaceConfiguration(
+ boolean shouldReplaceConfiguration) {
+ replaceConfiguration.set(shouldReplaceConfiguration);
+ }
+
+ public boolean isBridge() {
+ return bridge.get();
+ }
+
+ public LocalMaster getMaster() {
+ return master;
+ }
+
+ public synchronized String getName() {
+ return name;
+ }
+
+}
diff --git a/src/org/torproject/puppetor/rmi/tasks/AccessGoogleTask.java b/src/org/torproject/puppetor/rmi/tasks/AccessGoogleTask.java
new file mode 100644
index 0000000..e2dae7d
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/tasks/AccessGoogleTask.java
@@ -0,0 +1,74 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.tasks;
+
+import java.rmi.RemoteException;
+
+import org.torproject.puppetor.ClientApplication;
+import org.torproject.puppetor.ClientEventType;
+import org.torproject.puppetor.Event;
+import org.torproject.puppetor.EventListener;
+import org.torproject.puppetor.EventManager;
+import org.torproject.puppetor.Network;
+import org.torproject.puppetor.NetworkFactory;
+import org.torproject.puppetor.rmi.impl.AbstractTaskImpl;
+
+
+public class AccessGoogleTask extends AbstractTaskImpl {
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ /**
+ * The name of this tor node.
+ */
+ private final String nodeName;
+
+ public AccessGoogleTask(final int id, final String networkName,
+ final String nodeName) {
+ super(id, networkName);
+ this.nodeName = nodeName;
+ }
+
+ @Override
+ public void execute() throws InterruptedException, RemoteException {
+ final Network network = NetworkFactory.getNetworkByName(name);
+
+ final ClientApplication client =
+ network.createClient("client", "www.google.com", 80, network.getNode(nodeName)
+ .getSocksPort());
+
+ // create event listener to listen for client application events
+ final EventListener clientEventListener = new EventListener() {
+
+ // remember time when request was sent
+ //private long before;
+
+ public void handleEvent(Event event) {
+ if (event.getType() == ClientEventType.CLIENT_SENDING_REQUEST) {
+ //before = System.currentTimeMillis();
+ } else if (event.getType() == ClientEventType.CLIENT_REPLY_RECEIVED) {
+ // XXX Log how long requests took, see whether we need this all -SH
+ ;
+ }
+ }
+ };
+
+ // obtain reference to event manager to be able to respond to events
+ final EventManager manager = network.getEventManager();
+
+ manager.addEventListener(client.getClientApplicationName(),
+ clientEventListener);
+
+ // perform at most three request with a timeout of 20 seconds each
+ client.startRequests(3, 20000, true);
+
+ // block this thread as long as client requests are running, but don't wait forever.
+ reportResult(manager.waitForAnyOccurence(client.getClientApplicationName(),
+ ClientEventType.CLIENT_REQUESTS_PERFORMED, 75000));
+ }
+}
diff --git a/src/org/torproject/puppetor/rmi/tasks/AddHiddenServiceTask.java b/src/org/torproject/puppetor/rmi/tasks/AddHiddenServiceTask.java
new file mode 100644
index 0000000..cf206bb
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/tasks/AddHiddenServiceTask.java
@@ -0,0 +1,106 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.tasks;
+
+import java.rmi.RemoteException;
+import java.util.Set;
+
+import org.torproject.puppetor.ClientApplication;
+import org.torproject.puppetor.ClientEventType;
+import org.torproject.puppetor.Event;
+import org.torproject.puppetor.EventListener;
+import org.torproject.puppetor.EventManager;
+import org.torproject.puppetor.HiddenService;
+import org.torproject.puppetor.HiddenServiceEventType;
+import org.torproject.puppetor.Network;
+import org.torproject.puppetor.NetworkFactory;
+import org.torproject.puppetor.ProxyNode;
+import org.torproject.puppetor.PuppeTorException;
+import org.torproject.puppetor.ServerApplication;
+import org.torproject.puppetor.rmi.impl.AbstractTaskImpl;
+
+
+public class AddHiddenServiceTask extends AbstractTaskImpl {
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ /**
+ * The name of this tor node.
+ */
+ private final String nodeName;
+
+ public AddHiddenServiceTask(final int id, final String networkName,
+ final String nodeName) {
+ super(id, networkName);
+ this.nodeName = nodeName;
+ }
+
+ @Override
+ public void execute() throws InterruptedException, RemoteException {
+ final Network network = NetworkFactory.getNetworkByName(name);
+ ProxyNode node = network.getNode(nodeName);
+ try {
+ // add hidden service
+ final HiddenService hidServ = node.addHiddenService("hidServ");
+ node.writeConfiguration();
+ node.hup();
+ final EventManager manager = network.getEventManager();
+
+ // wait for 6 minutes that the proxy has published its first RSD
+ if (!manager.waitForAnyOccurence(node.getNodeName(),
+ HiddenServiceEventType.BOB_DESC_PUBLISHED_RECEIVED,
+ 6L * 60L * 1000L)) {
+ // failed to publish an RSD
+ reportResult(false);
+ return;
+ }
+
+ // create server application
+ final ServerApplication server =
+ network.createServer("server", hidServ.getServicePort());
+
+ // create client application, pick one of the nodes.
+ // XXX fix this so we can choose which node we want. -SH
+ Set<String> nodes = network.getAllNodes().keySet();
+ String[] localNodeNames = new String[10];
+ localNodeNames = nodes.toArray(localNodeNames);
+ final ClientApplication client =
+ network.createClient("client",
+ hidServ.determineOnionAddress(), hidServ
+ .getVirtualPort(), network.getNode(localNodeNames[0]).getSocksPort());
+
+ // register event listener
+ final EventListener clientAndServerEventListener = new EventListener() {
+ public void handleEvent(Event event) {
+ // XXX log -SH
+ }
+ };
+ manager.addEventListener(client.getClientApplicationName(),
+ clientAndServerEventListener);
+ manager.addEventListener(server.getServerApplicationName(),
+ clientAndServerEventListener);
+
+ // start server
+ server.startListening();
+
+ // perform at most five request with a timeout of 45 seconds each
+ client.startRequests(5, 45000, true);
+
+ // wait for request to be performed
+ manager.waitForAnyOccurence(client.getClientApplicationName(),
+ ClientEventType.CLIENT_REQUESTS_PERFORMED);
+ reportResult(true);
+ } catch (final PuppeTorException e) {
+ e.printStackTrace();
+ reportResult(false);
+ }
+
+
+ }
+
+}
diff --git a/src/org/torproject/puppetor/rmi/tasks/BuildCircuitsTask.java b/src/org/torproject/puppetor/rmi/tasks/BuildCircuitsTask.java
new file mode 100644
index 0000000..85d0b86
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/tasks/BuildCircuitsTask.java
@@ -0,0 +1,41 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.tasks;
+
+import java.rmi.RemoteException;
+
+import org.torproject.puppetor.Network;
+import org.torproject.puppetor.NetworkFactory;
+import org.torproject.puppetor.PuppeTorException;
+import org.torproject.puppetor.rmi.impl.AbstractTaskImpl;
+
+
+public class BuildCircuitsTask extends AbstractTaskImpl {
+
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ public BuildCircuitsTask(final int id, final String networkName) {
+ super(id, networkName);
+ }
+
+ @Override
+ public void execute() throws InterruptedException, RemoteException {
+ final Network network = NetworkFactory.getNetworkByName(name);
+ try {
+ if (!network.hupUntilUp(50, 10000)) {
+ reportResult(false);
+ }
+ } catch (final PuppeTorException e) {
+ e.printStackTrace();
+ reportResult(false);
+ }
+ reportResult(true);
+ }
+
+}
diff --git a/src/org/torproject/puppetor/rmi/tasks/ConfigureAsPrivateTask.java b/src/org/torproject/puppetor/rmi/tasks/ConfigureAsPrivateTask.java
new file mode 100644
index 0000000..2c6aa20
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/tasks/ConfigureAsPrivateTask.java
@@ -0,0 +1,50 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.tasks;
+
+import java.rmi.RemoteException;
+import java.util.Collection;
+import java.util.LinkedList;
+import java.util.List;
+
+import org.torproject.puppetor.Network;
+import org.torproject.puppetor.NetworkFactory;
+import org.torproject.puppetor.PuppeTorException;
+import org.torproject.puppetor.rmi.impl.AbstractTaskImpl;
+
+
+public class ConfigureAsPrivateTask extends AbstractTaskImpl {
+
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ final List<String> authorityInformation = new LinkedList<String>();
+
+ public ConfigureAsPrivateTask(final int id, final String networkName) {
+ super(id, networkName);
+ }
+
+ @Override
+ public void execute() throws InterruptedException, RemoteException {
+ final Network network = NetworkFactory.getNetworkByName(name);
+ network.configureAsPartOfPrivateNetwork(authorityInformation);
+ try {
+ network.writeConfigurations();
+ reportResult(true);
+ } catch (final PuppeTorException e) {
+ e.printStackTrace();
+ reportResult(false);
+ }
+ }
+
+ @Override
+ public void addAdditionalInformation(
+ Collection<String> additionalInformation) {
+ authorityInformation.addAll(additionalInformation);
+ }
+}
\ No newline at end of file
diff --git a/src/org/torproject/puppetor/rmi/tasks/CreateDirectoryAuthorityTask.java b/src/org/torproject/puppetor/rmi/tasks/CreateDirectoryAuthorityTask.java
new file mode 100644
index 0000000..e3bd07c
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/tasks/CreateDirectoryAuthorityTask.java
@@ -0,0 +1,48 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.tasks;
+
+import java.rmi.RemoteException;
+
+import org.torproject.puppetor.DirectoryNode;
+import org.torproject.puppetor.Network;
+import org.torproject.puppetor.NetworkFactory;
+import org.torproject.puppetor.PuppeTorException;
+import org.torproject.puppetor.rmi.impl.AbstractTaskImpl;
+
+
+public class CreateDirectoryAuthorityTask extends AbstractTaskImpl {
+
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ /**
+ * The name of this Tor authority.
+ */
+ private final String authorityName;
+
+ public CreateDirectoryAuthorityTask(final int id, final String networkName,
+ final String authorityName) {
+ super(id, networkName);
+ this.authorityName = authorityName;
+ }
+
+ @Override
+ public void execute() throws InterruptedException, RemoteException {
+ final Network network = NetworkFactory.getNetworkByName(name);
+ DirectoryNode node = network.createDirectory(authorityName);
+ // write configuration of proxy node
+ try {
+ network.writeConfigurations();
+ reportResult(true, node.getDirServerString());
+ } catch (final PuppeTorException e) {
+ e.printStackTrace();
+ reportResult(false);
+ }
+ }
+}
diff --git a/src/org/torproject/puppetor/rmi/tasks/CreateNetworkTask.java b/src/org/torproject/puppetor/rmi/tasks/CreateNetworkTask.java
new file mode 100644
index 0000000..31fea77
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/tasks/CreateNetworkTask.java
@@ -0,0 +1,31 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.tasks;
+
+import java.rmi.RemoteException;
+
+import org.torproject.puppetor.NetworkFactory;
+import org.torproject.puppetor.rmi.impl.AbstractTaskImpl;
+
+
+public class CreateNetworkTask extends AbstractTaskImpl {
+
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ public CreateNetworkTask(final int id, final String name) {
+ super(id, name);
+ }
+
+ @Override
+ public void execute() throws InterruptedException, RemoteException {
+ NetworkFactory.createNetwork(name);
+ reportResult(true);
+ }
+
+}
diff --git a/src/org/torproject/puppetor/rmi/tasks/CreateProxyTask.java b/src/org/torproject/puppetor/rmi/tasks/CreateProxyTask.java
new file mode 100644
index 0000000..5f4f5c0
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/tasks/CreateProxyTask.java
@@ -0,0 +1,48 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.tasks;
+
+import java.rmi.RemoteException;
+
+import org.torproject.puppetor.Network;
+import org.torproject.puppetor.NetworkFactory;
+import org.torproject.puppetor.PuppeTorException;
+import org.torproject.puppetor.rmi.impl.AbstractTaskImpl;
+
+
+public class CreateProxyTask extends AbstractTaskImpl {
+
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ /**
+ * The name of this tor proxy.
+ */
+ private final String proxyName;
+
+ public CreateProxyTask(final int id, final String networkName,
+ final String proxyName) {
+ super(id, networkName);
+ this.proxyName = proxyName;
+ }
+
+ @Override
+ public void execute() throws InterruptedException, RemoteException {
+ final Network network = NetworkFactory.getNetworkByName(name);
+ network.createProxy(proxyName);
+
+ // write configuration of proxy node
+ try {
+ network.writeConfigurations();
+ reportResult(true);
+ } catch (final PuppeTorException e) {
+ e.printStackTrace();
+ reportResult(false);
+ }
+ }
+}
diff --git a/src/org/torproject/puppetor/rmi/tasks/CreateRouterTask.java b/src/org/torproject/puppetor/rmi/tasks/CreateRouterTask.java
new file mode 100644
index 0000000..09b74cf
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/tasks/CreateRouterTask.java
@@ -0,0 +1,48 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.tasks;
+
+import java.rmi.RemoteException;
+
+import org.torproject.puppetor.Network;
+import org.torproject.puppetor.NetworkFactory;
+import org.torproject.puppetor.PuppeTorException;
+import org.torproject.puppetor.rmi.impl.AbstractTaskImpl;
+
+
+public class CreateRouterTask extends AbstractTaskImpl {
+
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ /**
+ * The name of this router.
+ */
+ private final String routerName;
+
+ public CreateRouterTask(final int id, final String networkName,
+ final String routerName) {
+ super(id, networkName);
+ this.routerName = routerName;
+ }
+
+ @Override
+ public void execute() throws InterruptedException, RemoteException {
+ final Network network = NetworkFactory.getNetworkByName(name);
+ network.createRouter(routerName);
+ // write configuration of router node
+ try {
+ network.writeConfigurations();
+ reportResult(true);
+ } catch (final PuppeTorException e) {
+ e.printStackTrace();
+ reportResult(false);
+ }
+
+ }
+}
diff --git a/src/org/torproject/puppetor/rmi/tasks/ShutdownNodesTask.java b/src/org/torproject/puppetor/rmi/tasks/ShutdownNodesTask.java
new file mode 100644
index 0000000..7837d5b
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/tasks/ShutdownNodesTask.java
@@ -0,0 +1,38 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.tasks;
+
+import java.rmi.RemoteException;
+
+import org.torproject.puppetor.Network;
+import org.torproject.puppetor.NetworkFactory;
+import org.torproject.puppetor.PuppeTorException;
+import org.torproject.puppetor.rmi.impl.AbstractTaskImpl;
+
+
+public class ShutdownNodesTask extends AbstractTaskImpl {
+
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ public ShutdownNodesTask(final int id, final String name) {
+ super(id, name);
+ }
+
+ @Override
+ public void execute() throws InterruptedException, RemoteException {
+ final Network network = NetworkFactory.getNetworkByName(name);
+ try {
+ network.shutdownNodes();
+ reportResult(true);
+ } catch (final PuppeTorException e) {
+ reportResult(false);
+ }
+ }
+
+}
diff --git a/src/org/torproject/puppetor/rmi/tasks/StartNodesTask.java b/src/org/torproject/puppetor/rmi/tasks/StartNodesTask.java
new file mode 100644
index 0000000..1049cd5
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/tasks/StartNodesTask.java
@@ -0,0 +1,42 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.tasks;
+
+import java.rmi.RemoteException;
+
+import org.torproject.puppetor.Network;
+import org.torproject.puppetor.NetworkFactory;
+import org.torproject.puppetor.PuppeTorException;
+import org.torproject.puppetor.rmi.impl.AbstractTaskImpl;
+
+
+public class StartNodesTask extends AbstractTaskImpl {
+
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ public StartNodesTask(final int id, final String networkName) {
+ super(id, networkName);
+ }
+
+ @Override
+ public void execute() throws InterruptedException, RemoteException {
+ final Network network = NetworkFactory.getNetworkByName(name);
+ try {
+ if (!network.startNodes(5000)) {
+ // failed to start the proxy
+ reportResult(false);
+ }
+ } catch (final PuppeTorException e) {
+ e.printStackTrace();
+ reportResult(false);
+ }
+ reportResult(true);
+ }
+
+}
diff --git a/src/org/torproject/puppetor/rmi/tasks/TerminateTask.java b/src/org/torproject/puppetor/rmi/tasks/TerminateTask.java
new file mode 100644
index 0000000..c2e1ac7
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/tasks/TerminateTask.java
@@ -0,0 +1,30 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.tasks;
+
+import java.rmi.RemoteException;
+
+import org.torproject.puppetor.rmi.impl.AbstractTaskImpl;
+
+
+public class TerminateTask extends AbstractTaskImpl {
+
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ public TerminateTask(final int id, final String name) {
+ super(id, name);
+ }
+
+ @Override
+ public void execute() throws InterruptedException, RemoteException {
+ Thread.currentThread().interrupt();
+ reportResult(true);
+ }
+
+}
diff --git a/src/org/torproject/puppetor/rmi/tests/AccessGoogleOverPublicTorNetwork.java b/src/org/torproject/puppetor/rmi/tests/AccessGoogleOverPublicTorNetwork.java
new file mode 100644
index 0000000..4eb6b8a
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/tests/AccessGoogleOverPublicTorNetwork.java
@@ -0,0 +1,40 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.tests;
+
+import java.util.Set;
+
+import org.torproject.puppetor.rmi.Network;
+import org.torproject.puppetor.rmi.TaskExecutionNotSuccessfulException;
+import org.torproject.puppetor.rmi.TorInstance;
+import org.torproject.puppetor.rmi.impl.AbstractTestImpl;
+import org.torproject.puppetor.rmi.impl.NetworkDescriptionImpl;
+import org.torproject.puppetor.rmi.tasks.AccessGoogleTask;
+
+
+public class AccessGoogleOverPublicTorNetwork extends AbstractTestImpl {
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ public AccessGoogleOverPublicTorNetwork() {
+ super("Use the public Tor network to connect to google.com and wait" +
+ " for a response", new NetworkDescriptionImpl(1, 0, 0));
+ }
+
+ public void doTest() {
+ Network nw = exec.getCurrentNetwork();
+ Set<TorInstance> proxies = nw.getProxies();
+ try {
+ exec.executeTaskForTorInstances(proxies, AccessGoogleTask.class, null, null);
+ System.out.println("Accessing google succeeded");
+ } catch (TaskExecutionNotSuccessfulException e) {
+ System.out.println("Accessing google failed");
+ // Log this properly, especially where we failed
+ }
+ }
+}
\ No newline at end of file
diff --git a/src/org/torproject/puppetor/rmi/tests/StartingAndAccessingHiddenServiceOverPrivateTorNetworkTest.java b/src/org/torproject/puppetor/rmi/tests/StartingAndAccessingHiddenServiceOverPrivateTorNetworkTest.java
new file mode 100644
index 0000000..b11a7c3
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/tests/StartingAndAccessingHiddenServiceOverPrivateTorNetworkTest.java
@@ -0,0 +1,41 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.tests;
+
+import java.util.Set;
+
+import org.torproject.puppetor.rmi.Network;
+import org.torproject.puppetor.rmi.TaskExecutionNotSuccessfulException;
+import org.torproject.puppetor.rmi.TorInstance;
+import org.torproject.puppetor.rmi.impl.AbstractTestImpl;
+import org.torproject.puppetor.rmi.impl.NetworkDescriptionImpl;
+import org.torproject.puppetor.rmi.tasks.AddHiddenServiceTask;
+
+
+public class StartingAndAccessingHiddenServiceOverPrivateTorNetworkTest extends
+ AbstractTestImpl {
+ /**
+ * Required for serialization. Needs to change for new released versions.
+ */
+ private static final long serialVersionUID = 1L;
+
+ public StartingAndAccessingHiddenServiceOverPrivateTorNetworkTest() {
+ super("Create a private Tor network, set up a hidden service, " +
+ "and access it", new NetworkDescriptionImpl(1, 3, 2));
+ }
+
+ public void doTest() {
+ Network nw = exec.getCurrentNetwork();
+ Set<TorInstance> proxies = nw.getProxiesOnly();
+ try {
+ exec.executeTaskForTorInstances(proxies, AddHiddenServiceTask.class, null, null);
+ System.out.println("Accessing the hidden service succeeded.");
+ } catch (TaskExecutionNotSuccessfulException e) {
+ System.out.println("Accessing the hidden service failed.");
+ // Log this properly, especially where we failed
+ }
+ }
+}
diff --git a/src/org/torproject/puppetor/rmi/tests/TestRegistration.java b/src/org/torproject/puppetor/rmi/tests/TestRegistration.java
new file mode 100644
index 0000000..0260fdd
--- /dev/null
+++ b/src/org/torproject/puppetor/rmi/tests/TestRegistration.java
@@ -0,0 +1,33 @@
+/*
+ * Copyright 2007 Karsten Loesing
+ * Copyright 2008-2009 Karsten Loesing and Sebastian Hahn
+ * See LICENSE for licensing information
+ */
+package org.torproject.puppetor.rmi.tests;
+
+import org.torproject.puppetor.rmi.TestExecutor;
+
+public class TestRegistration {
+
+ private static TestRegistration instance = new TestRegistration();
+
+ public void registerTests(TestExecutor testExecutor) {
+ testExecutor.registerTest(new AccessGoogleOverPublicTorNetwork());
+ testExecutor.registerTest(new
+ StartingAndAccessingHiddenServiceOverPrivateTorNetworkTest());
+ }
+
+ /**
+ * Singleton
+ */
+ private TestRegistration() {}
+
+ /**
+ * @return
+ * the <code>TestRegistration</code> instance.
+ */
+ public static TestRegistration getInstance() {
+ return instance;
+ }
+
+}
--
1.5.6.5