Skip to content

Commit

Permalink
gptel-context: Support add and remove files in directories to context
Browse files Browse the repository at this point in the history
  • Loading branch information
benthamite committed Dec 5, 2024
1 parent 6b2c0b9 commit 2796b1e
Showing 1 changed file with 42 additions and 18 deletions.
60 changes: 42 additions & 18 deletions gptel-context.el
Original file line number Diff line number Diff line change
Expand Up @@ -169,32 +169,54 @@ context chunk. This is accessible as, for example:
(eq buffer-file-coding-system 'no-conversion))
(file-missing (message "File \"%s\" is not readable." path))))

(defun gptel-context--handle-binary (path)
"Add binary file at PATH to context if supported.
Return PATH if added, nil if ignored."
(if-let* (((gptel--model-capable-p 'media))
(mime (mailcap-file-name-to-mime-type path))
((gptel--model-mime-capable-p mime)))
(prog1 path
(cl-pushnew (list path :mime mime)
gptel-context--alist :test #'equal)
(message "File \"%s\" added to context." path))
(message "Ignoring unsupported binary file \"%s\"." path)
nil))

(defun gptel-context--handle-directory (path action)
"Process all files in directory at PATH according to ACTION.
ACTION should be either `add' or `remove'."
(let ((files (directory-files-recursively path "." t)))
(mapc (lambda (file)
(unless (file-directory-p file)
(if (eq action 'add)
(if (gptel--file-binary-p file)
(gptel-context--handle-binary file)
(cl-pushnew (list file) gptel-context--alist :test #'equal)
(message "File \"%s\" added to context." file))
(setf (alist-get file gptel-context--alist nil 'remove #'equal) nil))))
files)))

(defun gptel-context-add-file (path)
"Add the file at PATH to the gptel context.
If PATH is a directory, recursively add all files in it.
PATH should be readable as text."
(interactive "fChoose file to add to context: ")
(if (gptel--file-binary-p path) ;Attach if supported
(if-let* (((gptel--model-capable-p 'media))
(mime (mailcap-file-name-to-mime-type path))
((gptel--model-mime-capable-p mime)))
(prog1 path
(cl-pushnew (list path :mime mime)
gptel-context--alist :test #'equal)
(message "File \"%s\" added to context." path))
(message "Ignoring unsupported binary file \"%s\"." path))
;; Add text file
(cl-pushnew (list path) gptel-context--alist :test #'equal)
(message "File \"%s\" added to context." path)
path))
(if (file-directory-p path)
(gptel-context--handle-directory path 'add)
(if (gptel--file-binary-p path)
(gptel-context--handle-binary path)
(cl-pushnew (list path) gptel-context--alist :test #'equal)
(message "File \"%s\" added to context." path)
path)))

;;;###autoload (autoload 'gptel-add-file "gptel-context" "Add files to gptel's context." t)
(defalias 'gptel-add-file #'gptel-context-add-file)

(defun gptel-context-remove (&optional context)
"Remove the CONTEXT overlay from the contexts list.
If CONTEXT is nil, removes the context at point.
If selection is active, removes all contexts within selection."
If selection is active, removes all contexts within selection.
If CONTEXT is a directory, recursively removes all files in it."
(cond
((overlayp context)
(delete-overlay context)
Expand All @@ -204,9 +226,11 @@ If selection is active, removes all contexts within selection."
for ov in (alist-get (current-buffer) gptel-context--alist)
thereis (overlay-start ov))
(setf (alist-get (current-buffer) gptel-context--alist nil 'remove) nil)))
((stringp context) ;file
(setf (alist-get context gptel-context--alist nil 'remove #'equal)
nil))
((stringp context) ;file or directory
(if (file-directory-p context)
(gptel-context--handle-directory context 'remove)
(setf (alist-get context gptel-context--alist nil 'remove #'equal)
nil)))
((region-active-p)
(when-let ((contexts (gptel-context--in-region (current-buffer)
(region-beginning)
Expand Down

0 comments on commit 2796b1e

Please sign in to comment.