psearch: 基于 pcase 的 elisp 代码搜索工具

比如在包 doom-modeline 中,函数 doom-modeline-segment--matches 是通过下面这个宏定义的:

(doom-modeline-def-segment matches
  "Docstring"
  (let ((meta (concat (doom-modeline--macro-recording)
                      (doom-modeline--anzu)
                      (doom-modeline--phi-search)
                      (doom-modeline--evil-substitute)
                      (doom-modeline--iedit)
                      (doom-modeline--symbol-overlay)
                      (doom-modeline--multiple-cursors))))
    (or (and (not (string-empty-p meta)) meta)
        (doom-modeline--buffer-size))))

我的 patch 代码没有什么特别,比如,就使用:

(psearch-patch doom-modeline-segment--matches
    (psearch-replace '`meta
                     '`meta_))

但最终的结果是:并没有重新定义 doom-modeline-segment--matches 这个函数,而是定义了一个新的函数 doom-modeline-segment--doom-modeline-segment--matches,而这,就是问题所在。

我明白了。

问题虽然已经明了,但是解决起来有点麻烦。

普通函数和 doom-modeline-def-segment 生成的函数是不一样的:

(psearch-patch--find-function 'doom-modeline-segment--foobar)
;; => (doom-modeline-def-segment foobar "Docstring" body...)
;;                    |             |         |        |
;;                    |             |         |        `--- [3]函数体
;;                    |             |         `------------ [2]文档
;;                    |             `---------------------- [1]函数名
;;                    `------------------------------------ [0]声明符

(psearch-patch--find-function 'foobar)
;; => (defun foobar nil "Docstring" body...)
;;       |     |     |       |        |
;;       |     |     |       |        `--- [4]函数体
;;       |     |     |       `------------ [3]文档
;;       |     |     `-------------------- [2]参数表
;;       |     `-------------------------- [1]函数名
;;       `-------------------------------- [0]声明符

从以上对比可以看出,doom-modeline-def-segment 定义函数是没有参数列表的,所以它的 Docstring 位置和 defun 定义的函数不一样:

而且这个不一样还doom-modeline-def-segment 独有的, 换个其它 xxx-def-yyy 有可能就不一样了。有的可能需要参数列表,有的可能不需要 Docstring。这就导致其生成的结构无法预料

需要有一种方法,可以让用户指定如何解析 def 生成的结构,这无疑会给用户增加负担。还要考虑到不改变 psearch-patch 的声明格式,不影响旧的 patch 代码,或许可以在外层套个 let:

(let ((psearch-patch-symbol-definition-pospec
       '((symbol-name . 1)
         (docstring . 2))))
  (psearch-patch doom-modeline-segment--matches
    (psearch-replace '`meta
                     '`meta_)))

让我再想一想,有可能指定一个 (let ((psarch-patch-docpos 2)) ...) 就够了。

1 个赞

真是很漂亮、精确和简洁的解释!没有认真读过前辈的代码,但是通过这个回答,学到了不少!

再次感谢这个很实用的工具!

同时,我有两个不太相关的小问题:

  1. 这部分的注释,是怎么排版成这种“图形化”的纯字符呢?有什么小工具吗?
  1. 我之所以会尝试使用如下“看似没用”的代码:

是因为,我 patch 了 doom-modeline--multiple-cursors 这个函数(它是用 defsubst 定义的),并且被 doom-modeline-segment--matches 调用。

但是仅仅做这个 patch 没有效果(是因为内联?还是 Doom Emacs 的黑魔法?),并且我发现,只要我重新求值一下 doom-modeline-segment--matches 的定义,就有效果了。(并且先 patch 和先重新求值皆可)

这才有了我上面这个“看似没用”的代码。但最终我是通过这个代码来强制重新求值的:

(eval (psearch-patch--find-function 'doom-modeline-segment--matches))

所以,前辈有遇到类似的情况吗?有相关原因和更漂亮的解决方案吗?

手打,没工具。

内联。

最新版本 Add support for macro-generated function · twlz0ne/psearch.el@b027a9d · GitHub 应该可以支持 doom-modeline-segment 函数了,用法:

(let ((psearch-patch-function-definition-docpos 2))
  (psearch-patch doom-modeline-segment--matches
    (psearch-replace '`MATCH-PATTERN
                     '`REPLACEMENT)))
1 个赞

嗯嗯,学到了!感谢!

我尝试这个版本发现,可以 patch defun 定义的函数,但是无法 patch defsubst 定义的函数呢(该提交之前,patch 功能是正常的)。

所以是不是把 defsubst 考虑进去比较好:

@@ -797,7 +797,7 @@
     `(let ((func-def (psearch-patch--find-function ',orig-func-spec)))
        (with-temp-buffer
          ;; Modifiy function name
-         (when (and (eq 'defun (nth 0 func-def)) (not (eq ',name (nth 1 func-def))))
+         (when (and (memq (nth 0 func-def) '(defun defsubst)) (not (eq ',name (nth 1 func-def))))
            (setcdr func-def (cons ',name (nthcdr 2 func-def))))
          (print func-def (current-buffer))
          ;; Modify docstring.
@@ -805,7 +805,7 @@
          (down-list)
          (forward-sexp
           (+ ,docpos (if (equal 'lambda (sexp-at-point)) 0
-                       (if (memq (sexp-at-point) '(defun cl-defgeneric cl-defmethod)) 1
+                       (if (memq (sexp-at-point) '(defun defsubst cl-defgeneric cl-defmethod)) 1
                          0))))
          (let ((str "[PATCHED]"))
            (goto-char (car (bounds-of-thing-at-point 'sexp)))

defubst 本身就算改了定义也没法反映到已经编译的函数里,没啥做的意义

是这样的,但是有时候确实需要改动一些内联函数,所以我会把某些受影响的函数重新求值一下(假设 FUN 调用了某个被 patch 的内联函数):

(eval (psearch-patch--find-function 'FUN))

能否提一个pr?同时记得修改一下头注释把你的名字加上去:

;; Last-Updated: <当下的时间戳>
;;           By: <你的名字>

嗯呢 已经提了,请看看有没有问题:-)

1 个赞

已合并。​​​​​

我发现使用 psearch-patch 之后,就无法再用 elisp-def 定位到函数之前的位置了。

所以我对 psearch-patch-define 做了一些修改:在重新评估 patch 之后的函数体前,先设置好之前的路径。这样 patch 之后,也能找到函数的位置。

(带来的另外一个好处是,可以一边修改 psearch-replace 的内容,一边反复执行 psearch-patch,这在之前直不行的,因为运行一次 psearch-patch,就找不到函数的位置了,也就拿不到原来的定义了)

@@ -817,7 +817,8 @@ See `psearch-patch' for explanation on arguments ORIG-FUNC-SPEC and PATCH-FORM."
          ;; Apply patch
          (goto-char (point-min))
          (if (progn ,@patch-form)
-             (eval-region (point-min) (point-max))
+             (let ((buffer-file-name (file-name-sans-extension (cdr (find-function-library ',name 'lisp-only t)))))
+               (eval-region (point-min) (point-max)))
            (signal 'psearch-patch-failed
                    (list ',orig-func-spec "PATCH-FORM not applied")))))))

请前辈看看有没有什么问题和建议?

感谢。

不过有几个测试没通过,我看看是怎么回事。

嗯嗯好的!我确实没有经过广泛的测试,目前我的几个使用场景倒是没出啥问题。

我自己尝试测试了一下 psearch-test 里面的代码,发现即使不引入我的修改,似乎也有测试不通过的例子?(在 psearch-test-cl-function-patch-1 aborted),是我打开方式不对吗? :saluting_face:

  1. 用 run-test.sh 脚本测试。
  2. 我做了一些改动,把错误消除了,你可以用它提一个 pr,记得改一下 Last-Updated: 时间并签上你的名字:
diff --git a/psearch.el b/psearch.el
index b77f659..6bbc683 100644
--- a/psearch.el
+++ b/psearch.el
@@ -817,7 +817,11 @@ See `psearch-patch' for explanation on arguments ORIG-FUNC-SPEC and PATCH-FORM."
          ;; Apply patch
          (goto-char (point-min))
          (if (progn ,@patch-form)
-             (eval-region (point-min) (point-max))
+             (let* ((lib (cdr (condition-case nil
+                                  (find-function-library ',name 'lisp-only t)
+                                (void-function nil))))
+                    (buffer-file-name (if lib (file-name-sans-extension lib))))
+               (eval-region (point-min) (point-max)))
            (signal 'psearch-patch-failed
                    (list ',orig-func-spec "PATCH-FORM not applied")))))))

好的,我刚刚创建了一个 PR,请前辈看看,有问题只管修改就行。

合并了。

有时候,patch 的函数在源文件中,并不是顶格的 :rofl:

比如 Doom Emacs 中就有很多的 hack 是这种形式: image 这个 doom-use-helpful-a 就不是顶格的。

如果我想通过 psearch-patch 对上面的函数进行 hack,就会报错:

原因是因为:psearch-patch--xref-function-def 这个函数只能定位到「函数定义位置的行首」(上图光标所在位置),这个时候 (bounds-of-thing-at-point 'sexp) 就会返回 nil,从而报错。

因此,建议:

diff --git a/psearch.el b/psearch.el
index 316c8b7..118d74c 100644
--- a/psearch.el
+++ b/psearch.el
@@ -683,6 +683,7 @@ Examples:
                (`(,buf . ,pos) (find-function-search-for-symbol fun type file)))
     (with-current-buffer buf
       (goto-char pos)
+      (skip-chars-forward " \t")
       (let ((bounds (bounds-of-thing-at-point 'sexp)))
         (read (buffer-substring-no-properties (car bounds) (cdr bounds)))))))

这样就可以跳过行首的空白字符。

请前辈看看,有问题没?

个人感觉 “[PATCHED]” 不是很显眼: image

不如换成 “<PATCHED> ”: image

当然只是建议,前辈不必太在意 :rofl: