From 72621f162e78150f459dd5e91ae098774b0aa207 Mon Sep 17 00:00:00 2001 From: Sibi Prabakaran Date: Sat, 26 Feb 2022 11:13:05 +0530 Subject: [PATCH 1/5] Make directory traversing smart --- justl.el | 92 +++++++++++++++++++++++++++++++++++++++----------------- 1 file changed, 64 insertions(+), 28 deletions(-) diff --git a/justl.el b/justl.el index 583cae6..32a2a4c 100644 --- a/justl.el +++ b/justl.el @@ -154,13 +154,47 @@ NAME is the buffer name." (goto-char (point-max)) (insert (format "%s\n" str)))) +(defconst justl--justfile-found "justfile" + "Constant to indicate that we know a justfile exists.") + +(defvar-local justl--justfile nil + "Buffer local variable which points to the justfile. + +If this is NIL, it means that no justfiles was found. In any +other cases, it's a known path.") + +(defun justl--find-any-justfiles (dir filename) + "Find just FILENAME inside a sub-directory DIR or a parent directory. + +Returns the absolute path if FILENAME exists or nil if no path +was found." + (let ((justfile-dir (f-traverse-upwards + (lambda (path) + (f-exists? (f-expand filename path))) + dir) )) + (if justfile-dir + (f-expand filename justfile-dir) + (let ((justfile-paths (directory-files-recursively dir filename))) + (if justfile-paths + (car justfile-paths) + nil))))) + (defun justl--find-justfiles (dir) - "Find all the justfiles inside a directory. + "Find justfiles inside a sub-directory DIR or a parent directory. -DIR represents the directory where search will be carried -out. The search will be performed recursively." - (let ((case-fold-search t)) - (directory-files-recursively dir "justfile"))) +DIR represents the directory where search will be carried out. +It searches either for the filename justfile or .justfile" + (let ((justfile-path (justl--find-any-justfiles dir "justfile"))) + (if justfile-path + (progn + (setq justl--justfile justfile-path) + justfile-path) + (let ((dot-justfile-path (justl--find-any-justfiles dir ".justfile"))) + (if dot-justfile-path + (progn + (setq justl--justfile dot-justfile-path) + dot-justfile-path) + nil))))) (defun justl--get-recipe-name (str) "Compute the recipe name from the string STR." @@ -175,7 +209,7 @@ out. The search will be performed recursively." (make-justl-jarg :arg (nth 0 arg) :default (nth 1 arg)))) (defun justl--str-to-jarg (str) - "Convert string STR to liat of JARG. + "Convert string STR to list of JARG. The string after the recipe name and before the build constraints is expected." @@ -253,7 +287,7 @@ ARGS is a ist of arguments." (setq process-name "just")) (let ((buffer-name justl--output-process-buffer) (error-buffer (justl--process-error-buffer process-name)) - (cmd (append (list justl-executable) args))) + (cmd (append (list justl-executable (justl--justfile-argument)) args))) (when (get-buffer buffer-name) (kill-buffer buffer-name)) (when (get-buffer error-buffer) @@ -293,10 +327,14 @@ and output of process." justl-executable))))) (mapcar #'string-trim-right recipies))) -(defun justl--get-recipies-with-desc () +(defun justl--justfile-argument () + (format "--justfile=%s" justl--justfile)) + +(defun justl--get-recipies-with-desc (justfile) "Return all the recipies with description." - (let* ((recipe-status (justl--exec-to-string-with-exit-code (format "%s --list --unsorted" - justl-executable))) + (let* ((recipe-status (justl--exec-to-string-with-exit-code + (format "%s --justfile=%s --list --unsorted" + justl-executable justfile))) (just-status (nth 0 recipe-status)) (recipe-lines (split-string (nth 1 recipe-status) @@ -339,8 +377,9 @@ and output of process." (defun justl--buffer-name () "Return justl buffer name." - (format "*just [%s]" - default-directory)) + (let ((justfile (justl--find-justfiles default-directory))) + (format "*just [%s]" + (f-dirname justfile)))) (defvar justl--line-number nil "Store the current line number to jump back after a refresh.") @@ -362,14 +401,16 @@ and output of process." "Opens eshell buffer but does not execute it. Populates the eshell buffer with RECIPE name so that it can be tweaked further by the user." - (let* ((eshell-buffer-name (format "justl - eshell - %s" recipe))) + (let* ((eshell-buffer-name (format "justl - eshell - %s" recipe)) + (default-directory (f-dirname justl--justfile))) (progn (eshell) (insert (format "just %s" recipe))))) (defun justl--exec-with-eshell (recipe) "Opens eshell buffer and execute the just RECIPE." - (let* ((eshell-buffer-name (format "justl - eshell - %s" recipe))) + (let* ((eshell-buffer-name (format "justl - eshell - %s" recipe)) + (default-directory (f-dirname justl--justfile))) (progn (eshell) (insert (format "just %s" recipe)) @@ -379,8 +420,7 @@ tweaked further by the user." "Execute just recipe in eshell." (interactive) (let* ((recipe (justl--get-word-under-cursor)) - (justfile (justl--find-justfiles default-directory)) - (justl-recipe (justl--get-recipe-from-file (car justfile) recipe)) + (justl-recipe (justl--get-recipe-from-file justl--justfile recipe)) (t-args (transient-args 'justl-help-popup)) (recipe-has-args (justl--jrecipe-has-args-p justl-recipe))) (if recipe-has-args @@ -398,8 +438,7 @@ tweaked further by the user." "Open eshell with the recipe but do not execute it." (interactive) (let* ((recipe (justl--get-word-under-cursor)) - (justfile (justl--find-justfiles default-directory)) - (justl-recipe (justl--get-recipe-from-file (car justfile) recipe)) + (justl-recipe (justl--get-recipe-from-file justl--justfile recipe)) (t-args (transient-args 'justl-help-popup)) (recipe-has-args (justl--jrecipe-has-args-p justl-recipe))) (if recipe-has-args @@ -456,8 +495,7 @@ tweaked further by the user." "Execute just recipe." (interactive) (let* ((recipe (justl--get-word-under-cursor)) - (justfile (justl--find-justfiles default-directory)) - (justl-recipe (justl--get-recipe-from-file (car justfile) recipe)) + (justl-recipe (justl--get-recipe-from-file justl--justfile recipe)) (t-args (transient-args 'justl-help-popup)) (recipe-has-args (justl--jrecipe-has-args-p justl-recipe))) (if recipe-has-args @@ -475,8 +513,7 @@ tweaked further by the user." "Execute just recipe with arguments." (interactive) (let* ((recipe (justl--get-word-under-cursor)) - (justfile (justl--find-justfiles default-directory)) - (justl-recipe (justl--get-recipe-from-file (car justfile) recipe)) + (justl-recipe (justl--get-recipe-from-file justl--justfile recipe)) (t-args (transient-args 'justl-help-popup)) (user-args (read-from-minibuffer (format "Arguments seperated by spaces:")))) @@ -491,10 +528,9 @@ tweaked further by the user." "Go to the recipe on justfile." (interactive) (let* ((recipe (justl--get-word-under-cursor)) - (justfile (justl--find-justfiles default-directory)) - (justl-recipe (justl--get-recipe-from-file (car justfile) recipe))) + (justl-recipe (justl--get-recipe-from-file justl--justfile recipe))) (progn - (find-file (car justfile)) + (find-file justl--justfile) (goto-char 0) (when (re-search-forward (concat (justl-jrecipe-name justl-recipe) ".*:") nil t 1) (let ((start (point))) @@ -523,8 +559,8 @@ tweaked further by the user." "Special mode for justl buffers." (buffer-disable-undo) (setq truncate-lines t) - (let ((justfiles (justl--find-justfiles default-directory)) - (entries (justl--get-recipies-with-desc))) + (let* ((justfiles (justl--find-justfiles default-directory)) + (entries (justl--get-recipies-with-desc justfiles))) (if (or (null justfiles) (not (eq justl--list-command-exit-code 0)) ) (progn (when (null justfiles) @@ -540,7 +576,7 @@ tweaked further by the user." (tabulated-list-init-header) (tabulated-list-print t) (hl-line-mode 1) - (message (concat "Just: " default-directory)) + (message (concat "Just: " (f-dirname justfiles))) (justl--jump-back-to-line)))) (provide 'justl) From 733a2c607b01ff384c484fa158ccd31b7030c02a Mon Sep 17 00:00:00 2001 From: Sibi Prabakaran Date: Sat, 26 Feb 2022 11:22:29 +0530 Subject: [PATCH 2/5] Fix docs --- justl.el | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/justl.el b/justl.el index 32a2a4c..fd80d98 100644 --- a/justl.el +++ b/justl.el @@ -328,10 +328,11 @@ and output of process." (mapcar #'string-trim-right recipies))) (defun justl--justfile-argument () + "Provides justfile argument with the proper location." (format "--justfile=%s" justl--justfile)) (defun justl--get-recipies-with-desc (justfile) - "Return all the recipies with description." + "Return all the recipies in JUSTFILE with description." (let* ((recipe-status (justl--exec-to-string-with-exit-code (format "%s --justfile=%s --list --unsorted" justl-executable justfile))) From 382362ad90a0adb8546a5146991d4fd5670580d3 Mon Sep 17 00:00:00 2001 From: Sibi Prabakaran Date: Sat, 26 Feb 2022 11:35:14 +0530 Subject: [PATCH 3/5] Cleanup and add more tests --- justl.el | 5 +---- test/justl-test.el | 14 +++++++++++--- 2 files changed, 12 insertions(+), 7 deletions(-) diff --git a/justl.el b/justl.el index fd80d98..c75c74d 100644 --- a/justl.el +++ b/justl.el @@ -154,9 +154,6 @@ NAME is the buffer name." (goto-char (point-max)) (insert (format "%s\n" str)))) -(defconst justl--justfile-found "justfile" - "Constant to indicate that we know a justfile exists.") - (defvar-local justl--justfile nil "Buffer local variable which points to the justfile. @@ -171,7 +168,7 @@ was found." (let ((justfile-dir (f-traverse-upwards (lambda (path) (f-exists? (f-expand filename path))) - dir) )) + dir))) (if justfile-dir (f-expand filename justfile-dir) (let ((justfile-paths (directory-files-recursively dir filename))) diff --git a/test/justl-test.el b/test/justl-test.el index de77444..26356d6 100644 --- a/test/justl-test.el +++ b/test/justl-test.el @@ -6,6 +6,7 @@ (require 'justl) (require 'ert) +(require 'f) (ert-deftest justl--get-recipies-test () (should (equal @@ -75,9 +76,6 @@ (justl--is-recipe-line-p " just --list") nil))) -(ert-deftest justl--find-justfiles-test () - (should (equal (length (justl--find-justfiles ".")) 1))) - (ert-deftest justl--get-recipe-from-file-test () (should (equal (justl--get-recipe-from-file "./justfile" "default") @@ -191,6 +189,16 @@ (kill-buffer (justl--buffer-name)) (kill-buffer justl--output-process-buffer)) +(ert-deftest justl--find-justfiles-check () + (should (equal (f-filename (justl--find-justfiles default-directory)) "justfile"))) + +(ert-deftest justl--get-recipies-with-desc-check () + (let* ((justfile (justl--find-justfiles default-directory)) + (recipies (justl--get-recipies-with-desc justfile))) + (should (member (list "default" "List all recipies") recipies)) + (should (member (list "push" nil) recipies)) + (should (member (list "push2" nil) recipies)))) + ;; (ert "justl--**") (provide 'justl-test) From c91a1c691a6d609c4536111c2697eacfe8aee078 Mon Sep 17 00:00:00 2001 From: Sibi Prabakaran Date: Sat, 26 Feb 2022 11:37:20 +0530 Subject: [PATCH 4/5] Update changelog --- Changelog.org | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/Changelog.org b/Changelog.org index c2a4836..e17f7c0 100644 --- a/Changelog.org +++ b/Changelog.org @@ -1,3 +1,11 @@ +* 0.6 + +- Make the directory parsing smart. Now invoking justl from any + subdirectory should work. +- eshell executes from the directory where justfile is placed. +- Introduce buffer local variable ~~justl--justfile~~ which contains + the justfile location for a particular justl-mode buffer. + * 0.5 - Stop following symbolic link as it leads to a high CPU usage. This From d605c3d592a1ef1b48d4d6dd713474940e98f7d0 Mon Sep 17 00:00:00 2001 From: Sibi Prabakaran Date: Sat, 26 Feb 2022 11:41:16 +0530 Subject: [PATCH 5/5] Add justl--exec-without-justfile --- justl.el | 26 +++++++++++++++++++++++++- 1 file changed, 25 insertions(+), 1 deletion(-) diff --git a/justl.el b/justl.el index c75c74d..e660b99 100644 --- a/justl.el +++ b/justl.el @@ -299,6 +299,30 @@ ARGS is a ist of arguments." :command cmd) (pop-to-buffer buffer-name))) +(defun justl--exec-without-justfile (process-name args) + "Utility function to run commands in the proper context and namespace. + +PROCESS-NAME is an identifier for the process. Default to \"just\". +ARGS is a ist of arguments." + (when (equal process-name "") + (setq process-name "just")) + (let ((buffer-name justl--output-process-buffer) + (error-buffer (justl--process-error-buffer process-name)) + (cmd (append (list justl-executable) args))) + (when (get-buffer buffer-name) + (kill-buffer buffer-name)) + (when (get-buffer error-buffer) + (kill-buffer error-buffer)) + (justl--log-command process-name cmd) + (make-process :name process-name + :buffer buffer-name + :filter 'justl--xterm-color-filter + :sentinel #'justl--sentinel + :file-handler t + :stderr nil + :command cmd) + (pop-to-buffer buffer-name))) + (defun justl--exec-to-string (cmd) "Replace \"shell-command-to-string\" to log to process buffer. @@ -358,7 +382,7 @@ and output of process." (interactive) (let* ((recipies (completing-read "Recipies: " (justl--get-recipies) nil nil nil nil "default"))) - (justl--exec "just" (list recipies)))) + (justl--exec-without-justfile "just" (list recipies)))) (defvar justl-mode-map (let ((map (make-sparse-keymap)))