2

我有一个 defresource,应该接受 POST 请求,在 :malformed-decision 中验证请求正文,在 :post!-decision 中将正文保存到数据库中,然后在 :handle-created 中返回保存的正文。

(defn parse-project [context] (json/read-str
                               (slurp (get-in context [:request :body]))
                               :key-fn keyword))    
(defresource add-new-project
         :malformed? (fn[ctx] (not (project-is-valid (parse-project ctx))))
         :handle-malformed (fn [_] (generate-string (str "Malformed json!")))
         ...
         :post! (fn [ctx] (save-to-db (parse-project ctx))
         :handle-created  (fn [ctx] (... parse-project ...))

所以我的代码使用 slurp 函数读取了 ByteArrayInputStream(来自 :request :body)的三倍。第一次有效,但第二次调用 slurp 时,nil 作为参数传递并出现 java.io.EOFException: JSON 错误。我认为读者开始阅读上次离开的地方。

我怎么能读三遍请求的正文?或者有没有很好的方法将读取结果保存到变量并将其传递给其他解放者决策?

4

1 回答 1

3

context可以通过每个决策和操作函数的结果进行更新。您可以解析项目一次malformed?并返回一个包含已解析项目的映射,该映射将被合并到上下文中,因此可用于以下决策和操作。例如:

(defresource add-new-project
  :malformed? (fn[ctx] (let [project (parse-project ctx)]
                         (when (project-is-valid project)
                           {:project project})))
  :handle-malformed (fn [_] (generate-string (str "Malformed json!")))
  :post! (fn [ctx] (save-to-db (:project ctx)))
  :handle-created (fn [ctx] (do-something (:project ctx))))

如果项目有效,则:malformed?返回{:project project} 将合并到上下文中以用于下一个决策和操作的映射。如果项目无效,它将返回nil以便继续执行:handle-malformed

有关 liberator 执行模型的更多信息,请参阅https://clojure-liberator.github.io/liberator/doc/execution-model.html

于 2015-07-12T19:10:02.193 回答