TrafficSpy
Abstract
In this project you’ll use Ruby, Sinatra, and ActiveRecord to build a web traffic tracking and analysis tool.
Your application will be live on the Internet. It will recieve data over HTTP from a simulation engine. The simulator will construct and transmit HTTP requests which include tracking data.
Imagine that I run a commercial website and embed JavaScript code which gets activated each time a page is viewed on my site. That JavaScript captures information about the visitor and the page they’re viewing then, in the background, submits that data to your TrafficSpy application.
Your application parses and stores that data.
Later, I visit your site and can view data about my traffic through a HTML interface.
Learning Goals
- Understand how web traffic works
- Dig into HTTP concepts including headers, referrers, and payload
- Design a normalized SQL-based database structure
- Use ActiveRecord to interface with the database from Ruby
- Practice fundamental database storage and retrieval
- Understand and practice HTTP verbs including GET, PUT, and POST
- Practice using fundamental HTML and CSS to create a usable web interface
Getting Started
Clone the Project
- One team member forks the skeleton repository at https://github.com/turingschool-examples/traffic-spy-skeleton
- Add the other team members as collaborators
Requirements
The project must use:
You’ll want to set up the DatabaseCleaner gem in order to have a clean database each time you run tests. Follow the instructions for setting up the gem. Due to a bug in the most recent version of the gem, you’ll need to use this line when you set the strategy in your test helper file:
1
|
|
Want to read more about the bug? Click here.
You will also probably want to set up a rake task to run your tests.
See the "Resources" section at the bottom of this page for additional helpful documentation.
Before starting, make sure you have the Postgres App installed.
Restrictions
The project may not use:
Rails
Base Expectations
Application Registration
To register with your application, the client will submit a request to:
1
http://yourapplication:port/sources
Parameters:
- identifier
- rootUrl
Example Request:
Terminal
$ |
|
cURL is a command-line tool for sending data using URL syntax.
Wondering what -i
and -d
mean? Type man curl
in your Terminal to see the cURL docs!
Possible Responses:
- Missing Parameters - 400 Bad Request
If missing any of the required parameters return status 400 Bad Request
with
a descriptive error message.
Wondering how to send back a status code from a Sinatra app? Check out the Sinatra docs.
- Identifier Already Exists - 403 Forbidden
If that identifier already exists return status 403 Forbidden
with a
descriptive error message.
- Success - 200 OK
When the request contains all the required parameters return status 200 OK
with the following data for the client:
1
|
|
- identifier - the unique identitier for the application that has been created for the client.
You can use ActiveRecord’s validations feature.
Processing Requests
A registered application will send POST
requests to the following URL:
1
|
|
Where IDENTIFIER
is the unique identifier generated previously for this site.
The request will contain a parameter named ‘payload’ which contains JSON data with this structure:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
|
Find the Ruby JSON docs here.
Your application should extract, analyze, and store all the content in the payload based on the view requirements defined below.
Example Request:
Terminal
|
Possible Outcomes:
Missing Payload - 400 Bad Request
If the payload is missing return status 400 Bad Request
with a descriptive
error message.
Already Received Request - 403 Forbidden
If the request payload has already been received return status 403 Forbidden
with a descriptive error message.
Application Not Registered - 403 Forbidden
When data is submitted to an application URL that does not exist, return a 403 Forbidden
with a descriptive error message.
Success - 200 OK
When the request contains a unique payload return status 200 OK
.
Viewing Data & Statistics
Application Details
A client is able to view aggregate site data at the following address:
1
|
|
When an identifer exists return a page that displays the following:
- Most requested URLS to least requested URLS (url)
- Web browser breakdown across all requests (userAgent)
- OS breakdown across all requests (userAgent)
- Screen Resolution across all requests (resolutionWidth x resolutionHeight)
- Longest, average response time per URL to shortest, average response time per URL
- Hyperlinks of each url to view url specific data
- Hyperlink to view aggregate event data
When an identifier does not exist return a page that displays the following:
- Message that the identifier does not exist
Application URL Statistics
A client is able to view URL specific data at the following address:
1 2 3 4 5 6 7 |
|
When the url for the identifier does exists:
- Longest response time
- Shortest response time
- Average response time
- Which HTTP verbs have been used
- Most popular referrrers
- Most popular user agents
When the url for the identifier does not exist:
- Message that the url has not been requested
Application Events Index
A client is able to view aggregate event data at the following address:
1
|
|
When events have been defined:
- Most received event to least received event
- Hyperlinks of each event to view event specific data
When no events have been defined:
- Message that no events have been defined
Application Event Details
A client is able to view event specific data at the following address:
1 2 3 4 5 6 7 |
|
When the event has been defined:
- Hour by hour breakdown of when the event was received. How many were shown at noon? at 1pm? at 2pm? Do it for all 24 hours.
- How many times it was recieved overall
When the event has not been defined:
- Message that no event with the given name has been defined
- Hyperlink to the Application Events Index
Extension: Campaigns
A client wants to aggregate event information into campaigns to better visualize AB testing and successes through a series of events.
Registering a Campaign
To register a campaign with your application, the client will submit a POST
request to:
1
http://yourapplication:port/sources/IDENTIFIER/campaigns
Parameters:
- ‘campaignName’
- ‘eventNames’
Example AB Test Campaign:
Terminal
$ |
|
Example AB Test Campaign:
Terminal
$ |
|
Possible Outcomes:
- Missing Parameters - 400 Bad Request
If missing any of the required parameters return status 400 Bad Request
with
a descriptive error message.
- Campaign Already Exists - 403 Forbidden
If that campaign name already exists return status 403 Forbidden
with a
descriptive error message.
- Success - 200 OK
When the request contains all the required parameters return status 200 OK
Viewing a Campaign
Application Campaigns Index
A client is able to view all defined campaigns at the following address:
1
|
|
When campaigns exist:
- Hyperlinks of each campaign to view campaign specific data
When there are no campaigns defined:
- A message that states no campaigns have been defined
Application Campaign Details
A client is able to view campaign specific data at the following address:
1
|
|
When the campaign exists:
- Most received event to least receieved event
- Hyperlinks of each event to view event specific data
When the campaign does not exist:
- message that states no campaign with that name exists
- Hyperlink back to the Application Campaigns Index
Extension: Provide a JSON API
Provide JSON API endpoints to your site.
- http://localhost:9393/sources/IDENTIFIER.json
- http://localhost:9393/sources/IDENTIFIER/urls.json
- http://localhost:9393/sources/IDENTIFIER/events.json
- http://localhost:9393/sources/IDENTIFIER/campaigns.json
The content of the json response should be comparable set of information that a user sees on the equivalent HTML page.
Extension: Dynamic Display
Presuming that a user is likely to keep statistics pages open for a long time, use JavaScript to dynamically update the displayed statistics and rankings without a full-page refresh.
Extension: HTTP Authentication
Use HTTP Auth to "secure" the data dashboard.
Resources
Evaluation Rubric
The project will be assessed with the following rubric:
1. Functional Expectations
- 4: Application fulfills base expectations and adds two extensions
- 3: Application fulfills base expectations
- 2: Application has some small missing base functionality
- 1: Application is not usable
2. Test-Driven Development
- 4: Application is broken into components which are well tested in both isolation and integration using appropriate data
- 3: Application is well tested but does not balance isolation and integration/feature tests
- 2: Application makes some use of tests, but the coverage is insufficient
- 1: Application does not demonstrate strong use of TDD
3. Encapsulation / Breaking Logic into Components
- 4: Application is expertly divided into logical components each with a clear, single responsibility
- 3: Application effectively breaks logical components apart but breaks the principle of SRP
- 2: Application shows some effort to break logic into components, but the divisions are inconsistent or unclear
- 1: Application logic shows poor decomposition with too much logic mashed together
4. Fundamental Ruby & Style
- 4: Application demonstrates excellent knowledge of Ruby syntax, style, and refactoring
- 3: Application shows strong effort towards organization, content, and refactoring
- 2: Application runs but the code has long methods, unnecessary or poorly named variables, and needs significant refactoring
- 1: Application generates syntax error or crashes during execution
5. Sinatra / Web and Business Logic
- 4: Application takes advantage of all the features Sinatra has to offer and effectively separates the web application from the business logic.
- 3: Application makes good use of Sinatra but has some mixing of the web and business logic.
- 2: Application has web and business logic totally mixed together
- 1: Application demonstrates a weak understanding of Sinatra and how applications should be built.
6. View Layer
- 4: Application expertly breaks components out to view partials and makes use of both built-in and custom-written view helpers.
- 3: Application breaks components out to view partials but has some logic or complexity leaking into the view
- 2: Application has messy views that mix logic and presentation
- 1: Application shows a lack of understanding around view templates and how they should be used/constructed.