aboutsummaryrefslogtreecommitdiff
path: root/src/client/ps/generate.lisp
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)))))))))))