The goal of this project is to allow you to practice using low-level system calls related to sockets and networking. To do this, you will create two new programs:

  1. thor.py: This is a basic HTTP client that will hammer a remote HTTP server by making multiple requests.

  2. spidey.c: This is a basic HTTP server that supports directory listings, static files, and CGI scripts.

Once you have these programs, you will conduct experiments using thor.py to test the latency and throughput of spidey.py. You will then need to make a presentation about your project and demonstrate your code and discuss your results directly to a TA or the instructor.

For this project, you are to work in groups of 2 to 4 and record your source code and any responses to a new project GitLab repository. You should push your work to your GitLab repository by Noon, May 5, 2018.


Activity 0: GitLab Repository

Because you will be working in groups, you will need to fork and clone a new project repository:

https://gitlab.com/nd-cse-20289-sp18/cse-20289-sp18-project

To do this, you should follow the same instructions from Reading 00 (except adjust for the different repository location). Besure to do the following:

  1. Make your project repository is private.

  2. Give the teaching staff and your group members developer access to your project repository.

  3. Record your group members in the Project Description and in the README.md.

Note: You should only have one repository per group.

Once forked, you should clone your repository to a local machine. Inside the project folder, you should have the following files:

project
    \_ www                    # This is the default web root directory
        \_ html               # This is a folder for HTML documents
            \_ index.html     # This is an example HTML document
        \_ scripts            # This is a folder for CGI scripts
            \_ cowsay.sh      # This is an example CGI script
            \_ env.sh         # This is an example CGI script
        \_ text               # This is a folder for text documents
            \_ hackers.txt    # This is an example text document
            \_ lyrics.txt     # This is an example text document
    \_ Makefile               # This is the Makefile for building all the project artifacts
    \_ README.md              # This is the README file for recording your responses
    \_ forking.c              # This is the C99 implementation file for the forking mode
    \_ handler.c              # This is the C99 implementation file for the handler functions
    \_ request.c              # This is the C99 implementation file for the request functions
    \_ single.c               # This is the C99 implementation file for the single mode
    \_ socket.c               # This is the C99 implementation file for the socket functions
    \_ spidey.c               # This is the C99 implementation file for the main execution
    \_ spidey.h               # This is the C99 header file for the project
    \_ thor.py                # This is the Python script for the HTTP client
    \_ utils.c                # This is the C99 implementation file for various utility functions

The details on what you need to implement are described in the following sections.


Activity 1: Thor

The first program is thor.py, which is a basic HTTP client similar to curl or wget that supports the following features:

  1. Performs HTTP GET requests using the requests package.

  2. Supports utilizing multiple processes using the multiprocessing package.

  3. Performs multiple requests per process.

  4. Computes the elapsed times for each HTTP request.

Overview

As shown above, thor.py uses the multiprocessing module to start multiple processes. Each process than makes multiple HTTP requests using the requests module. Each HTTP request is timed using the time module and the elapsed time is displayed. Likewise, the average elapsed time for each process is also displayed, along with the average elapsed time for all the requests across all processes.

Usage

Given a URL, thor.py uses the HTTP protocol to fetch the contents of the URL. The -r flag sets the number of HTTP requests to be made per process (default is 1), while -p sets the number of processes to execute in parallel (default is 1). The -v flag forces the program to dump the contents of the URL to standard output.

# Display help message
$ ./thor.py
Usage: thor.py [-p PROCESSES -r REQUESTS -v] URL
    -h              Display help message
    -v              Display verbose output

    -p  PROCESSES   Number of processes to utilize (1)
    -r  REQUESTS    Number of requests per process (1)

Examples

Below are some examples of thor.py in action:

Single Request

# Perform single request
$ ./thor.py http://example.com
Process: 0, Request: 0, Elapsed Time: 0.03
Process: 0, AVERAGE   , Elapsed Time: 0.03
TOTAL AVERAGE ELAPSED TIME: 0.03

Multiple Requests

# Perform 10 requests
$ ./thor.py -r 10 http://example.com
Process: 0, Request: 0, Elapsed Time: 0.03
Process: 0, Request: 1, Elapsed Time: 0.03
Process: 0, Request: 2, Elapsed Time: 0.02
Process: 0, Request: 3, Elapsed Time: 0.03
Process: 0, Request: 4, Elapsed Time: 0.03
Process: 0, Request: 5, Elapsed Time: 0.03
Process: 0, Request: 6, Elapsed Time: 0.02
Process: 0, Request: 7, Elapsed Time: 0.02
Process: 0, Request: 8, Elapsed Time: 0.03
Process: 0, Request: 9, Elapsed Time: 0.02
Process: 0, AVERAGE   , Elapsed Time: 0.03
TOTAL AVERAGE ELAPSED TIME: 0.03

Multiple Requests with Multiple Processes

# Perform 5 requests with 5 processes
$ ./thor.py -r 5 -p 5 http://example.com
Process: 2, Request: 0, Elapsed Time: 0.03
Process: 4, Request: 0, Elapsed Time: 0.03
Process: 0, Request: 0, Elapsed Time: 0.03
Process: 1, Request: 0, Elapsed Time: 0.03
Process: 3, Request: 0, Elapsed Time: 0.03
Process: 2, Request: 1, Elapsed Time: 0.02
Process: 0, Request: 1, Elapsed Time: 0.03
Process: 1, Request: 1, Elapsed Time: 0.02
Process: 4, Request: 1, Elapsed Time: 0.03
Process: 3, Request: 1, Elapsed Time: 0.03
Process: 2, Request: 2, Elapsed Time: 0.03
Process: 0, Request: 2, Elapsed Time: 0.02
Process: 1, Request: 2, Elapsed Time: 0.03
Process: 4, Request: 2, Elapsed Time: 0.03
Process: 3, Request: 2, Elapsed Time: 0.02
Process: 0, Request: 3, Elapsed Time: 0.03
Process: 2, Request: 3, Elapsed Time: 0.03
Process: 3, Request: 3, Elapsed Time: 0.03
Process: 1, Request: 3, Elapsed Time: 0.03
Process: 4, Request: 3, Elapsed Time: 0.03
Process: 2, Request: 4, Elapsed Time: 0.03
Process: 0, Request: 4, Elapsed Time: 0.03
Process: 0, AVERAGE   , Elapsed Time: 0.03
Process: 2, AVERAGE   , Elapsed Time: 0.03
Process: 3, Request: 4, Elapsed Time: 0.03
Process: 3, AVERAGE   , Elapsed Time: 0.03
Process: 1, Request: 4, Elapsed Time: 0.03
Process: 1, AVERAGE   , Elapsed Time: 0.03
Process: 4, Request: 4, Elapsed Time: 0.02
Process: 4, AVERAGE   , Elapsed Time: 0.03
TOTAL AVERAGE ELAPSED TIME: 0.03

Single Request with Verbose Output

# Perform single request with verbose output
$ ./thor.py -v http://example.com
<!doctype html>
<html>
<head>
    <title>Example Domain</title>

    <meta charset="utf-8" />
    <meta http-equiv="Content-type" content="text/html; charset=utf-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1" />
    <style type="text/css">
    body {
        background-color: #f0f0f2;
        margin: 0;
        padding: 0;
        font-family: "Open Sans", "Helvetica Neue", Helvetica, Arial, sans-serif;

    }
    div {
        width: 600px;
        margin: 5em auto;
        padding: 50px;
        background-color: #fff;
        border-radius: 1em;
    }
    a:link, a:visited {
        color: #38488f;
        text-decoration: none;
    }
    @media (max-width: 700px) {
        body {
            background-color: #fff;
        }
        div {
            width: auto;
            margin: 0 auto;
            border-radius: 0;
            padding: 1em;
        }
    }
    </style>
</head>

<body>
<div>
    <h1>Example Domain</h1>
    <p>This domain is established to be used for illustrative examples in documents. You may use this
    domain in examples without prior coordination or asking for permission.</p>
    <p><a href="http://www.iana.org/domains/example">More information...</a></p>
</div>
</body>
</html>

Process: 0, Request: 0, Elapsed Time: 0.03
Process: 0, AVERAGE   , Elapsed Time: 0.03
TOTAL AVERAGE ELAPSED TIME: 0.03

Hints


Activity 2: Spidey

The second program is spidey, which is a basic HTTP server similar to Apache or NGINX that supports the following features:

An example of spidey in action can be found at: xavier.h4x0r.space:9898.

Overview

Overall, the implementation of a HTTP server is straightforward:

  1. First, we allocate a server socket, bind it to a port, and then listen for incoming connections.

  2. Next, we accept an incoming client connection and parse the input data stream into a HTTP request structure.

  3. Based on the request's parameters, we then form a response and send it back to the client.

  4. Continue to perform steps 2 and 3 for as long as the server is running. If we are in forking mode, then we simply fork after we accept a connection and let the child process handle parsing and responding to the request. Otherwise, we simply handle one client at a time in single connection mode.

Usage

When executed, spidey opens a socket on the PORT specified by the -p flag (default is 9898) and handles HTTP requests for files in the path directory specified by the -d flag (default is the www folder in the current directory). If -c forking is specified, then spidey will fork a child process for each incoming client request. The user may also set the default mimetype for files via the -M flag and set the path to the mime.types file via the -m flag (default is /etc/mime.types).

# Display help message
$ ./spidey -h
Usage: ./spidey [hcmMpr]
Options:
    -h            Display help message
    -c mode       Single or Forking mode
    -m path       Path to mimetypes file
    -M mimetype   Default mimetype
    -p port       Port to listen on
    -r path       Root directory

Examples

Below are some examples of spidey in action:

Single Connection Mode (with defaults)

$ ./spidey
[ 3734] LOG     spidey.c:86   Listening on port 9898
[ 3734] DEBUG   spidey.c:87   RootPath        = /home/pbui/src/teaching/cse.20289.sp18/project.pbui/www
[ 3734] DEBUG   spidey.c:88   MimeTypesPath   = /etc/mime.types
[ 3734] DEBUG   spidey.c:89   DefaultMimeType = text/plain
[ 3734] DEBUG   spidey.c:90   ConcurrencyMode = Single
[ 3734] LOG    request.c:61   Accepted request from 10.63.12.82:45320
[ 3734] DEBUG  request.c:181  HTTP METHOD: GET
[ 3734] DEBUG  request.c:182  HTTP URI:    /
[ 3734] DEBUG  request.c:183  HTTP QUERY:
[ 3734] DEBUG  request.c:255  HTTP HEADER Host = xavier.h4x0r.space:9898
[ 3734] DEBUG  request.c:255  HTTP HEADER User-Agent = Mozilla/5.0 (X11; Linux x86_64; rv:45.0) Gecko/20100101 Firefox/45.0
[ 3734] DEBUG  request.c:255  HTTP HEADER Accept = text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
[ 3734] DEBUG  request.c:255  HTTP HEADER Accept-Language = en-US,en;q=0.5
[ 3734] DEBUG  request.c:255  HTTP HEADER Accept-Encoding = gzip, deflate
[ 3734] DEBUG  request.c:255  HTTP HEADER Connection = keep-alive
[ 3734] DEBUG  handler.c:39   HTTP REQUEST PATH: /home/pbui/src/teaching/cse.20289.sp18/project.pbui/www
[ 3734] DEBUG  handler.c:47   HTTP REQUEST TYPE: BROWSE
[ 3734] LOG    handler.c:64   HTTP REQUEST STATUS: 200 OK

Forking Connection Mode (with defaults)

$ ./spidey -c forking
[ 3764] LOG     spidey.c:86   Listening on port 9898
[ 3764] DEBUG   spidey.c:87   RootPath        = /home/pbui/src/teaching/cse.20289.sp18/project.pbui/www
[ 3764] DEBUG   spidey.c:88   MimeTypesPath   = /etc/mime.types
[ 3764] DEBUG   spidey.c:89   DefaultMimeType = text/plain
[ 3764] DEBUG   spidey.c:90   ConcurrencyMode = Forking
[ 3764] LOG    request.c:61   Accepted request from 10.63.12.82:45324
[ 3765] DEBUG  forking.c:42   Child process: 3765
[ 3765] DEBUG  request.c:181  HTTP METHOD: GET
[ 3765] DEBUG  request.c:182  HTTP URI:    /
[ 3765] DEBUG  request.c:183  HTTP QUERY:
[ 3765] DEBUG  request.c:255  HTTP HEADER Host = xavier.h4x0r.space:9898
[ 3765] DEBUG  request.c:255  HTTP HEADER User-Agent = Mozilla/5.0 (X11; Linux x86_64; rv:45.0) Gecko/20100101 Firefox/45.0
[ 3765] DEBUG  request.c:255  HTTP HEADER Accept = text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
[ 3765] DEBUG  request.c:255  HTTP HEADER Accept-Language = en-US,en;q=0.5
[ 3765] DEBUG  request.c:255  HTTP HEADER Accept-Encoding = gzip, deflate
[ 3765] DEBUG  request.c:255  HTTP HEADER Connection = keep-alive
[ 3765] DEBUG  request.c:255  HTTP HEADER Cache-Control = max-age=0
[ 3765] DEBUG  handler.c:39   HTTP REQUEST PATH: /home/pbui/src/teaching/cse.20289.sp18/project.pbui/www
[ 3765] DEBUG  handler.c:47   HTTP REQUEST TYPE: BROWSE
[ 3765] LOG    handler.c:64   HTTP REQUEST STATUS: 200 OK

Building

As usual, the Makefile contains all the rules or recipes for building the project artifacts (e.g. spidey, etc.). Although the provided Makefile contains most of the variable definitions and test recipes, you must add the appropriate rules for spidey and any intermediate objects. The dependencies for these targets are shown in the DAG below:

Makefile Variables

You must use the CC, CFLAGS, LD, and LDFLAGS variables when appropriate in your rules.

Once you have a working Makefile, you should be able to run the following commands:

# Build spidey
$ make
Compiling forking.o...
Compiling handler.o...
Compiling request.o...
Compiling single.o...
Compiling socket.o...
Compiling spidey.o...
Compiling utils.o...
Linking spidey...

# Simulate build with tracing output
$ make -n
echo Compiling forking.o...
gcc -g -gdwarf-2 -Wall -std=gnu99 -c -o forking.o forking.c
echo Compiling handler.o...
gcc -g -gdwarf-2 -Wall -std=gnu99 -c -o handler.o handler.c
echo Compiling request.o...
gcc -g -gdwarf-2 -Wall -std=gnu99 -c -o request.o request.c
echo Compiling single.o...
gcc -g -gdwarf-2 -Wall -std=gnu99 -c -o single.o single.c
echo Compiling socket.o...
gcc -g -gdwarf-2 -Wall -std=gnu99 -c -o socket.o socket.c
echo Compiling spidey.o...
gcc -g -gdwarf-2 -Wall -std=gnu99 -c -o spidey.o spidey.c
echo Compiling utils.o...
gcc -g -gdwarf-2 -Wall -std=gnu99 -c -o utils.o utils.c
echo Linking spidey...
gcc -L. -o spidey forking.o handler.o request.o single.o socket.o spidey.o utils.o

Depending on your compiler, you may see some warnings with the initial starter code.

Memory Issues

Your program must also be free of memory issues such as invalid memory accesses and memory leaks. Use valgrind to verify the correctness of your program:

$ valgrind --leak-check=full ./spidey

Be sure to check using different command line arguments as well to ensure you verify all code paths.

Sockets

To implement part 1, you must implement the socket_listen function in the socket.c source file. You may use echo_server_forking.c as inspiration.

HTTP Requests

To implement part 2, you must implement the accept_request, free_request, parse_request, parse_request_method, and parse_request_headers functions in the request.c source file. These functions are used to accept in incoming client connection and to parse the request into a struct with the following fields:

typedef struct {
    int   fd;                 /*< Client socket file descriptor */
    FILE *file;               /*< Client socket file stream */
    char *method;             /*< HTTP method */
    char *uri;                /*< HTTP URI */
    char *path;               /*< Real path corresponding to URI and RootPath */
    char *query;              /*< HTTP query string */

    char host[NI_MAXHOST];    /*< Host name of client */
    char port[NI_MAXHOST];    /*< Port number of client */

    Header *headers;          /*< List of name, value Header pairs */
} Request;

For example, suppose a client connected to an instance of spidey from 127.0.0.1:54321 and requested the URI: /script.cgi?q=monkeys using the following HTTP request:

GET /script.cgi?q=monkeys HTTP/1.0
Host: xavier.h4x0r.space:9898

The request struct would then contain the following information such as:

method      = GET
uri         = /script.cgi
path        = /tmp/spidey/www/script.cgi
query       = q=monkeys
host        = 127.0.0.1
port        = 54321

headers = [{Host: xavier.h4x0r.space:9898}]

HTTP Response

To implement part 3, you must implement handle_request, handle_browse_request, handle_file_request, and handle_error. The first function is used to analyze the request and then dispatch the appropriate handler:

  1. handle_browse_request: The requested URI is a directory, and so the server will list all the contents of the directory in lexicographical order.

  2. handle_cgi_request: The requested URI is an executable file, and so the server will execute the file as a CGI script and send the output of the script to the client.

  3. handle_file_request: The requested URI is a readable file, and so the server will simply open and read the specified file and send it to the client.

  4. handle_error: There was an error in the request or in processing the request, so send the user an hilarious and vague error message. Although HTTP has quite a few HTTP status codes, we are only concerned with a handful: 200, 400, 404, and 500.

To implement these functions, you will also need to implement the determine_mimetype, determine_request_path, and http_status_string functions in utils.c. Details about these functions can be found in the associated code skeletons.

In general, a valid HTTP response looks like this:

HTTP/1.0 200 OK
Content-Type: text/html

<html>
...
</html>

The first line contains the HTTP status code for the request. The second line instructs the client (i.e. web browser) what type of data to expect (i.e. mimetype). Each of these lines should be terminated with \r\n. Additionally, it is important that after the Content-Type line you include a blank line consisting of only \r\n. Most web clients will expect this blank line before parsing for the actual content.

CGI

To enable CGI scripts to receive input from the HTTP client, we must pass the request parameters to the CGI script via environmental variables. A list of such variables can be found here and here. For this project, we are mainly concerned with: DOCUMENT_ROOT, QUERY_STRING, REMOTE_ADDR, REMOTE_PORT, REQUEST_METHOD, REQUEST_URI, SCRIPT_FILENAME, SERVER_PORT, HTTP_HOST, HTTP_ACCEPT, HTTP_ACCEPT_LANGUAGE, HTTP_ACCEPT_ENCODING, HTTP_CONNECTION, and HTTP_USER_AGENT.

Our strategy for implementing CGI is to simply export the variables to the environment and then call popen on the script.

Two example CGI scripts are provided for you to test in the starter code directory.

Cowsay

If your cowsay.sh CGI script doesn't work, it could be because cowsay is not your PATH. To fix this, add the following to the top of cowsay.sh:

export PATH=$PATH:/afs/nd.edu/user15/pbui/pub/bin

Concurrency Modes

As noted above, spidey supports a single connection mode (the default), and a forking mode. In single mode, one connection is handled at a time and thus there is no concurrency. In forking mode, a child process is forked after accepting a request and used to handle the request, thus allowing for multiple requests to be processed at the same time.

Because of the architecture of the spidey project, we only need to implement a high-level dispatch function for each of the modes and do not need to re-write any of the handlers or request functions.

Logging

To help you debug and monitor your webserver, you should use the log and debug macros liberally to record what is happening in your server. For instance, consider the following types of events:

Some of starter code already contains log and debug calls and you should follow that example. Basically if knowing this information is useful in debugging or monitoring the system, then you should log it!

Hints

File: socket.c

File: request.c

accept_request
parse_request_method
parse_request_headers

File: handler.c

handle_request
handle_browse_request
handle_file_request
handle_cgi_request
handle_error

File: single.c

File: forking.c

File: spidey.c

File: utils.c

determine_mimetype
determine_request_path
http_status_string
skip_nonwhitespace
skip_whitespace

Error Handling

Your program should check if system calls or functions fail and handle those situations accordingly.


Activity 3: Experiments

Once you have completed both thor.py and spidey, you are to conduct expermients that answer the following questions:

  1. What is the average latency of spidey in single connection vs forking mode for: directory listings, static files, CGI scripts?

  2. What is the average throughput of spidey in single connection vs forking mode for: small static files (1KB), medium static files (1MB), and large static files (1GB)?

For each question, you must determine how you want to explore the question and how you wish to use thor.py to test spidey.

You should create shell scripts to automate running your experiments multiple times to generate a reasonable amount of data.

Different Machines

To get reasonable times, be sure that you have the client and server running on different machines.


Guru Point: Web Programming (1 Point)

For extra credit, you are to extend your spidey.c HTTP server to do any of the following:

  1. Use Bootstrap to make directory listings and error pages more visually pleasing.

  2. Display thumbnails for images in directory listings.

  3. Write a guest book CGI script that allows users to add entries to a running message board.

  4. Write a survey or personality test CGI script.

  5. Write a multiple-choice quiz CGI script of the topic of your choice.

To receive credit, you must show a TA or the instructor your code and a demonstration of it in action.

Guru Point: Virtual Private Server (1 Point)

For extra extra credit, you are to sign up for virtual private server on a service such as Digital Ocean, Linode, Amazon Web Services, Microsoft Azure, Google Cloud, or Vultr and run your spidey HTTP server from that server.

To receive credit, you must have the spidey run in the background as a service on your virtual private server and then show a TA or the instructor your spidey running in the cloud.


Grading

Your project will be graded on the following metrics:

Metric Points
Source Code
  1. General
    • Builds and cleans without warnings or errors
    • Manages resources such as memory and files appropriately
    • Uses system calls appropriately
    • Is consistent, readable, and organized
  2. Thor
    • Usage
    • Verbose
    • Single Process
    • Single Process, Multiple Requests
    • Multiple Process
    • Multiple Process, Multiple Requests
  3. Spidey
    • Usage
    • Parsing command line arguments
    • Accepting Requests
    • Parsing Requests
    • Handle Browse Requests
    • Handle File Requests
    • Handle CGI Requests
    • Handle Errors
    • Determining Mimetype
    • Single Server
    • Forking Server
16.00
  1. 3.00
    • 0.50
    • 1.00
    • 1.00
    • 0.50
  2. 2.50
    • 0.25
    • 0.25
    • 0.50
    • 0.50
    • 0.50
    • 0.50
  3. 10.50
    • 0.25
    • 0.50
    • 1.00
    • 1.25
    • 1.50
    • 1.00
    • 1.50
    • 0.50
    • 1.00
    • 0.50
    • 1.50
Demonstration
  1. Slides
  2. Experiments - Latency
  3. Experiments - Throughput
  4. Analysis
  5. Summary
3.50
  1. 0.50
  2. 1.00
  3. 1.00
  4. 0.50
  5. 0.50
Documentation
  1. Individual Contributions
0.50
  1. 0.50

Demonstration

As part of your grade, you will need to present your thor.py and spidey to a TA where you will demonstrate the correctness of your programs and discuss the results of your experiments.

Scripts

To demonstrate your project, you will be asked to download and execute the following scripts:

# Download thor test script
$ curl -LO https://gitlab.com/nd-cse-20289-sp18/cse-20289-sp18-project/raw/master/test_thor.sh

$ chmod +x test_thor.sh

# Execute thor test script
$ ./test_thor.sh
Testing thor.py...
 Functions                                                        ... Success

 Usage
     no arguments                                                 ... Success
     bad arguments                                                ... Success
     -h                                                           ... Success

 Single Process
     https://example.com                                          ... Success
     https://example.com (-v)                                     ... Success
     https://yld.me                                               ... Success
     https://yld.me (-v)                                          ... Success
     https://yld.me/izE?raw=1                                     ... Success
     https://yld.me/izE?raw=1 (-v)                                ... Success

 Single Process, Multiple Requests
     https://example.com (-r 4)                                   ... Success
     https://example.com (-r 4 -v)                                ... Success
     https://yld.me (-r 4)                                        ... Success
     https://yld.me (-r 4 -v)                                     ... Success
     https://yld.me/izE?raw=1 (-r 4)                              ... Success
     https://yld.me/izE?raw=1 (-r 4 -v)                           ... Success

 Multiple Processes
     https://example.com (-p 2)                                   ... Success
     https://example.com (-p 2 -v)                                ... Success
     https://yld.me (-p 2)                                        ... Success
     https://yld.me (-p 2 -v)                                     ... Success
     https://yld.me/izE?raw=1 (-p 2)                              ... Success
     https://yld.me/izE?raw=1 (-p 2 -v)                           ... Success

 Multiple Processes, Multiple Requests
     https://example.com (-p 2 -r 4)                              ... Success
     https://example.com (-p 2 -r 4 -v)                           ... Success
     https://yld.me (-p 2 -r 4)                                   ... Success
     https://yld.me (-p 2 -r 4 -v)                                ... Success
     https://yld.me/izE?raw=1 (-p 2 -r 4)                         ... Success
     https://yld.me/izE?raw=1 (-p 2 -r 4 -v)                      ... Success

Inconsistent Failures

Because the test_thor.sh script hits real servers, it is quite possible that some requests might fail, which will then throw off the test script.

For this project, you are not required to handle these failures in any particular way, so just be aware that if the test_thor.sh script inconsistently reports failures (particularly the multiprocessing cases), then it might have to do with the response from the servers and not your own script.

As long as you can demonstrate at least one successful run of the test_thor.sh script, you will be given full credit.

# Download spidey test script
$ curl -LO https://gitlab.com/nd-cse-20289-sp18/cse-20289-sp18-project/raw/master/test_spidey.sh

$ chmod +x test_spidey.sh

# Execute spidey test script
$ ./test_spidey.sh
 ______________________________________________________________________
/ On another machine, please run:                                      \
|                                                                      |
| valgrind --leak-check=full ./spidey -r ~pbui/pub/www -p PORT -c MODE |
|                                                                      |
| - Where PORT is a number between 9000 - 9999                         |
|                                                                      |
\ - Where MODE is either single or forking                             /
 ----------------------------------------------------------------------
        \   ^__^
         \  (oo)\_______
            (__)\       )\/\
                ||----w |
                ||     ||

Server Host: xavier.h4x0r.space
Server Port: 9898

Testing spidey server on xavier.h4x0r.space:9898 ...

 Handle Browse Requests                                           ...
     /                                                            ... Success
     /html                                                        ... Success
     /scripts                                                     ... Success
     /text                                                        ... Success

 Handle File Requests                                             ...
     /html/index.html                                             ... Success
     /text/hackers.txt                                            ... Success
     /text/lyrics.txt                                             ... Success
     /song.txt                                                    ... Success

 Handle CGI Requests                                              ...
     /scripts/env.sh                                              ... Success
     /scripts/cowsay.sh                                           ... Success
     /scripts/cowsay.sh?message=hi                                ... Success
     /scripts/cowsay.sh?message=hi&template=vader                 ... Success

 Handle Errors                                                    ...
     /asdf                                                        ... Success
     Bad Request                                                  ... Success
     Bad Headers                                                  ... Success

You will execute the test_spidey.sh script in both single and forking mode.

Testing Spidey

Note, with the test_spidey.sh script above, you want to make sure you enter the host where your spidey is running (not xavier.h4x0r.space).

If you wish to test individual URLs, remember that you can always use curl:

$ curl -v http://HOST:PORT

Where HOST and PORT are the hostname and port number of your spidey server.

Presentation

As part of your demonstration, you must provide a presentation (between 5 - 10 slides) with the following content:

  1. Summarize what was accomplished in the project (ie. what worked and what didn't work) and how the work was divided among the group members.

  2. Describe how your group went about measuring the average latency of the different types of requests. You must have a table or diagram produced by your experiments that you analyze and explain.

  3. Describe how your group went about measuring the average throughput of the different file sizes. You must have a table or diagram produced by your experiments that you analyze and explain.

  4. Discuss the results of your experiments and explain why you received the results you did. What are the advantages and disadvantages to the forking model?

  5. Wrap up the report by describing what you learned from not only the experiments but also the lab assignment as a whole.

Note, you must incorporate images, graphs, diagrams and other visual elements as part of your presentation where reasonable.

Please upload these slides to Google Drive and place a link in your README.md.

Be prepared to be asked about different aspects of your project, as the TA may ask you questions to probe your understanding of the material and your work.

Documentation

As noted above, the project repository includes a README.md file with the following sections:

You must complete this document report as part of your project.

Deliverables

To submit your assignment, please commit your work your project repository on GitLab. Your project folder should only contain at the following files:

Note: You must submit your project deliverables and perform the demonstration before Noon, Saturday May 5, 2018.