Skip to content

Commit 18051a1

Browse files
authored
Add rest-api (#147)
1 parent ba62aa1 commit 18051a1

13 files changed

Lines changed: 730 additions & 174 deletions

File tree

bin/generate-spec

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -132,7 +132,7 @@ canonical_data = json.decode read_file(canonical_data_path), 1, json.null
132132
tests_toml_path = exercise_directory .. '/.meta/tests.toml'
133133
included_tests = included_tests_from_toml tests_toml_path
134134

135-
package.moonpath = "#{exercise_directory}/.meta/?.moon;#{package.moonpath}"
135+
package.moonpath = "#{exercise_directory}/.meta/?.moon;./lib/?.moon;#{package.moonpath}"
136136
spec_generator = require 'spec_generator'
137137

138138
local spec

config.json

Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -842,6 +842,14 @@
842842
"prerequisites": [],
843843
"difficulty": 7
844844
},
845+
{
846+
"slug": "rest-api",
847+
"name": "REST API",
848+
"uuid": "a999597a-d5db-4a72-83f7-133a2b7f3add",
849+
"practices": [],
850+
"prerequisites": [],
851+
"difficulty": 7
852+
},
845853
{
846854
"slug": "satellite",
847855
"name": "Satellite",
Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
return {
2+
default = {
3+
ROOT = { '.' }
4+
}
5+
}
Lines changed: 48 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,48 @@
1+
# Instructions
2+
3+
Implement a RESTful API for tracking IOUs.
4+
5+
Four roommates have a habit of borrowing money from each other frequently, and have trouble remembering who owes whom, and how much.
6+
7+
Your task is to implement a simple [RESTful API][restful-wikipedia] that receives [IOU][iou]s as POST requests, and can deliver specified summary information via GET requests.
8+
9+
## API Specification
10+
11+
### User object
12+
13+
```json
14+
{
15+
"name": "Adam",
16+
"owes": {
17+
"Bob": 12.0,
18+
"Chuck": 4.0,
19+
"Dan": 9.5
20+
},
21+
"owed_by": {
22+
"Bob": 6.5,
23+
"Dan": 2.75
24+
},
25+
"balance": "<(total owed by other users) - (total owed to other users)>"
26+
}
27+
```
28+
29+
### Methods
30+
31+
| Description | HTTP Method | URL | Payload Format | Response w/o Payload | Response w/ Payload |
32+
| ------------------------ | ----------- | ------ | ------------------------------------------------------------------------- | -------------------------------------- | ------------------------------------------------------------------------------- |
33+
| List of user information | GET | /users | `{"users":["Adam","Bob"]}` | `{"users":<List of all User objects>}` | `{"users":<List of User objects for <users> (sorted by name)}` |
34+
| Create user | POST | /add | `{"user":<name of new user (unique)>}` | N/A | `<User object for new user>` |
35+
| Create IOU | POST | /iou | `{"lender":<name of lender>,"borrower":<name of borrower>,"amount":5.25}` | N/A | `{"users":<updated User objects for <lender> and <borrower> (sorted by name)>}` |
36+
37+
## Other Resources
38+
39+
- [REST API Tutorial][restfulapi]
40+
- Example RESTful APIs
41+
- [GitHub][github-rest]
42+
- [Reddit][reddit-rest]
43+
44+
[restful-wikipedia]: https://en.wikipedia.org/wiki/Representational_state_transfer
45+
[iou]: https://en.wikipedia.org/wiki/IOU
46+
[github-rest]: https://docs.github.com/en/rest
47+
[reddit-rest]: https://web.archive.org/web/20231202231149/https://www.reddit.com/dev/api/
48+
[restfulapi]: https://restfulapi.net/
Lines changed: 17 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
{
2+
"authors": [
3+
"glennj"
4+
],
5+
"files": {
6+
"solution": [
7+
"rest_api.moon"
8+
],
9+
"test": [
10+
"rest_api_spec.moon"
11+
],
12+
"example": [
13+
".meta/example.moon"
14+
]
15+
},
16+
"blurb": "Implement a RESTful API for tracking IOUs."
17+
}
Lines changed: 63 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,63 @@
1+
contains = (list, elem) ->
2+
for item in *list
3+
return true if item == elem
4+
false
5+
6+
sort_users = (list) ->
7+
table.sort list, (a, b) -> a.name < b.name
8+
list
9+
10+
-- The tests do not require the payload to be validated.
11+
-- Also, I'm not concerned about accepting references to user's tables,
12+
-- or returning references to internal tables.
13+
14+
class RestApi
15+
new: (@db) =>
16+
17+
GET: (url, payload) => @route 'GET', url, payload
18+
POST: (url, payload) => @route 'POST', url, payload
19+
20+
route: (method, url, payload) =>
21+
subject = url\match '^/(%w+)'
22+
func = "#{subject}_#{method}"
23+
return {error: "Unknown endpoint: #{method} #{url}"} if not @[func]
24+
@[func] @, payload
25+
26+
users_GET: (payload) =>
27+
filter = if payload
28+
(user) -> contains payload.users, user.name
29+
else
30+
(user) -> true
31+
{users: sort_users [u for u in *@db.users when filter u]}
32+
33+
add_POST: (payload) =>
34+
user = {
35+
name: payload.user
36+
balance: 0
37+
owes: {}
38+
owed_by: {}
39+
}
40+
table.insert @db.users, user
41+
user
42+
43+
iou_POST: (payload) =>
44+
{users: {lender}} = @users_GET {users: {payload.lender}}
45+
{users: {borrower}} = @users_GET {users: {payload.borrower}}
46+
borrower.balance -= payload.amount
47+
lender.balance += payload.amount
48+
49+
debt = (lender.owes[borrower.name] or 0) - (lender.owed_by[borrower.name] or 0) - payload.amount
50+
51+
lender.owes[borrower.name] = nil
52+
lender.owed_by[borrower.name] = nil
53+
borrower.owes[lender.name] = nil
54+
borrower.owed_by[lender.name] = nil
55+
56+
if debt > 0
57+
lender.owes[borrower.name] = debt
58+
borrower.owed_by[lender.name] = debt
59+
elseif debt < 0
60+
lender.owed_by[borrower.name] = -debt
61+
borrower.owes[lender.name] = -debt
62+
63+
{users: sort_users {lender, borrower}}
Lines changed: 25 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
test_helpers = require 'test_helpers'
2+
import table_dump from test_helpers
3+
4+
{
5+
module_name: 'RestApi',
6+
7+
generate_test: (case, level) ->
8+
lines = {
9+
"database = #{table_dump case.input.database, level}",
10+
"api = RestApi database"
11+
}
12+
call = "result = api\\#{case.property\upper!} #{quote case.input.url}"
13+
if case.input.payload
14+
table.insert lines, "payload = #{table_dump case.input.payload, level}"
15+
call ..= ", payload"
16+
table.insert lines, call
17+
table.insert lines, "expected = #{table_dump case.expected, level}"
18+
table.insert lines, "assert.are.same expected, result"
19+
table.concat [indent line, level for line in *lines], '\n'
20+
21+
-- we have deep tables to compare, display it all when not the same
22+
test_helpers: [[
23+
assert\set_parameter "TableFormatLevel", 4
24+
]]
25+
}
Lines changed: 37 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
# This is an auto-generated file.
2+
#
3+
# Regenerating this file via `configlet sync` will:
4+
# - Recreate every `description` key/value pair
5+
# - Recreate every `reimplements` key/value pair, where they exist in problem-specifications
6+
# - Remove any `include = true` key/value pair (an omitted `include` key implies inclusion)
7+
# - Preserve any other key/value pair
8+
#
9+
# As user-added comments (using the # character) will be removed when this file
10+
# is regenerated, comments can be added via a `comment` key.
11+
12+
[5be01ffb-a814-47a8-a19f-490a5622ba07]
13+
description = "user management -> no users"
14+
15+
[382b70cc-9f6c-486d-9bee-fda2df81c803]
16+
description = "user management -> add user"
17+
18+
[d624e5e5-1abb-4f18-95b3-45d55c818dc3]
19+
description = "user management -> get single user"
20+
21+
[7a81b82c-7276-433e-8fce-29ce983a7c56]
22+
description = "iou -> both users have 0 balance"
23+
24+
[1c61f957-cf8c-48ba-9e77-b221ab068803]
25+
description = "iou -> borrower has negative balance"
26+
27+
[8a8567b3-c097-468a-9541-6bb17d5afc85]
28+
description = "iou -> lender has negative balance"
29+
30+
[29fb7c12-7099-4a85-a7c4-9c290d2dc01a]
31+
description = "iou -> lender owes borrower"
32+
33+
[ce969e70-163c-4135-a4a6-2c3a5da286f5]
34+
description = "iou -> lender owes borrower less than new loan"
35+
36+
[7f4aafd9-ae9b-4e15-a406-87a87bdf47a4]
37+
description = "iou -> lender owes borrower same as new loan"
Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
error 'Implement me'

0 commit comments

Comments
 (0)