feat(web/panettone): Allow editing issues

Allow editing both the subject and the body of issues, recording events
indicating the edit and displaying those events in the issue history.

Fixes: #14
Change-Id: I9ed05271ce9bf6bda4e56f15e249c0f28c862b27
Reviewed-on: https://cl.tvl.fyi/c/depot/+/1517
Tested-by: BuildkiteCI
Reviewed-by: tazjin <mail@tazj.in>
This commit is contained in:
Griffin Smith 2020-08-01 12:10:27 -04:00 committed by glittershark
parent 2646e57aae
commit b6bab664db
4 changed files with 146 additions and 41 deletions

View file

@ -4,6 +4,9 @@
(defparameter color/black (defparameter color/black
"rgb(24, 24, 24)") "rgb(24, 24, 24)")
(defparameter color/light-gray
"#EEE")
(defparameter color/gray (defparameter color/gray
"#8D8D8D") "#8D8D8D")
@ -105,7 +108,8 @@
((:and input (:= type "submit")) ((:and input (:= type "submit"))
:-webkit-appearance "none" :-webkit-appearance "none"
:border "none" :border "none"
:cursor "pointer") :cursor "pointer"
:font-size "1rem")
,@(button '(:and input (:= type "submit"))) ,@(button '(:and input (:= type "submit")))
@ -126,6 +130,16 @@
:justify-content "space-between" :justify-content "space-between"
:align-items "center" :align-items "center"
,@(button '.edit-issue)
(.created-by-at
:flex 1)
(.edit-issue
:background-color ,color/light-gray
:flex 0
:margin-right "0.5rem")
(.close-issue (.close-issue
:background-color ,color/failure)))) :background-color ,color/failure))))

View file

@ -74,6 +74,9 @@
(cl-postgres:to-sql-string "open")) (cl-postgres:to-sql-string "open"))
(defmethod cl-postgres:to-sql-string ((kw (eql :closed))) (defmethod cl-postgres:to-sql-string ((kw (eql :closed)))
(cl-postgres:to-sql-string "closed")) (cl-postgres:to-sql-string "closed"))
(defmethod cl-postgres:to-sql-string ((ts local-time:timestamp))
(cl-postgres:to-sql-string
(local-time:timestamp-to-unix ts)))
(defmethod initialize-instance :after (defmethod initialize-instance :after
((issue issue) &rest initargs &key &allow-other-keys) ((issue issue) &rest initargs &key &allow-other-keys)
@ -292,6 +295,29 @@ the issue doesn't exist, signals `issue-not-found'"
:new-value status) :new-value status)
(values))) (values)))
(defun update-issue (issue &rest attrs)
"Update the fields of ISSUE with the given ATTRS, which is a plist of slot and
value, and record events for the updates"
(let ((set-fields
(iter (for slot in '(subject body))
(for new-value = (getf attrs slot))
(appending
(let ((previous-value (slot-value issue slot)))
(when (and new-value (not (equalp
new-value
previous-value)))
(record-issue-event (id issue)
:field slot
:previous-value previous-value
:new-value new-value)
(setf (slot-value issue slot) new-value)
(list slot new-value)))))))
(execute
(sql-compile
`(:update issues
:set ,@set-fields
:where (:= id ,(id issue)))))))
(defun create-issue-comment (&rest attrs &key issue-id &allow-other-keys) (defun create-issue-comment (&rest attrs &key issue-id &allow-other-keys)
"Insert a new issue comment into the database with the given ATTRS and "Insert a new issue comment into the database with the given ATTRS and
ISSUE-ID, which should be a plist of initforms, and return an instance of ISSUE-ID, which should be a plist of initforms, and return an instance of

View file

@ -19,8 +19,8 @@
(defpackage panettone.model (defpackage panettone.model
(:nicknames :model) (:nicknames :model)
(:use :cl :panettone.util :klatre :postmodern) (:use :cl :panettone.util :klatre :postmodern :iterate)
(:import-from :alexandria :if-let :define-constant) (:import-from :alexandria :if-let :when-let :define-constant)
(:export (:export
:connect-postgres :ddl/init :connect-postgres :ddl/init
@ -29,7 +29,7 @@
:field :previous-value :new-value :field :previous-value :new-value
:get-issue :issue-exists-p :list-issues :create-issue :set-issue-status :get-issue :issue-exists-p :list-issues :create-issue :set-issue-status
:delete-issue :issue-not-found :update-issue :delete-issue :issue-not-found
:issue-events :issue-events

View file

@ -107,6 +107,20 @@
(when ,footer (when ,footer
(render/footer-nav))))))) (render/footer-nav)))))))
(defun form-button (&key
class
input-class
href
label
(method "post"))
(who:with-html-output (*standard-output*)
(:form :class class
:method method
:action href
(:input :type "submit"
:class input-class
:value label))))
(defun render/alert (message) (defun render/alert (message)
"Render an alert box for MESSAGE, if non-null" "Render an alert box for MESSAGE, if non-null"
(check-type message (or null string)) (check-type message (or null string))
@ -209,28 +223,38 @@
(:a :href "/" "View open isues")) (:a :href "/" "View open isues"))
(render/issue-list :issues issues)))) (render/issue-list :issues issues))))
(defun render/new-issue (&optional message) (defun render/issue-form (&optional issue message)
(render () (render ()
(:header (:header
(:h1 "New Issue")) (:h1
(who:esc
(if (id issue) "Edit Issue" "New Issue"))))
(:main (:main
(render/alert message) (render/alert message)
(:form :method "post" (:form :method "post"
:action "/issues" :action (if (id issue)
(format nil "/issues/~A"
(id issue))
"/issues")
:class "issue-form" :class "issue-form"
(:div (:div
(:input :type "text" (:input :type "text"
:id "subject" :id "subject"
:name "subject" :name "subject"
:placeholder "Subject")) :placeholder "Subject"
:value (subject issue)))
(:div (:div
(:textarea :name "body" (:textarea :name "body"
:placeholder "Description" :placeholder "Description"
:rows 10)) :rows 10
(who:esc (body issue))))
(:input :type "submit" (:input :type "submit"
:value "Create Issue"))))) :value
(if (id issue)
"Save Issue"
"Create Issue"))))))
(defun render/new-comment (issue-id) (defun render/new-comment (issue-id)
(who:with-html-output (*standard-output*) (who:with-html-output (*standard-output*)
@ -261,19 +285,28 @@
(who:esc (format-dottime (created-at comment))))))))) (who:esc (format-dottime (created-at comment)))))))))
(defmethod render/issue-history-item ((event model:issue-event)) (defmethod render/issue-history-item ((event model:issue-event))
(when (string= (field event) "STATUS") (let ((user (find-user-by-dn (acting-user-dn event))))
(who:with-html-output (*standard-output*) (who:with-html-output (*standard-output*)
(let ((user (find-user-by-dn (acting-user-dn event)))) (:li
(who:htm :class "event"
(:li (who:esc (displayname user))
:class "event" (if (string= (field event) "STATUS")
(who:esc (displayname user)) (who:htm
(who:esc (who:esc
(switch ((new-value event) :test #'string=) (switch ((new-value event) :test #'string=)
("OPEN" " reopened ") ("OPEN" " reopened ")
("CLOSED" " closed "))) ("CLOSED" " closed ")))
" this issue at " " this issue ")
(who:esc (format-dottime (created-at event))))))))) (who:htm
" changed the "
(who:esc (string-downcase (field event)))
" of this issue from \""
(who:esc (previous-value event))
"\" to \""
(who:esc (new-value event))
"\""))
" at "
(who:esc (format-dottime (created-at event)))))))
(defun render/issue (issue) (defun render/issue (issue)
(check-type issue model:issue) (check-type issue model:issue)
@ -291,20 +324,26 @@
(when *user* (when *user*
(who:htm (who:htm
(:form :class "set-issue-status" (when (string= (author-dn issue)
:method "post" (dn *user*))
:action (format nil "/issues/~A/~A" (who:htm
issue-id (:a :class "edit-issue"
(case issue-status :href (format nil "/issues/~A/edit"
(:open "close") issue-id)
(:closed "open"))) "Edit")))
(:input :type "submit" (form-button
:class (case issue-status :class "set-issue-status"
(:open "close-issue") :href (format nil "/issues/~A/~A"
(:closed "open-issue")) issue-id
:value (case issue-status (case issue-status
(:open "Close") (:open "close")
(:closed "Reopen"))))))) (:closed "open")))
:input-class (case issue-status
(:open "close-issue")
(:closed "open-issue"))
:label (case issue-status
(:open "Close")
(:closed "Reopen"))))))
(:p (who:str (render-markdown (body issue)))) (:p (who:str (render-markdown (body issue))))
(let* ((comments (issue-comments issue)) (let* ((comments (issue-comments issue))
(events (issue-events issue)) (events (issue-events issue))
@ -388,17 +427,19 @@
(render/closed-issues :issues issues))) (render/closed-issues :issues issues)))
(defroute new-issue ("/issues/new" :decorators (@auth)) () (defroute new-issue ("/issues/new" :decorators (@auth)) ()
(render/new-issue)) (render/issue-form))
(defroute handle-create-issue (defroute handle-create-issue
("/issues" :method :post :decorators (@auth @txn)) ("/issues" :method :post :decorators (@auth @txn))
(&post subject body) (&post subject body)
(if (string= subject "") (if (string= subject "")
(render/new-issue "Subject is required") (render/issue-form
(make-instance 'model:issue :subject subject :body body)
"Subject is required")
(progn (progn
(model:create-issue :subject subject (model:create-issue :subject subject
:body body :body body
:author-dn (dn *user*)) :author-dn (dn *user*))
(hunchentoot:redirect "/")))) (hunchentoot:redirect "/"))))
(defroute show-issue (defroute show-issue
@ -409,6 +450,30 @@
(subject issue)))) (subject issue))))
(render/issue issue))) (render/issue issue)))
(defroute edit-issue
("/issues/:id/edit" :decorators (@auth @handle-issue-not-found))
(&path (id 'integer))
(let* ((issue (model:get-issue id))
(*title* "Edit Issue | Panettone"))
(render/issue-form issue)))
(defroute update-issue
("/issues/:id" :decorators (@auth @handle-issue-not-found @txn)
;; NOTE: this should be a put, but we're all HTML forms
;; right now and those don't support PUT
:method :post)
(&path (id 'integer) &post subject body)
(let ((issue (model:get-issue id)))
;; only the original author can edit an issue
(if (string-equal (author-dn issue)
(dn *user*))
(progn
(model:update-issue issue
'model:subject subject
'model:body body)
(hunchentoot:redirect (format nil "/issues/~A" id)))
(render/not-found "Issue"))))
(defroute handle-create-comment (defroute handle-create-comment
("/issues/:id/comments" ("/issues/:id/comments"
:decorators (@auth @handle-issue-not-found @txn) :decorators (@auth @handle-issue-not-found @txn)
@ -435,7 +500,7 @@
(defroute open-issue (defroute open-issue
("/issues/:id/open" :decorators (@auth) ("/issues/:id/open" :decorators (@auth)
:method :post) :method :post)
(&path (id 'integer)) (&path (id 'integer))
(model:set-issue-status id :open) (model:set-issue-status id :open)
(hunchentoot:redirect (format nil "/issues/~A" id))) (hunchentoot:redirect (format nil "/issues/~A" id)))