Initial thoughts around API, DB layer.
This commit is contained in:
commit
0d2efe4bcc
BIN
api/.api.txt.swp
Normal file
BIN
api/.api.txt.swp
Normal file
Binary file not shown.
BIN
api/.personal_measure_api.nimble.swp
Normal file
BIN
api/.personal_measure_api.nimble.swp
Normal file
Binary file not shown.
25
api/Makefile
Normal file
25
api/Makefile
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
PGSQL_CONTAINER_ID=`cat postgres.container.id`
|
||||||
|
DB_NAME="personal_measure"
|
||||||
|
|
||||||
|
start-db: start-postgres
|
||||||
|
|
||||||
|
postgres.container.id:
|
||||||
|
docker run --name postgres-$(DB_NAME) -e POSTGRES_PASSWORD=password -p 5500:5432 -d postgres > postgres.container.id
|
||||||
|
sleep 5
|
||||||
|
PGPASSWORD=password psql -p 5500 -U postgres -h localhost -c "CREATE DATABASE $(DB_NAME);"
|
||||||
|
db_migrate up -c database-local.json
|
||||||
|
|
||||||
|
start-postgres: postgres.container.id
|
||||||
|
docker start $(PGSQL_CONTAINER_ID)
|
||||||
|
db_migrate up -c database-local.json
|
||||||
|
|
||||||
|
stop-postgres: postgres.container.id
|
||||||
|
docker stop $(PGSQL_CONTAINER_ID)
|
||||||
|
|
||||||
|
delete-postgres-container:
|
||||||
|
-docker stop $(PGSQL_CONTAINER_ID)
|
||||||
|
docker container rm $(PGSQL_CONTAINER_ID)
|
||||||
|
rm postgres.container.id
|
||||||
|
|
||||||
|
connect:
|
||||||
|
PGPASSWORD=password psql -p 5500 -U postgres -h localhost
|
31
api/api.txt
Normal file
31
api/api.txt
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
Personal Measure API
|
||||||
|
====================
|
||||||
|
|
||||||
|
### Measure
|
||||||
|
|
||||||
|
☐ GET /measure Get a list of all defined measures for this user.
|
||||||
|
☐ POST /measure Create a new measure (post the definition).
|
||||||
|
☐ GET /measure/<measure-slug> Get the definition for a specific measure.
|
||||||
|
☐ DELETE /measure/<measure-slug> Delete a measure (and all values associated with it).
|
||||||
|
|
||||||
|
### Values
|
||||||
|
|
||||||
|
☐ GET /<measure-slug> Get a list of values for a measure.
|
||||||
|
☐ POST /<measure-slug> Add a new value for a measure.
|
||||||
|
☐ GET /<measure-slug>/<id> Get the details for a specific value by id.
|
||||||
|
☐ PUT /<measure-slug>/<id> Update a value by id.
|
||||||
|
☐ DELETE /<measure-slug> Delete a value by id.
|
||||||
|
|
||||||
|
### Auth
|
||||||
|
|
||||||
|
☐ GET /auth-token Given a valid username/password combo, get an auth token.
|
||||||
|
☐ GET /user Given a valid auth token, return the user details.
|
||||||
|
☐ PSOT /app-token With a valid session, create a new app token.
|
||||||
|
|
||||||
|
Legend
|
||||||
|
------
|
||||||
|
|
||||||
|
☐ TODO
|
||||||
|
☑ Done
|
||||||
|
☒ Will not do
|
||||||
|
|
5
api/database-local.json
Normal file
5
api/database-local.json
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
{
|
||||||
|
"driver": "postgres",
|
||||||
|
"connectionString": "host=localhost port=5500 dbname=personal_measure user=postgres password=password",
|
||||||
|
"sqlDir": "src/main/sql/migrations"
|
||||||
|
}
|
BIN
api/personal_measure_api
Executable file
BIN
api/personal_measure_api
Executable file
Binary file not shown.
14
api/personal_measure_api.nimble
Normal file
14
api/personal_measure_api.nimble
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
# Package
|
||||||
|
|
||||||
|
version = "0.1.0"
|
||||||
|
author = "Jonathan Bernard"
|
||||||
|
description = "JDB\'s Personal Measures API"
|
||||||
|
license = "MIT"
|
||||||
|
srcDir = "src/main/nim"
|
||||||
|
bin = @["personal_measure_api"]
|
||||||
|
|
||||||
|
|
||||||
|
# Dependencies
|
||||||
|
|
||||||
|
requires @["nim >= 0.19.4", "bcrypt", "docopt >= 0.6.8", "isaac >= 0.1.3", "jester >= 0.4.1", "jwt", "tempfile",
|
||||||
|
"timeutils >= 0.4.0", "uuids >= 0.1.10" ]
|
1
api/postgres.container.id
Normal file
1
api/postgres.container.id
Normal file
@ -0,0 +1 @@
|
|||||||
|
7fcd40172b1207f53db40a9d4843ef5815588a742d8a334a767d07367281ff84
|
BIN
api/src/main/nim/.personal_measure_api.nim.swp
Normal file
BIN
api/src/main/nim/.personal_measure_api.nim.swp
Normal file
Binary file not shown.
25
api/src/main/nim/personal_measure_api.nim
Normal file
25
api/src/main/nim/personal_measure_api.nim
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
import asyncdispatch, bcrypt, docopt, jester, json, jwt
|
||||||
|
|
||||||
|
import personal_measure_apipkg/db
|
||||||
|
|
||||||
|
type
|
||||||
|
PersonalMeasureApiConfig = object
|
||||||
|
port*: int
|
||||||
|
pwdCost*: int
|
||||||
|
dbConnString*: string
|
||||||
|
|
||||||
|
proc start*(cfg: StrawBossConfig): void =
|
||||||
|
|
||||||
|
var stopFuture = newFuture[void]()
|
||||||
|
var workers: seq[Worker] = @[]
|
||||||
|
|
||||||
|
settings:
|
||||||
|
port = Port(cfg.port)
|
||||||
|
appName = "/api"
|
||||||
|
|
||||||
|
routes:
|
||||||
|
|
||||||
|
get "/version":
|
||||||
|
resp($(%("strawboss v" & SB_VERSION)), JSON)
|
||||||
|
|
||||||
|
|
BIN
api/src/main/nim/personal_measure_apipkg/.db.nim.swp
Normal file
BIN
api/src/main/nim/personal_measure_apipkg/.db.nim.swp
Normal file
Binary file not shown.
BIN
api/src/main/nim/personal_measure_apipkg/db
Executable file
BIN
api/src/main/nim/personal_measure_apipkg/db
Executable file
Binary file not shown.
170
api/src/main/nim/personal_measure_apipkg/db.nim
Normal file
170
api/src/main/nim/personal_measure_apipkg/db.nim
Normal file
@ -0,0 +1,170 @@
|
|||||||
|
import db_postgres, macros, options, postgres, sequtils, strutils, times, timeutils, uuids
|
||||||
|
|
||||||
|
import nre except toSeq
|
||||||
|
from unicode import capitalize, toLower
|
||||||
|
|
||||||
|
type
|
||||||
|
User* = object
|
||||||
|
id*: UUID
|
||||||
|
displayName*, email*, hashedPwd*: string
|
||||||
|
|
||||||
|
ApiToken* = object
|
||||||
|
id*, userId: UUID
|
||||||
|
name*, hashedToken: string
|
||||||
|
expires: Option[DateTime]
|
||||||
|
|
||||||
|
Measure* = object
|
||||||
|
id*, userId*: UUID
|
||||||
|
slug*, name*, description*, domainUnits*, rangeUnits*: string
|
||||||
|
domainSource*, rangeSource*: Option[string]
|
||||||
|
analysis*: seq[string]
|
||||||
|
|
||||||
|
Value* = object
|
||||||
|
id*, measureId*: UUID
|
||||||
|
value: int
|
||||||
|
timestamp: DateTime
|
||||||
|
extData: string
|
||||||
|
|
||||||
|
MutateClauses = object
|
||||||
|
columns*: seq[string]
|
||||||
|
placeholders*: seq[string]
|
||||||
|
values*: seq[string]
|
||||||
|
|
||||||
|
let UPPERCASE_PATTERN = re"(.)(\p{Lu})"
|
||||||
|
|
||||||
|
proc newMutateClauses(): MutateClauses =
|
||||||
|
return MutateClauses(
|
||||||
|
columns: @[],
|
||||||
|
placeholders: @[],
|
||||||
|
values: @[])
|
||||||
|
|
||||||
|
proc dbFormat(s: string): string = return s
|
||||||
|
|
||||||
|
proc dbFormat(dt: DateTime): string = return dt.formatIso8601
|
||||||
|
|
||||||
|
proc dbFormat[T](list: seq[T]): string =
|
||||||
|
return "{" & list.mapIt("'" & dbFormat(it) & "'").join(",") & "}"
|
||||||
|
|
||||||
|
proc dbFormat[T](item: T): string = return $item
|
||||||
|
|
||||||
|
macro populateMutateClauses(t: typed, newRecord: bool, mc: var MutateClauses): untyped =
|
||||||
|
|
||||||
|
# Must be working with an object.
|
||||||
|
let tTypeImpl = t.getTypeImpl
|
||||||
|
if not (tTypeImpl.typeKind == ntyObject):
|
||||||
|
error $t & " is not an object."
|
||||||
|
|
||||||
|
result = newStmtList()
|
||||||
|
|
||||||
|
# iterate over all the object's fields
|
||||||
|
for child in tTypeImpl[2].children:
|
||||||
|
|
||||||
|
# ignore AST nodes that are not field definitions
|
||||||
|
if child.kind == nnkIdentDefs:
|
||||||
|
|
||||||
|
# grab the field, it's string name, and it's type
|
||||||
|
let field = child[0]
|
||||||
|
let fieldType = child[1]
|
||||||
|
let fieldName = $field
|
||||||
|
|
||||||
|
# we do not update the ID, but we do check: if we're creating a new
|
||||||
|
# record, we should not have an existing ID
|
||||||
|
if fieldName == "id":
|
||||||
|
result.add quote do:
|
||||||
|
if `newRecord` and not `t`.id.isZero:
|
||||||
|
raise newException(
|
||||||
|
AssertionError,
|
||||||
|
"Trying to create a new record, but the record already has an ID.")
|
||||||
|
|
||||||
|
# if we're looking at an optional field, add logic to check for presence
|
||||||
|
elif fieldType.kind == nnkBracketExpr and
|
||||||
|
fieldType.len > 0 and
|
||||||
|
fieldType[0] == Option.getType:
|
||||||
|
|
||||||
|
result.add quote do:
|
||||||
|
`mc`.columns.add(`fieldName`)
|
||||||
|
if `t`.`field`.isSome:
|
||||||
|
`mc`.placeholders.add("?")
|
||||||
|
`mc`.values.add(dbFormat(`t`.`field`.get))
|
||||||
|
else:
|
||||||
|
`mc`.placeholders.add("NULL")
|
||||||
|
|
||||||
|
# otherwise assume we can convert and go ahead.
|
||||||
|
else:
|
||||||
|
result.add quote do:
|
||||||
|
`mc`.columns.add(`fieldName`)
|
||||||
|
`mc`.placeholders.add("?")
|
||||||
|
`mc`.values.add(dbFormat(`t`.`field`))
|
||||||
|
|
||||||
|
#echo result.repr
|
||||||
|
|
||||||
|
# TODO
|
||||||
|
#macro rowToRecord(recType: untyped, row: seq[string]): untyped
|
||||||
|
# echo recType
|
||||||
|
|
||||||
|
macro recordName(rec: typed): string =
|
||||||
|
return $rec.getTypeInst
|
||||||
|
|
||||||
|
proc tableNameForRecord[T](rec: T): string =
|
||||||
|
return recordName(rec).replace(UPPERCASE_PATTERN, "$1_$2").toLower() & "s"
|
||||||
|
|
||||||
|
proc createRecord[T](db: DbConn, rec: T): T =
|
||||||
|
var mc = newMutateClauses()
|
||||||
|
populateMutateClauses(rec, true, mc)
|
||||||
|
|
||||||
|
# Confusingly, getRow allows inserts and updates. We use it to get back the ID
|
||||||
|
# we want from the row.
|
||||||
|
let newIdStr = db.getValue(sql(
|
||||||
|
"INSERT INTO " & tableNameForRecord(rec) &
|
||||||
|
" (" & mc.columns.join(",") & ") " &
|
||||||
|
" VALUES (" & mc.placeholders.join(",") & ") " &
|
||||||
|
" RETURNING id"), mc.values)
|
||||||
|
|
||||||
|
result = rec
|
||||||
|
result.id = parseUUID(newIdStr)
|
||||||
|
|
||||||
|
proc updateRecord[T](db: DbConn, rec: T): bool =
|
||||||
|
var mc = newMutateClauses()
|
||||||
|
populateMutateClauses(rec, false, mc)
|
||||||
|
|
||||||
|
let setClause = zip(mc.columns, mc.placeholders).mapIt(it.a & " = " it.b).join(',')
|
||||||
|
let numRowsUpdated = db.execAffectedRows(sql(
|
||||||
|
"UPDATE " & tableNameForRecord(rec) &
|
||||||
|
" SET " & setClause &
|
||||||
|
" WHERE id = ? "), mc.values.concat(@[rec.id]))
|
||||||
|
|
||||||
|
return numRowsUpdated > 0;
|
||||||
|
|
||||||
|
# TODO
|
||||||
|
#proc getRecord[T](db: DbConn, UUID id): T =
|
||||||
|
|
||||||
|
macro listFieldNames(t: typed): untyped =
|
||||||
|
let tTypeImpl = t.getTypeImpl
|
||||||
|
|
||||||
|
if not (tTypeImpl.typeKind == ntyObject):
|
||||||
|
error $t & " is not an object."
|
||||||
|
|
||||||
|
var fieldNames: seq[tuple[n: string, t: string]] = @[]
|
||||||
|
for child in tTypeImpl[2].children:
|
||||||
|
if child.kind == nnkIdentDefs:
|
||||||
|
echo $child[1]
|
||||||
|
fieldNames.add((n: $child[0], t: $child[1]))
|
||||||
|
|
||||||
|
result = newLit(fieldNames)
|
||||||
|
|
||||||
|
|
||||||
|
proc createUser*(db: DbConn, user: User): User = db.createRecord(user)
|
||||||
|
proc createApiToken*(db: DbConn, token: ApiToken): ApiToken = db.createRecord(token)
|
||||||
|
proc createMeasure*(db: DbConn, measure: Measure): Measure = db.createRecord(measure)
|
||||||
|
proc createValue*(db: DbConn, value: Value): Value = db.createRecord(value)
|
||||||
|
|
||||||
|
#[
|
||||||
|
when isMainModule:
|
||||||
|
|
||||||
|
let u = User(
|
||||||
|
displayName: "Bob",
|
||||||
|
email: "bob@bobsco.com",
|
||||||
|
hashedPwd: "test")
|
||||||
|
|
||||||
|
echo createRecord(nil, u)
|
||||||
|
]#
|
Binary file not shown.
@ -0,0 +1,5 @@
|
|||||||
|
-- DOWN script for initial-schema (20190214122514)
|
||||||
|
drop table if exists "values";
|
||||||
|
drop table if exists "measures";
|
||||||
|
drop table if exists "api_tokens";
|
||||||
|
drop table if exists "users";
|
@ -0,0 +1,38 @@
|
|||||||
|
-- UP script for initial-schema (20190214122514)
|
||||||
|
create extension if not exists "uuid-ossp";
|
||||||
|
|
||||||
|
create table "users" (
|
||||||
|
id uuid default uuid_generate_v4() primary key,
|
||||||
|
display_name varchar not null,
|
||||||
|
email varchar not null,
|
||||||
|
hashedpwd varchar not null
|
||||||
|
);
|
||||||
|
|
||||||
|
create table "api_tokens" (
|
||||||
|
id uuid default uuid_generate_v4() primary key,
|
||||||
|
user_id uuid not null references users (id) on delete cascade on update cascade,
|
||||||
|
name varchar not null,
|
||||||
|
expires timestamp with time zone default null,
|
||||||
|
hashedToken varchar not null
|
||||||
|
);
|
||||||
|
|
||||||
|
create table "measures" (
|
||||||
|
id uuid default uuid_generate_v4() primary key,
|
||||||
|
user_id uuid not null references users (id) on delete cascade on update cascade,
|
||||||
|
slug varchar not null,
|
||||||
|
name varchar not null,
|
||||||
|
description varchar not null default '',
|
||||||
|
domain_source varchar default null ,
|
||||||
|
domain_units varchar not null default '',
|
||||||
|
range_source varchar default null,
|
||||||
|
range_units varchar not null default '',
|
||||||
|
analysis varchar[] not null default '{}'
|
||||||
|
);
|
||||||
|
|
||||||
|
create table "values" (
|
||||||
|
id uuid default uuid_generate_v4() primary key,
|
||||||
|
measure_id uuid not null references measures (id) on delete cascade on update cascade,
|
||||||
|
value integer not null,
|
||||||
|
"timestamp" timestamp not null default current_timestamp,
|
||||||
|
ext_data jsonb not null default '{}'::json
|
||||||
|
);
|
Loading…
x
Reference in New Issue
Block a user