Maps Back End

Cover Page

DUE Wed, 04/02, 2 pm

As when we added support for images in the previous lab, we must prepare the Chatter back end to support geodata.

Install updates

Remember to install updates available to your Ubuntu back end. If N in the following notice you see when you ssh to your back-end server is not 0,

N updates can be applied immediately.

run the following:

server$ sudo apt update
server$ sudo apt upgrade

If you see *** System restart required *** when you ssh to your server, please run:

server$ sync
server$ sudo reboot

Your ssh session will be ended at the server. Wait a few minutes for the system to reboot before you ssh to your server again.

Failure to update your packages could lead to your lab back end **not performing correctly** and also make you vulnerable to security hacks.

Modified Chatter API data formats

In this lab, we will add the user’s geodata, consisting of their geolocation and velocity (facing and speed) at the time of posting, to a chatt.

As in previous labs, the chatts retrieval API will send back all accumulated chatts in the form of a JSON array of string arrays. In addition to the three elements “username”, “message”, “timestamp” of the first lab, each string array now carries an additional element which is itself an array containing the user’s geodata (in order): latitude (lat), longitue (lon), location (corresponding to the lat/lon), compass point facing, and speed.

[
  ["username0", "message0", "id0", "timestamp0", "[lat0, lon0, \"loc0\", \"facing0\", \"speed0\"]"],
  ["username1", "message1", "id1", "timestamp1", "[lat1, lon1, \"loc1\", \"facing1\", \"speed1\"]"], 
  ... 
]

To post a chatt, the client correspondingly sends a JSON object with keys “username”, “message”, and “geodata”, where the value for key “geodata” conforms to the format above. For example:

{	
   "username": "YOUR_UNIQNAME",	
   "message": "Hello world!",	
   "geodata": "[42.29, -83.72, \"Ann Arbor\", \"South\", \"walking\"]"
}

Notice that both the “facing” and “speed” elements are descriptive, as we’ll explain below.

Database table

Add one new column named geodata of type text to the chatts table in your chatterdb database. If you’re uncertain as to how to add a column to a PostgreSQL database table, please review the Audio and Chatter labs’ back-end specs.

Web server

We now add new URL paths and their corresponding handlers to your back-end server.

server$ cd ~/441/chatterd

Go with Echo

Go with Echo

Editing handlers.go

Add one new property to the Chatt struct in handlers.go:

type Chatt struct {
    // . . .
    Geodata  *string    `json:"geodata"`
}

As with Audio data in the previous lab, Geodata is just a string to the back-end server, set to type *string to allow for nil.

Make a copy of your postaudio() in handlers.go and call it postmaps(). In the newly copied postmaps(), find the one occurence of Audio and replace it with Geodata.

Next, make a copy of your getaudio() inside your handlers.go and name the copy getmaps(). In the newly copied getmaps(), find all two occurences of Audio and replace each with Geodata.

Save and exit handlers.go.

Routing for new URLs

For the newly added getmaps() and postmaps() functions, add the following new routes to the routes array in main.go:

var routes = []Route {
    // . . .
    {"GET", "/getmaps/", getmaps},
    {"POST", "/postmaps/", postmaps},
}

Save and exit main.go.

:point_right:Go is a compiled language, like C/C++ and unlike Python, which is an interpreted language. This means you must run go build each and every time you made changes to your code, for the changes to show up in your executable.

Rebuild and restart chatterd:

server$ go build
server$ sudo systemctl restart chatterd

JavaScript with Express

JavaScript with Express

Editing handlers.ts

Add one new property to the Chatt struct in handlers.ts:

interface Chatt {
    // . . .
    geodata: string | null

As with the audio data in the previous lab, geodata is just a string to the back-end server.

Make a copy of your postaudio() in handlers.ts and call it postmaps(). In the newly copied postmaps(), find the two occurences of audio and replace them both with geodata.

Next, make a copy of your getaudio() in your handlers.ts and name the copy getmaps(). In the newly copied getmaps(), find the one occurence of audio and replace it with geodata.

Save and exit handlers.ts.

Routing for new URLs

For the newly added getmaps() and postmaps() functions, add the following new routes to the Express app instantiation in main.ts:

      .get('/getmaps/', handlers.getmaps)
      .post('/postmaps/', handlers.postmaps)

Save and exit main.ts. Rebuild and restart chatterd:

server$ npx tsc
server$ sudo systemctl restart chatterd

Python with Starlette

Python with Starlette

Editing handlers.py

Add one new property to the Chatt struct in handlers.py:

class Chatt:
    // . . .
    geodata: Optional[str] = None

As with the audio data in the previous lab, geodata is just a string to the back-end server.

Make a copy of your postaudio() in handlers.py and call it postmaps(). In the newly copied postmaps(), find the two occurences of audio and replace them both with geodata.

Next, make a copy of your getaudio() in your handlers.py and name the copy getmaps(). In the newly copied getmaps(), find the one occurence of audio and replace it with geodata.

Save and exit handlers.py.

Routing for new urls

For the newly added getmaps() and postmaps() functions, add the following new routes to the routes array in main.py:

    Route('/getmaps/', handlers.getmaps, methods=['GET']),
    Route('/postmaps/', handlers.postmaps, methods=['POST']),

Save and exit main.py and restart chatterd:

server$ sudo systemctl restart chatterd

Rust with axum

Rust with axum

Editing handlers.rs

Add one new property to the Chatt struct in handlers.rs:

pub struct Chatt {
    // . . .
    geodata: Option<String>,
}

As with audio in the previous lab, the audio data is just a string to the back-end server.

Make a copy of your postaudio() in handlers.rs and call it postmaps(). In the newly copied postmaps(), find the two occurences of audio and replace both with geodata.

Next, make a copy of your getaudio() inside your handlers.rs and name the copy getmaps(). In the newly copied getmaps(), find the one occurence of audio and replace it with geodata.

Save and exit handlers.rs.

Routing for new URLs

For the newly added getmaps() and postmaps() functions, add the following new routes to the Router instantiation of the router variable in main.rs, before the .layer(/*...*/) line:

        .route("/getmaps/", get(handlers::getmaps))
        .route("/postmaps/", post(handlers::postmaps))

Save and exit main.rs. Rebuild and restart chatterd:

server$ cargo build --release
server$ sudo systemctl restart chatterd

Testing geodata upload

You can test your postmaps/ API using the example JSON above. After a successful POST, your database should contain:

chatterdb=# SELECT * FROM chatts;
  username | message | id  | time | ... |                      geodata                     |
--_--------+---------+-----+------+-----+--------------------------------------------------|
 testuser1 | Hello!  | ... | ...  | ... | [42.29, -83.72, "Ann Arbor", "South", "walking"] |
(1 row)

Submitting your back end

Congratulations! You’ve completed the back end modifications necessary to support geodata!


Prepared for EECS 441 by Wendan Jiang, Ollie Elmgren, Benjamin Brengman, Mark Wassink, Alexander Wu, Yibo Pi, and Sugih Jamin Last updated: December 20th, 2024