-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathgraph.lisp
86 lines (78 loc) · 3.29 KB
/
graph.lisp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
(defpackage :co.fwoar.cl-git.graph
(:use :cl :fwoar.cl-git)
(:export ))
(in-package :co.fwoar.cl-git.graph)
(defclass git-graph ()
((%repo :initarg :repo :reader repo)
(%depth :initarg :depth :reader depth)
(%stops :initarg :stops :reader stops :initform ())
(%branches :reader branches)
(%node-cache :reader node-cache :initform (make-hash-table :test 'equal))
(%edge-cache :reader edge-cache :initform (make-hash-table :test 'equal))))
(defmethod initialize-instance :after ((object git-graph) &key)
(setf
(slot-value object '%branches)
(fw.lu:alist-string-hash-table
(funcall (data-lens:over
(data-lens:<>1 (data-lens:applying #'cons)
(data-lens:transform-head
(serapeum:op (subseq _1 0
(min (length _1)
8))))
#'reverse))
(fwoar.cl-git::branches (repo object))))))
(defun git-graph (repo)
(fw.lu:new 'git-graph repo))
(defun get-commit-parents (repository commit)
#+lispworks
(declare (notinline mismatch serapeum:string-prefix-p))
(when commit
(co.fwoar.git:with-repository (repository)
(alexandria:when-let*
((ref (fwoar.cl-git:ensure-ref commit))
(direct-obj (fwoar.cl-git::extract-object
ref))
(obj direct-obj)
(parents (fwoar.cl-git:component
:parents
obj)))
(cond ((null parents) parents)
((null (cdr parents))
(let ((maybe-branch (get-commit-parents repository
(car parents))))
(if maybe-branch
maybe-branch
parents)))
(t parents))))))
(defmethod cl-dot:graph-object-node ((graph git-graph) (commit string))
(alexandria:ensure-gethash
commit
(node-cache graph)
(make-instance 'cl-dot:node
:attributes `(:label ,(gethash #1=(subseq commit 0 8)
(branches graph)
#1#)))))
(defmethod cl-dot:graph-object-points-to
((graph git-graph) (commit string))
(unless (member commit (stops graph)
:test 'serapeum:string-prefix-p)
(funcall (data-lens:<>1
(data-lens:over (serapeum:op
(setf (gethash (list commit _1)
(edge-cache graph))
t)
_1))
(data-lens:exclude (serapeum:op
(gethash (list commit _1)
(edge-cache graph))))
(data-lens:over (serapeum:op (subseq _ 0 8))))
(get-commit-parents (repo graph) commit))))
(defun graph-repository (path roots)
(co.fwoar.git:with-repository (path)
(cl-dot:generate-graph-from-roots
(make-instance 'git-graph
:repo fwoar.cl-git::*git-repository*
:stops ())
(mapcar (data-lens:∘ 'fwoar.cl-git::ref-hash
'co.fwoar.git::resolve-refish)
roots))))