api: Extract common Notion functionality to library.

This commit is contained in:
Jonathan Bernard 2021-10-30 22:36:41 -05:00
parent 3553ce4ea1
commit 06d2b3f384
8 changed files with 51 additions and 30 deletions

View File

@ -18,4 +18,7 @@ requires "https://git.jdb-software.com/jdb/nim-time-utils.git >= 0.5.0"
requires "https://git.jdb-software.com/jdb/update-nim-package-version" requires "https://git.jdb-software.com/jdb/update-nim-package-version"
task updateVersion, "Update the version of this package.": task updateVersion, "Update the version of this package.":
exec "update_nim_package_version hff_entry_forms_api 'src/hff_entry_forms_apipkg/version.nim'" exec "update_nim_package_version hff_entry_forms_api 'src/hff_entry_forms_apipkg/version.nim'"
task unittest, "Runs the unit test suite.":
exec "nim c -r test/runner"

View File

@ -42,7 +42,7 @@ Usage:
Options: Options:
-C, --config <cfgFile> Location of the config file to use (defaults to -c, --config <cfgFile> Location of the config file to use (defaults to
hff_entry_forms_api.config.json) hff_entry_forms_api.config.json)
-d, --debug Log debugging information. -d, --debug Log debugging information.
@ -52,7 +52,8 @@ Options:
let consoleLogger = newConsoleLogger( let consoleLogger = newConsoleLogger(
levelThreshold=lvlInfo, levelThreshold=lvlInfo,
fmtStr="$app - $levelname: ") fmtStr="$appname - $levelname: ",
useStderr=true)
logging.addHandler(consoleLogger) logging.addHandler(consoleLogger)
# Initialize our service context # Initialize our service context

View File

@ -1,4 +1,7 @@
import json, times, timeutils import std/json, std/times
import timeutils
import ../../../../notion_utils/src/notion_utils
type type
EventProposal* = object EventProposal* = object
@ -26,14 +29,6 @@ proc getOrFail(n: JsonNode, key: string): JsonNode =
proc parseIso8601(n: JsonNode, key: string): DateTime = proc parseIso8601(n: JsonNode, key: string): DateTime =
return parseIso8601(n.getOrFail(key).getStr) return parseIso8601(n.getOrFail(key).getStr)
proc textProp(value: string): JsonNode =
return %*[
{
"type": "text",
"text": { "content": value }
}
]
proc parseEventProposal*(n: JsonNode): EventProposal {.raises: [JsonParsingError].} = proc parseEventProposal*(n: JsonNode): EventProposal {.raises: [JsonParsingError].} =
try: try:
@ -52,33 +47,33 @@ proc parseEventProposal*(n: JsonNode): EventProposal {.raises: [JsonParsingError
proc asNotionPage*(ep: EventProposal): JsonNode = proc asNotionPage*(ep: EventProposal): JsonNode =
result = %*{ result = %*{
"properties": { "properties": {
"Event": { "title": textProp(ep.name) }, "Event": makeTextProp("title", ep.name),
"Date": { "date": { "start": formatIso8601(ep.date) } }, "Date": { "date": { "start": formatIso8601(ep.date) } },
"Department": { "multi_select": [ { "name": ep.department } ] }, "Department": { "multi_select": [ { "name": ep.department } ] },
"Location": { "rich_text": textProp(ep.location) }, "Location": makeTextProp("rich_text", ep.location),
"Owner": { "rich_text": textProp(ep.owner) }, "Owner": makeTextProp("rich_text", ep.owner),
"State": { "select": { "name": "Proposed" } } "State": { "select": { "name": "Proposed" } }
}, },
"children": [ "children": [
{ {
"object": "block", "object": "block",
"type": "heading_2", "type": "heading_2",
"heading_2": { "text": textProp("Purpose") } "heading_2": makeTextProp("text", "Purpose")
}, },
{ {
"object": "block", "object": "block",
"type": "paragraph", "type": "paragraph",
"paragraph": { "text": textProp(ep.purpose) } "paragraph": makeTextProp("text", ep.purpose)
}, },
{ {
"object": "block", "object": "block",
"type": "heading_2", "type": "heading_2",
"heading_2": { "text": textProp("Description") } "heading_2": makeTextProp("text", "Description")
}, },
{ {
"object": "block", "object": "block",
"type": "paragraph", "type": "paragraph",
"paragraph": { "text": textProp(ep.description) } "paragraph": makeTextProp("text", ep.description)
} }
] ]
} }

View File

@ -1,18 +1,12 @@
import json, logging, std/httpclient, sequtils, strutils import json, logging, std/httpclient, sequtils, strutils
import ../../../../notion_utils/src/notion_utils
import ./models, ./service import ./models, ./service
proc makeHttpClient(cfg: HffEntryFormsApiConfig): HttpClient =
let headers = newHttpHeaders([
("Content-Type", "application/json"),
("Authorization", "Bearer " & cfg.integrationToken),
("Notion-Version", cfg.notionVersion)
], true)
debug $headers
return newHttpClient(headers = headers, )
proc getEventProposalConfig*(cfg: HffEntryFormsApiConfig): EventProposalConfig = proc getEventProposalConfig*(cfg: HffEntryFormsApiConfig): EventProposalConfig =
let http = makeHttpClient(cfg) let http = newNotionClient(
apiVersion = cfg.notionVersion,
integrationToken = cfg.integrationToken)
let apiResp = http.get(cfg.notionApiBaseUrl & "/databases/" & cfg.eventParentId) let apiResp = http.get(cfg.notionApiBaseUrl & "/databases/" & cfg.eventParentId)
debug apiResp.status debug apiResp.status
@ -37,7 +31,10 @@ proc getEventProposalConfig*(cfg: HffEntryFormsApiConfig): EventProposalConfig =
) )
proc createProposedEvent*(cfg: HffEntryFormsApiConfig, ep: EventProposal): bool = proc createProposedEvent*(cfg: HffEntryFormsApiConfig, ep: EventProposal): bool =
let http = makeHttpClient(cfg) let http = newNotionClient(
apiVersion = cfg.notionVersion,
integrationToken = cfg.integrationToken)
let epNotionPage = ep.asNotionPage let epNotionPage = ep.asNotionPage
epNotionPage["parent"] = %*{ "database_id": cfg.eventParentId } epNotionPage["parent"] = %*{ "database_id": cfg.eventParentId }

2
api/test/config.nims Normal file
View File

@ -0,0 +1,2 @@
switch("path", "../src")
switch("verbosity", "0")

BIN
api/test/runner Executable file

Binary file not shown.

3
api/test/runner.nim Normal file
View File

@ -0,0 +1,3 @@
import std/unittest
import ./tmodels

20
api/test/tmodels.nim Normal file
View File

@ -0,0 +1,20 @@
import std/json, std/times, std/unittest
import hff_entry_forms_apipkg/models
suite "models":
test "asNotionPage(EventProposal)":
let ep = EventProposal(
name: "Test Event",
description: "A test event.",
purpose: "Event example for unit testing.",
department: "Testing",
location: "Hope Family Fellowship",
owner: "Jonathan Bernard",
date: parse("2021-10-30", "YYYY-MM-dd"),
budgetInDollars: 56)
let expectedJson = """{"properties":{"Event":{"title":[{"type":"text","text":{"content":"Test Event"}}]},"Date":{"date":{"start":"2021-10-30T00:00:00.000-05:00"}},"Department":{"multi_select":[{"name":"Testing"}]},"Location":{"rich_text":[{"type":"text","text":{"content":"Hope Family Fellowship"}}]},"Owner":{"rich_text":[{"type":"text","text":{"content":"Jonathan Bernard"}}]},"State":{"select":{"name":"Proposed"}}},"children":[{"object":"block","type":"heading_2","heading_2":{"text":[{"type":"text","text":{"content":"Purpose"}}]}},{"object":"block","type":"paragraph","paragraph":{"text":[{"type":"text","text":{"content":"Event example for unit testing."}}]}},{"object":"block","type":"heading_2","heading_2":{"text":[{"type":"text","text":{"content":"Description"}}]}},{"object":"block","type":"paragraph","paragraph":{"text":[{"type":"text","text":{"content":"A test event."}}]}}]}"""
check $(ep.asNotionPage) == expectedJson