feat(web/panettone): Display who opened issues and when
Add a line to the issue show page displaying who opened the issue and when, the latter formatted in dottime. Change-Id: Ie70d7fd9e62ae92f9a479969d4ea21daddccee40 Reviewed-on: https://cl.tvl.fyi/c/depot/+/1345 Reviewed-by: glittershark <grfn@gws.fyi> Tested-by: BuildkiteCI
This commit is contained in:
parent
32c3f7731b
commit
d3b7de0783
2 changed files with 39 additions and 8 deletions
|
@ -7,8 +7,9 @@ depot.nix.buildLisp.program {
|
||||||
cl-prevalence
|
cl-prevalence
|
||||||
cl-who
|
cl-who
|
||||||
defclass-std
|
defclass-std
|
||||||
hunchentoot
|
|
||||||
easy-routes
|
easy-routes
|
||||||
|
hunchentoot
|
||||||
|
local-time
|
||||||
trivial-ldap
|
trivial-ldap
|
||||||
|
|
||||||
depot.lisp.klatre
|
depot.lisp.klatre
|
||||||
|
|
|
@ -14,14 +14,15 @@
|
||||||
|
|
||||||
(defclass/std issue-comment ()
|
(defclass/std issue-comment ()
|
||||||
((body :type string)
|
((body :type string)
|
||||||
(author-dn :type string)))
|
(author-dn :type string)
|
||||||
|
(created-at :type local-time:timestamp)))
|
||||||
|
|
||||||
(defclass/std issue (cl-prevalence:object-with-id)
|
(defclass/std issue (cl-prevalence:object-with-id)
|
||||||
((subject body :type string :std "")
|
((subject body :type string :std "")
|
||||||
(author-dn :type string)
|
(author-dn :type string)
|
||||||
(comments :std nil :type list :with-prefix)
|
(comments :std nil :type list :with-prefix)
|
||||||
(created-at :type integer
|
(created-at :type local-time:timestamp
|
||||||
:std (get-universal-time))))
|
:std (local-time:now))))
|
||||||
|
|
||||||
(defclass/std user ()
|
(defclass/std user ()
|
||||||
((cn dn mail displayname :type string)))
|
((cn dn mail displayname :type string)))
|
||||||
|
@ -55,15 +56,28 @@
|
||||||
(ldap:search
|
(ldap:search
|
||||||
*ldap*
|
*ldap*
|
||||||
`(and (= objectClass organizationalPerson)
|
`(and (= objectClass organizationalPerson)
|
||||||
(= cn ,username))
|
(or
|
||||||
|
(= cn ,username)
|
||||||
|
(= dn ,username)))
|
||||||
;; TODO(grfn): make this configurable
|
;; TODO(grfn): make this configurable
|
||||||
:base "ou=users,dc=tvl,dc=fyi")
|
:base "ou=users,dc=tvl,dc=fyi")
|
||||||
(ldap:next-search-result *ldap*))
|
(ldap:next-search-result *ldap*))
|
||||||
|
|
||||||
(defun find-user (username)
|
(defun find-user (username)
|
||||||
(check-type username (simple-array character (*)))
|
(check-type username (simple-array character (*)))
|
||||||
(ldap-entry->user
|
(when-let ((ldap-entry (find-user/ldap username)))
|
||||||
(find-user/ldap username)))
|
(ldap-entry->user ldap-entry)))
|
||||||
|
|
||||||
|
(defun find-user-by-dn (dn)
|
||||||
|
(ldap:search *ldap* `(= objectClass organizationalPerson)
|
||||||
|
:base dn
|
||||||
|
:scope 'ldap:base)
|
||||||
|
(when-let ((ldap-entry (ldap:next-search-result *ldap*)))
|
||||||
|
(ldap-entry->user ldap-entry)))
|
||||||
|
|
||||||
|
(comment
|
||||||
|
(user-by-dn "cn=glittershark,ou=users,dc=tvl,dc=fyi")
|
||||||
|
)
|
||||||
|
|
||||||
(defun authenticate-user (user-or-username password)
|
(defun authenticate-user (user-or-username password)
|
||||||
"Checks the given USER-OR-USERNAME has the given PASSWORD, by making a bind
|
"Checks the given USER-OR-USERNAME has the given PASSWORD, by making a bind
|
||||||
|
@ -81,6 +95,9 @@ successful, `nil' otherwise"
|
||||||
(when (equalp code-sym 'trivial-ldap:success)
|
(when (equalp code-sym 'trivial-ldap:success)
|
||||||
user))))
|
user))))
|
||||||
|
|
||||||
|
(defun author (object)
|
||||||
|
(find-user-by-dn (author-dn object)))
|
||||||
|
|
||||||
;;;
|
;;;
|
||||||
;;; Persistence
|
;;; Persistence
|
||||||
;;;
|
;;;
|
||||||
|
@ -138,6 +155,7 @@ updated issue"
|
||||||
(ensure-directories-exist data-dir)
|
(ensure-directories-exist data-dir)
|
||||||
(setq *p-system* (cl-prevalence:make-prevalence-system data-dir))
|
(setq *p-system* (cl-prevalence:make-prevalence-system data-dir))
|
||||||
|
|
||||||
|
|
||||||
(when (null (list-issues *p-system*))
|
(when (null (list-issues *p-system*))
|
||||||
(cl-prevalence:tx-create-id-counter *p-system*)))
|
(cl-prevalence:tx-create-id-counter *p-system*)))
|
||||||
|
|
||||||
|
@ -208,11 +226,22 @@ updated issue"
|
||||||
(:input :type :submit
|
(:input :type :submit
|
||||||
:value "Create Issue"))))
|
:value "Create Issue"))))
|
||||||
|
|
||||||
|
(defun created-by-at (issue)
|
||||||
|
(format nil "Opened by ~A at ~A"
|
||||||
|
(when-let ((author (author issue)))
|
||||||
|
(displayname author))
|
||||||
|
(format-dottime (created-at issue))))
|
||||||
|
|
||||||
|
(comment
|
||||||
|
(format nil "foo: ~A" "foo")
|
||||||
|
)
|
||||||
|
|
||||||
(defun render/issue (issue)
|
(defun render/issue (issue)
|
||||||
(check-type issue issue)
|
(check-type issue issue)
|
||||||
(render
|
(render
|
||||||
(:h1 (who:esc (subject issue)))
|
(:h1 (who:esc (subject issue)))
|
||||||
(:div (who:esc (body issue)))))
|
(:p (who:esc (created-by-at issue)))
|
||||||
|
(:p (who:esc (body issue)))))
|
||||||
|
|
||||||
(defun render/not-found (entity-type)
|
(defun render/not-found (entity-type)
|
||||||
(render
|
(render
|
||||||
|
@ -301,6 +330,7 @@ updated issue"
|
||||||
(sb-thread:list-all-threads)))))
|
(sb-thread:list-all-threads)))))
|
||||||
|
|
||||||
(comment
|
(comment
|
||||||
|
(setq hunchentoot:*catch-errors-p* nil)
|
||||||
(start-panettone :port 6161
|
(start-panettone :port 6161
|
||||||
:data-dir "/tmp/panettone"
|
:data-dir "/tmp/panettone"
|
||||||
:ldap-port 3899)
|
:ldap-port 3899)
|
||||||
|
|
Loading…
Reference in a new issue