make server
./server.out <SERVER_PORT>
- A server will be initialised on the local IP and port
SERVER_PORT
. If the port is busy, an error will be thrown.
make client
./client.out <SERVER_IP> <SERVER_PORT> <CLIENT_PORT>
SERVER_IP
: IP address of the server
SERVER_PORT
: Port name of the server
CLIENT_PORT
: Port of which client should establish it's own server to accept commands and run locally (see below for explanation)
- The clustershell server establishes a server on the given port and waits for client connections. On receiving a connection request, the server creates a new thread for each client. We have chosen threads instead of processes as the clients and the main process have to share some data. When a new request comes, the server checks the IP in config file and gets the machine name. If the machine name is not found, the connection is closed. On successful connection and teardown, the client thread informs the parent about the connection establishment/teardown and parent uses this information to keep track of active connections.
- When the server first establishes a connection with a client, the first message it expects is the client port on which the clustershell client is running it's own server to accept commands.
- When the server received a command from the client, it parses the pipe-separated commands and runs each command on the specific machine (or sequentially runs of each active connection in case of
n*
and gathers output). This is done by establishing a TCP connection with the IP given in config file and port given by client port. Piping is performed through sending the output of previous command along with the actual command to the client.
- Each client runs two processes. The main process runs the user-facing shell and acts as a netowrking client connected with the clustershell server. Upon receiving commands from the user through
stdin
, the main process sends the command to the server.
- The second process establishes it's own server on
CLIENT_PORT
and listens to requests from clustershell server to run commands on the machine and return the output.
- The figure below shows an overview or server - client communication as explained above.


- Server initialised: When the server is run, it binds to the port given by command line argument
SERVER_PORT
and listens to client requests.
- First machine connects with server: When the first machine establishes a TCP connection with the server, the server creates a new thread for handling requests from this machine. The server first verifies that the IP exists in config and gets the machine name (n1). Then, the server expects the client to send the client port before sending any shell commands.
Additionally, the client on initialising forks a child process that binds to a port and listens to requests from the server.
- Second machine connects with server: Same as step 2.
- n1 sends
n1.ls | n2.wc
command to server: The server parses the command and creates a linked list of pipe separated commands. Further, the server performs the below steps.
- Server sends command
ls
to n1: The server establishes a TCP connection with n1 and sends the ls
command to n1 and waits for the output. Note that here the clustershell server actually acts like a client.
- n1 responds back with output: n1 sends the output of
ls
to server.
- Server sends command
wc
to n2: The server establishes a TCP connection with n2 and sends the wc
command along with output of n1.ls
to n2 and waits for the output. Note that here the clustershell server actually acts like a client.
- n2 responds back with output: n2 sends the output of
n1.ls | n2.wc
to server.
- Server sends final output to n1: The server sends the final output to requesting client n1.
- The server keeps track of open connections that can be queried by a client using
nodes
command.
- The server accepts connections only from clients specified in the config file.
- There can be more than one clients running on the same IP. The command line argument
CLIENT_PORT
is used to differentiate between such clients.
- The client can run commands locally (no machine specified, eg: ls), or on a particular machine (eg: n2.ls), or on all current active connections (eg: n*.ls)
- The shell is able to execute piped commands (eg: n2.ls | n1.wc)
- The shell supports the
cd
command
- To exit server, press
Ctrl+C
- To exit shell on client, run
exit
n1.cat file | n2.sort | n3.uniq
