-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcore.lisp
166 lines (147 loc) · 7.27 KB
/
core.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
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
;;;; cl-tree: Reimplementing Unix 'tree' command in Common Lisp
;;;; Copyright (C) 2015 Florian Patzl
;;;;
;;;; This program is free software: you can redistribute it and/or modify
;;;; it under the terms of the GNU General Public License as published by
;;;; the Free Software Foundation, either version 3 of the License, or
;;;; (at your option) any later version.
;;;; This program is distributed in the hope that it will be useful,
;;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;;;; GNU General Public License for more details.
;;;; You should have received a copy of the GNU General Public License
;;;; along with this program. If not, see <http://www.gnu.org/licenses/>.
;;;; -----------------------------------------------------------------------------------------------
(in-package :cl-user)
(defpackage #:com.github.flpa.cl-tree.core
(:use :common-lisp)
(:import-from :uiop/pathname
:directory-pathname-p
:hidden-pathname-p)
(:import-from :uiop/filesystem
:directory-files)
(:export :tree))
(in-package :com.github.flpa.cl-tree.core)
(defparameter *line-middle* "├── ")
(defparameter *line-straight* "│ ")
(defparameter *line-end* "└── ")
;;;; ---------------
;;;; Public API
;;;; ---------------
;; TODO: not satisfied with the function signature
(defun tree (directories &key (noreport nil)
(show-hidden nil)
(directories-only nil)
(prune-empty nil)
(file-limit nil))
"TREE traverses a list of DIRECTORIES, represented as pathnames, while printing their contents
according to the specified parameters."
(fresh-line) ; We always want to start on a fresh line.
(walk-tree
directories
'()
(build-predicates show-hidden
directories-only
prune-empty
(when file-limit file-limit))
0
0
(not noreport)))
;; pros of flat list: convert to tail call recursion, direct passing of counts, convert to list
;; cons: how to do things before and after folders? (indentation, xml?)
(defun walk-tree (frontier prefixes predicates dircount filecount print-report)
"A recursive implementation of the tree functionality, inspired by depth-first-search as
showcased in AIMA by Norvig.
FRONTIER is the LIFO list of pathnames elements to be handled, the list of PREDICATES controls
which elements are printed (and explored, in the case of directories).
PREFIXES is the list of strings to be printed before the current file. DIRCOUNT and FILECOUNT
are numbers maintained for a final report, which is printed if PRINT-REPORT is true."
(if frontier
(let ((current (first frontier)))
(if (equal current :closedir)
(walk-tree (rest frontier) (rest prefixes) predicates dircount filecount print-report)
(progn
(format t "~{~a~}~a~%"
(if (equal :closedir (first (rest frontier)))
(append (butlast prefixes) (list *line-end*))
prefixes)
(base-name current))
(if (directory-pathname-p current)
(walk-tree (append (sort-with-hidden
(filter-pathnames (directory-files current) predicates))
(list :closedir)
(rest frontier))
(cons (if prefixes *line-straight* *line-middle*) prefixes)
predicates
(if prefixes (1+ dircount) dircount)
filecount
print-report)
(walk-tree (rest frontier) prefixes predicates dircount (1+ filecount) print-report)))))
(progn
(when print-report
(print-report dircount filecount))
(fresh-line))))
;;;; ---------------
;;;; Internals
;;;; ---------------
;;; macros like 'def-directory-predicate', 'def-file-predicate', 'def-general-predicate' ?
(defun build-predicates (show-hidden directories-only prune-empty &optional file-limit)
"Builds a list of predicates to be applied to every folder and file based on the parameter.
This effectively defines the mapping between parameters and predicate-functions."
;; TODO: macro for better readability of parameter<>predicate mapping?
(loop for x in `((,(not show-hidden) t t ,#'visible-p)
(,directories-only nil t ,(constantly nil))
(,prune-empty t nil ,(complement #'empty-dir-p))
(,file-limit t nil ,(dir-within-limit-p file-limit)))
if (and (first x) (second x)) collect (car (last x)) into dir-predicates
if (and (first x) (third x)) collect (car (last x)) into file-predicates
finally (return (list dir-predicates file-predicates))))
;; TODO ... why do I need to do this?
(defun base-name (p)
"Determines the base name of the `pathname' P, i.e. the directory name for directories or the file
name, including the extension, for files."
(if (directory-pathname-p p)
(car (last (pathname-directory p)))
(if (pathname-type p)
(concatenate 'string (pathname-name p) "." (pathname-type p))
(pathname-name p))))
(defun filter-pathnames (pathnames predicates)
"Returns the filtered sequence of PATHNAMES that satisfy all of the sequence of PREDICATES."
(remove-if #'(lambda (item)
(notevery #'(lambda (p) (funcall p item)) (if (directory-pathname-p item)
(car predicates)
(cadr predicates))))
pathnames))
;; TODO: no longer needed?
(defun filter-items (items predicates)
"Returns the filtered sequence of ITEMS that satisfy all of the sequence of PREDICATES."
(remove-if #'(lambda (item)
(notevery #'(lambda (p) (funcall p item)) predicates))
items))
(defun sort-with-hidden (pathnames)
"Sorts a list of PATHNAMES, ignoring leading dots and case."
(sort pathnames #'string-lessp
:key #'(lambda(x) (remove-leading-dots (base-name x)))))
(defun remove-leading-dots (input)
"Removes any number of leading dots from the `string' INPUT."
(string-left-trim (list #\.) input))
(defun print-report (dircount filecount)
"Prints the number of directories and files, also taking taking care of pluralization.
e.g.: '5 directories, 1 file'.
Output is preceded by a blank line."
(format t "~&~%~a director~:@P, ~a file~:P" dircount filecount))
(defun visible-p (pathname)
"Determines whether a given PATHNAME is visible, i.e. not hidden, by Unix conventions:
Hidden files and directories start with a dot (.)"
(not (char-equal #\.
(aref (base-name pathname) 0))))
(defun empty-dir-p (pathname)
"Checks whether a given PATHNAME denotes an empty directory."
(endp (directory-files pathname)))
;; TODO actually not a predicate, misleading
;; TODO actual tree does print directory but with msg like
;; [200 entries exceeds filelimit, not opening dir]
(defun dir-within-limit-p (limit)
"Returns a predicate function that only directories with no more than LIMIT children."
(lambda (pathname)
(<= (length (directory-files pathname)) limit)))