0

这周我开始学习 Clojure,特别是我正在学习 Luminus 的 Web 开发。因为我想了解 CRUD 过程,所以我设置了一个函数来将我的帖子保存到数据库中:

(defn save-post! [{:keys [params]}]
  (if-let [errors (validate-post params)]
    (-> (response/found "/posts")
      (assoc :flash (assoc params :errors errors)))
  (do
    (db/save-post!
      (assoc params :created_at (java.util.Date.)))
    (response/found "/posts"))))

查询非常基本:

-- :name save-post! :! :n
-- :doc creates a new post record
 INSERT INTO posts
 (title, body, active, created_at)
 VALUES (:title, :body, :active, :created_at)

但 HTML 表单有一个复选框字段:

<input type="checkbox" name="active" value="1">Published<br />

未选中时,不发送该字段,SQL 插入查询发送错误消息“No active field”。如何检查是否设置了元素“活动”并将其添加到“参数”中作为真或假?

就像是:

  (assoc params :active (if (nil? params/active) false true))  

在 ":created_at (java.util.Date.)" 行之后。

4

1 回答 1

2

如何检查是否设置了元素“活动”并将其添加到“参数”中作为真或假?

看起来你的代码离工作不远了。您需要检查params地图以查看它是否具有复选框的值。如果(:active params)等于"1"选中复选框时,那么您可能会执行以下操作:

(assoc params :active (= "1" (:active params)))

但这真正要做的是更新地图中的特定值,这可以更惯用地完成:

(update params :active #(= "1" %))

最后一个参数是一个函数,它接受关键字的任何当前值并返回新值。

另一个潜在的问题:您可能不想将params映射用作数据库查询的直接输入,因为它很容易包含您不想要或不期望的键/值。仅从其中显式提取您需要的值会更安全,例如(select-keys params [:title :body :active]).

(def params {:active "1", :admin true}) ;; wouldn't want admin to leak through!
(-> params
    (select-keys [:title :body :active])
    (assoc :created_at (java.util.Date.))
    (update :active #(= "1" %)))
;;=> {:active true, :created_at #inst "2017-10-09T20:16:06.167-00:00"}
于 2017-10-09T20:19:52.027 回答