-
-
Notifications
You must be signed in to change notification settings - Fork 9
/
affe-backend.el
202 lines (173 loc) · 7.8 KB
/
affe-backend.el
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
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
;;; affe-backend.el --- Affe backend -*- lexical-binding: t -*-
;; Copyright (C) 2021-2025 Daniel Mendler
;; This file is not part of GNU Emacs.
;; 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 <https://www.gnu.org/licenses/>.
;;; Commentary:
;; Affe backend
;;; Code:
(setq gc-cons-threshold 67108864
gc-cons-percentage 0.5)
(require 'server)
(defvar affe-backend--search-head (list nil))
(defvar affe-backend--search-tail affe-backend--search-head)
(defvar affe-backend--search-found 0)
(defvar affe-backend--search-limit 0)
(defvar affe-backend--search-regexps nil)
(defvar affe-backend--producer-head (list nil))
(defvar affe-backend--producer-tail affe-backend--producer-head)
(defvar affe-backend--producer-total 0)
(defvar affe-backend--producer-done nil)
(defvar affe-backend--producer-rest "")
(defvar affe-backend--client-rest "")
(defvar affe-backend--client nil)
(defvar affe-backend--restrict-regexp nil)
(defun affe-backend--send (expr)
"Send EXPR."
(process-send-string
affe-backend--client
(let ((print-escape-newlines t))
(concat (prin1-to-string expr) "\n"))))
(defun affe-backend--producer-filter (_ out)
"Process filter for the producer process receiving OUT string."
(let ((lines (split-string out "[\r\n]+")))
(if (not (cdr lines))
(setq affe-backend--producer-rest (concat affe-backend--producer-rest (car lines)))
(setcar lines (concat affe-backend--producer-rest (car lines)))
(let* ((len (length lines))
(last (nthcdr (- len 2) lines))
(rest (cadr last)))
(setcdr affe-backend--producer-tail lines)
(setcdr last nil)
(when affe-backend--restrict-regexp
(while lines
(let ((line (car lines)))
(when (string-match affe-backend--restrict-regexp line)
(setcar lines (match-string 1 line))
(add-text-properties 0 1 `(affe--prefix
,(substring line 0 (match-beginning 1))
affe--suffix
,(substring line (match-end 1)))
(car lines))))
(pop lines)))
(setq affe-backend--producer-rest rest
affe-backend--producer-total (+ affe-backend--producer-total len -1)
affe-backend--producer-tail last)))))
(defun affe-backend--producer-sentinel (_ status)
"Sentinel for the producer process, receiving STATUS."
(affe-backend--log "Sentinel: %s\n" status)
(with-current-buffer (get-buffer-create "*producer stderr*")
(affe-backend--log "Stderr:\n%s\n" (buffer-string)))
(setq affe-backend--producer-done t)
(unless (equal affe-backend--producer-rest "")
(setq affe-backend--producer-total (1+ affe-backend--producer-total)
affe-backend--producer-tail (setcdr affe-backend--producer-tail (list affe-backend--producer-rest)))))
(defun affe-backend--producer-start (cmd)
"Start backend CMD."
(affe-backend--log "Starting %S\n" cmd)
(make-process
:name (car cmd)
:noquery t
:command cmd
:connection-type 'pipe
:stderr "*producer stderr*"
:sentinel #'affe-backend--producer-sentinel
:filter #'affe-backend--producer-filter))
(defun affe-backend--server-filter (client out)
"Server filter function receiving CLIENT and OUT string."
(let ((lines (split-string out "\n")))
(if (not (cdr lines))
(setq affe-backend--client-rest (concat affe-backend--client-rest (car lines)))
(setcar lines (concat affe-backend--client-rest (car lines)))
(let ((last (last lines 2)))
(setq affe-backend--client-rest (cadr last))
(setcdr last nil))
(dolist (line lines)
(pcase (read line)
('exit (kill-emacs))
(`(search ,limit . ,regexps)
(affe-backend--append-producer)
(setq affe-backend--producer-head affe-backend--search-head
affe-backend--producer-tail affe-backend--search-tail
affe-backend--search-head (list nil)
affe-backend--search-tail affe-backend--search-head
affe-backend--search-limit limit
affe-backend--search-found 0
affe-backend--search-regexps regexps))
(`(start ,regexp . ,cmd)
(setq affe-backend--client client
affe-backend--restrict-regexp regexp)
(run-at-time 0.5 0.5 #'affe-backend--producer-refresh)
(run-at-time 0.1 0.1 #'affe-backend--search-refresh)
(affe-backend--producer-start cmd))))
(when (/= 0 affe-backend--search-limit)
(affe-backend--search)
(run-at-time 0.5 nil #'affe-backend--flush)))))
(defun affe-backend--log (&rest msg)
"Send log message MSG."
(affe-backend--send `(log ,(apply #'format msg))))
(defun affe-backend--flush ()
"Send a flush if no matching strings are found."
(when (= 0 affe-backend--search-found)
(affe-backend--send 'flush)))
(defun affe-backend--producer-refresh ()
"Refresh producer status."
(affe-backend--send
`(producer ,affe-backend--producer-total
,affe-backend--producer-done)))
(defun affe-backend--search-refresh ()
"Refresh search."
(when (/= 0 affe-backend--search-limit)
(affe-backend--search)))
(defun affe-backend--search-status ()
"Send status to the CLIENT."
(affe-backend--send `(search ,(/= 0 affe-backend--search-limit))))
(defun affe-backend--search-match-found (match)
"Called when matching string MATCH has been found."
(affe-backend--search-status)
(affe-backend--flush)
(affe-backend--send `(match ,(get-text-property 0 'affe--prefix match)
,(substring-no-properties match)
,(get-text-property 0 'affe--suffix match)))
(when (>= (setq affe-backend--search-found (1+ affe-backend--search-found))
affe-backend--search-limit)
(throw 'affe-backend--search-done nil))
nil)
(defun affe-backend--append-producer ()
"Append producer list to search list."
(when-let (head (cdr affe-backend--producer-head))
(setcdr affe-backend--search-tail head)
(setq affe-backend--search-tail affe-backend--producer-tail
affe-backend--producer-head (list nil)
affe-backend--producer-tail affe-backend--producer-head)))
(defun affe-backend--search ()
"Search and send matching lines to client."
(affe-backend--search-status)
(when-let (head (cdr affe-backend--producer-head))
(affe-backend--append-producer)
(catch 'affe-backend--search-done
(let ((completion-regexp-list affe-backend--search-regexps)
(completion-ignore-case t))
(all-completions "" head #'affe-backend--search-match-found))))
(when (or (>= affe-backend--search-found affe-backend--search-limit)
(and affe-backend--producer-done
(not (cdr affe-backend--producer-head))))
(setq affe-backend--search-limit 0))
(when (= 0 affe-backend--search-limit)
(affe-backend--flush))
(affe-backend--search-status))
(defun affe-backend--setup ()
"Setup backend server."
(set-process-coding-system server-process 'utf-8 'utf-8)
(set-process-filter server-process #'affe-backend--server-filter))
(add-hook 'emacs-startup-hook #'affe-backend--setup)
(provide 'affe-backend)
;;; affe-backend.el ends here