blob: 0b1b458b88236f57a09994d07cd55770003c3ef0 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
|
(defpackage #:weekend.client.ps
(:use #:cl)
(:import-from #:flatbind #:do>)
(:local-nicknames
(#:wknd #:weekend)
(#:a #:alexandria-2))
(:export #:generate))
(in-package #:weekend.client.ps)
;; this function largely mirrors the implementation of construct-route-builder
(defun generate (class &key alt-name (content-type "application/json"))
(when (symbolp class) (setf class (find-class class)))
(let*
((args
(mapcar (a:compose #'intern #'string)
(wknd:class-initargs class)))
(parts
(loop
:for part :in (wknd:route-builder-parts class)
:when (or (stringp part) (characterp part))
:collect part
:else
:collect (intern (string part))))
(method
(wknd:request-method class))
(body
(when (wknd:body-expected-p method)
`(ps:chain -J-S-O-N
(stringify
(ps:create
,@(loop :for arg :in args
:collect arg :collect arg))))))
(url-param-names
(unless (wknd:body-expected-p method)
(set-difference args parts :test 'eq)))
(url-params
(when url-param-names
(cons "?"
(loop :for (param . more) :on url-param-names
:collect (string param)
:collect "="
:collect param
:when more
:collect "&")))))
`(defun ,(or alt-name (class-name class)) ,args
(ps:chain
(fetch (+ "" ,@parts ,@url-params)
(ps:create
method ,(string method)
cache "no-cache"
,@(when (wknd:body-expected-p method)
`(headers (ps:create "Content-Type" ,content-type)))
redirect "follow"
,@(when body (list 'body body))))
(then
(lambda (resp)
(and resp
(= 200 (ps:@ resp status))
,(cond ((string= content-type "application/json")
`(ps:chain resp (json)))
((string= content-type "text/" :end1 5)
`(ps:chain resp (text)))
(t
`(ps:chain resp (blob)))))))))))
|