mirror of
https://github.com/penpot/penpot.git
synced 2025-01-10 00:40:30 -05:00
185 lines
5.5 KiB
Clojure
185 lines
5.5 KiB
Clojure
;; This Source Code Form is subject to the terms of the Mozilla Public
|
|
;; License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
;; file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
;;
|
|
;; Copyright (c) UXBOX Labs SL
|
|
|
|
(ns app.http
|
|
(:require
|
|
["cookies" :as Cookies]
|
|
["http" :as http]
|
|
["inflation" :as inflate]
|
|
["raw-body" :as raw-body]
|
|
["stream" :as stream]
|
|
[app.common.logging :as l]
|
|
[app.common.transit :as t]
|
|
[app.config :as cf]
|
|
[app.handlers :as handlers]
|
|
[cuerdas.core :as str]
|
|
[lambdaisland.uri :as u]
|
|
[promesa.core :as p]))
|
|
|
|
(l/set-level! :info)
|
|
|
|
(defprotocol IStreamableResponseBody
|
|
(write-body! [_ response]))
|
|
|
|
(extend-protocol IStreamableResponseBody
|
|
string
|
|
(write-body! [data response]
|
|
(.write ^js response data)
|
|
(.end ^js response))
|
|
|
|
js/Buffer
|
|
(write-body! [data response]
|
|
(.write ^js response data)
|
|
(.end ^js response))
|
|
|
|
stream/Stream
|
|
(write-body! [data response]
|
|
(.pipe ^js data response)
|
|
(.on ^js data "error" (fn [cause]
|
|
(js/console.error cause)
|
|
(.end response)))))
|
|
|
|
(defn- handle-response
|
|
[{:keys [:response/body
|
|
:response/headers
|
|
:response/status
|
|
response]
|
|
:as exchange}]
|
|
(let [status (or status 200)
|
|
headers (clj->js headers)
|
|
body (or body "")]
|
|
(.writeHead ^js response status headers)
|
|
(write-body! body response)))
|
|
|
|
(defn- parse-headers
|
|
[req]
|
|
(let [orig (unchecked-get req "headers")]
|
|
(persistent!
|
|
(reduce #(assoc! %1 (str/lower %2) (unchecked-get orig %2))
|
|
(transient {})
|
|
(js/Object.keys orig)))))
|
|
|
|
(defn- wrap-body-params
|
|
[handler]
|
|
(let [opts #js {:limit "60mb" :encoding "utf8"}]
|
|
(fn [{:keys [:request/method :request/headers request] :as exchange}]
|
|
(let [ctype (get headers "content-type")]
|
|
(if (= method "post")
|
|
(-> (raw-body (inflate request) opts)
|
|
(p/then (fn [data]
|
|
(cond-> data
|
|
(= ctype "application/transit+json")
|
|
(t/decode-str))))
|
|
(p/then (fn [data]
|
|
(handler (assoc exchange :request/body-params data)))))
|
|
(handler exchange))))))
|
|
|
|
(defn- wrap-params
|
|
[handler]
|
|
(fn [{:keys [:request/body-params :request/query-params] :as exchange}]
|
|
(handler (assoc exchange :request/params (merge query-params body-params)))))
|
|
|
|
(defn- wrap-response-format
|
|
[handler]
|
|
(fn [exchange]
|
|
(p/then
|
|
(handler exchange)
|
|
(fn [{:keys [:response/body :response/status] :as exchange}]
|
|
(cond
|
|
(map? body)
|
|
(let [data (t/encode-str body {:type :json-verbose})]
|
|
(-> exchange
|
|
(assoc :response/body data)
|
|
(assoc :response/status 200)
|
|
(update :response/headers assoc "content-type" "application/transit+json")
|
|
(update :response/headers assoc "content-length" (count data))))
|
|
|
|
(and (nil? body)
|
|
(= 200 status))
|
|
(-> exchange
|
|
(assoc :response/body "")
|
|
(assoc :response/status 204)
|
|
(assoc :response/headers {"content-length" 0}))
|
|
|
|
:else
|
|
exchange)))))
|
|
|
|
(defn- wrap-query-params
|
|
[handler]
|
|
(fn [{:keys [:request/uri] :as exchange}]
|
|
(handler (assoc exchange :request/query-params (u/query-string->map (:query uri))))))
|
|
|
|
(defn- wrap-error
|
|
[handler on-error]
|
|
(fn [exchange]
|
|
(-> (p/do (handler exchange))
|
|
(p/catch (fn [cause] (on-error cause exchange))))))
|
|
|
|
(defn- wrap-auth
|
|
[handler cookie-name]
|
|
(fn [{:keys [:request/cookies] :as exchange}]
|
|
(let [token (.get ^js cookies cookie-name)]
|
|
(handler (cond-> exchange token (assoc :request/auth-token token))))))
|
|
|
|
(defn- wrap-health
|
|
"Add /healthz entry point intercept."
|
|
[handler]
|
|
(fn [{:keys [:request/path] :as exchange}]
|
|
(if (= path "/readyz")
|
|
(assoc exchange
|
|
:response/status 200
|
|
:response/body "OK")
|
|
(handler exchange))))
|
|
|
|
(defn- create-adapter
|
|
[handler]
|
|
(fn [req res]
|
|
(let [cookies (Cookies. req res)
|
|
headers (parse-headers req)
|
|
uri (u/uri (unchecked-get req "url"))
|
|
exchange {:request/method (str/lower (unchecked-get req "method"))
|
|
:request/path (:path uri)
|
|
:request/uri uri
|
|
:request/headers headers
|
|
:request/cookies cookies
|
|
:request req
|
|
:response res}]
|
|
(-> (p/do (handler exchange))
|
|
(p/then handle-response)))))
|
|
|
|
(defn- create-server
|
|
[handler]
|
|
(.createServer ^js http (create-adapter handler)))
|
|
|
|
(def instance (atom nil))
|
|
|
|
(defn init
|
|
[]
|
|
(let [handler (-> handlers/handler
|
|
(wrap-health)
|
|
(wrap-auth "auth-token")
|
|
(wrap-response-format)
|
|
(wrap-params)
|
|
(wrap-query-params)
|
|
(wrap-body-params)
|
|
(wrap-error handlers/on-error))
|
|
server (create-server handler)
|
|
port (cf/get :http-server-port 6061)]
|
|
(.listen server port)
|
|
(l/info :hint "welcome to penpot"
|
|
:module "exporter"
|
|
:version (:full @cf/version))
|
|
(l/info :hint "starting http server" :port port)
|
|
(reset! instance server)))
|
|
|
|
(defn stop
|
|
[]
|
|
(if-let [server @instance]
|
|
(p/create (fn [resolve]
|
|
(.close server (fn []
|
|
(l/info :hint "shutdown http server")
|
|
(resolve)))))
|
|
(p/resolved nil)))
|