]>
Commit | Line | Data |
---|---|---|
d3372da9 | 1 | #!/usr/bin/guile -s |
2 | !# | |
3 | ||
4 | (use-modules (dolcon ui)) | |
5 | (use-modules (ice-9 pretty-print)) | |
6 | ||
7 | (define sr '()) | |
8 | (define lastsearch 0) | |
9 | (define info-searcheta 0) | |
10 | (define info-numavail 0) | |
11 | (define info-numreal 0) | |
12 | (define info-numtotal 0) | |
13 | (define lastparse 0) | |
14 | (define srchid -1) | |
15 | (define session '()) | |
16 | (define trans '()) | |
17 | (define dpeers '()) | |
18 | (define lastdl 0) | |
19 | ||
20 | (define (logf msg) | |
21 | (write-line msg (current-output-port)) | |
22 | (catch 'system-error (lambda () | |
23 | (fsync (current-output-port))) | |
24 | (lambda (key . err) #f)) | |
25 | ) | |
26 | ||
27 | (define (make-getopt opts optdesc) | |
28 | (let ((arg opts) (curpos 0) (rest '())) | |
29 | (lambda () | |
30 | (if (eq? arg '()) rest | |
31 | (let ((ret #f)) | |
32 | (while (not ret) | |
33 | (if (= curpos 0) | |
34 | (if (eq? (string-ref (car arg) 0) #\-) | |
35 | (set! curpos 1) | |
36 | (begin | |
37 | (set! rest (append rest (list (car arg)))) | |
38 | (set! arg (cdr arg)) | |
39 | (if (eq? arg '()) | |
40 | (set! ret #t))))) | |
41 | (if (> curpos 0) | |
42 | (if (< curpos (string-length (car arg))) | |
43 | (begin (set! ret (string-ref (car arg) curpos)) (set! curpos (+ curpos 1))) | |
44 | (begin (set! curpos 0) (set! arg (cdr arg)) (if (eq? arg '()) (set! ret #t)))))) | |
45 | (if (eq? ret #t) rest | |
46 | (let ((opt (string-index optdesc ret))) | |
47 | (if (eq? opt #f) (throw 'illegal-option ret) | |
48 | (if (and (< opt (- (string-length optdesc) 1)) (eq? (string-ref optdesc (+ opt 1)) #\:)) | |
49 | (let ((ret | |
50 | (cons ret (let ((optarg | |
51 | (if (< curpos (string-length (car arg))) | |
52 | (substring (car arg) curpos) | |
53 | (begin (set! arg (cdr arg)) (if (eq? arg '()) (throw 'requires-argument ret)) (car arg))))) | |
54 | (set! arg (cdr arg)) optarg)))) | |
55 | (set! curpos 0) | |
56 | ret) | |
57 | (list ret)))))))))) | |
58 | ||
59 | (define (ftime) | |
60 | (let ((ctime (gettimeofday))) | |
61 | (+ (car ctime) (/ (cdr ctime) 1000000)))) | |
62 | ||
63 | (define (wanttosearch) | |
64 | (> (- (current-time) lastsearch) | |
65 | (if (> (length trans) 0) 300 60)) | |
66 | ) | |
67 | ||
68 | (define defspeed '()) | |
69 | (let ((matchlist (list | |
70 | (cons (make-regexp "^[][{}() ]*BBB" regexp/icase) 100000)))) | |
71 | (set! defspeed | |
72 | (lambda (sr) | |
73 | (catch 'ret | |
74 | (lambda () | |
75 | (for-each (lambda (o) | |
76 | (if (regexp-exec (car o) (cadr (cdr (assoc 'peer sr)))) | |
77 | (throw 'ret (cdr o)))) | |
78 | matchlist) | |
79 | 15000) | |
80 | (lambda (sig ret) | |
81 | ret)) | |
82 | ))) | |
83 | ||
84 | (define (sr-less? sr1 sr2) | |
85 | (let ((s1 (if (cdr (assoc 'speed sr1)) (cdr (assoc 'speed sr1)) (defspeed sr1))) | |
86 | (s2 (if (cdr (assoc 'speed sr2)) (cdr (assoc 'speed sr2)) (defspeed sr2)))) | |
87 | (if (= s1 s2) | |
88 | (< (cdr (assoc 'resptime sr1)) (cdr (assoc 'resptime sr2))) | |
89 | (> s1 s2))) | |
90 | ) | |
91 | ||
92 | (define (srg-less? g1 g2) | |
93 | (or (> (length (cdr g1)) (length (cdr g2))) | |
94 | (and (= (length (cdr g1)) (length (cdr g2))) | |
95 | (> (car g1) (car g2)))) | |
96 | ) | |
97 | ||
98 | (define (gettrbysize size) | |
99 | (catch 'ret | |
100 | (lambda () | |
101 | (for-each (lambda (o) | |
102 | (if (= (cdr (assoc 'size (cdr o))) size) | |
103 | (throw 'ret (cdr o)))) | |
104 | trans) | |
105 | #f) | |
106 | (lambda (sig ret) | |
107 | ret)) | |
108 | ) | |
109 | ||
110 | (define (download sr) | |
111 | (let ((resp #f)) | |
112 | (let ((args (list "download" | |
113 | (car (cdr (assoc 'peer sr))) | |
114 | (cadr (cdr (assoc 'peer sr))) | |
115 | (cdr (assoc 'filename sr)) | |
116 | (cdr (assoc 'size sr))))) | |
3b0de0fd | 117 | (let ((hash (assoc 'hash sr))) |
118 | (if (and hash (not (equal? (cdr hash) ""))) (set! args (append args (list "hash" (cdr hash)))))) | |
d3372da9 | 119 | (let ((tag (assoc 'tag session))) |
120 | (if tag (set! args (append args (list "tag" (cdr tag)))))) | |
121 | (let ((uarg (assoc 'uarg session))) | |
122 | (if uarg (set! args (append args (list "user" (cdr uarg)))))) | |
123 | (set! resp (apply dc-ecmd-assert 200 args))) | |
124 | (let ((id (car (dc-intresp resp)))) | |
125 | (set! trans | |
126 | (cons (cons id (list (assoc 'size sr) | |
127 | (assoc 'peer sr) | |
128 | (assoc 'filename sr) | |
129 | (assoc 'resptime sr) | |
130 | '(curpos . 0) | |
131 | '(state . wait) | |
132 | '(curspeed . #f) | |
133 | '(lastpos . 0) | |
134 | (cons 'id id) | |
135 | (cons 'lasttime (current-time)) | |
136 | (cons 'lastprog (current-time)))) | |
137 | trans)) | |
138 | (logf (string-append "downloading " | |
139 | (cdr (assoc 'filename sr)) | |
140 | " from " | |
141 | (cadr (cdr (assoc 'peer sr))) | |
142 | ", " | |
143 | (number->string (cdr (assoc 'size sr))) | |
144 | " bytes (id " | |
145 | (number->string id) | |
146 | ", " | |
147 | (number->string (cdr (assoc 'slots sr))) | |
148 | " slots), timing out in " | |
149 | (number->string (max 10 (* (cdr (assoc 'resptime sr)) 2))) | |
150 | " seconds")))) | |
151 | (set! lastdl (current-time)) | |
152 | ) | |
153 | ||
154 | (define (disablepeer peer) | |
155 | (let ((newglist '()) (numrem 0)) | |
156 | (for-each (lambda (g) | |
157 | (let ((newlist '())) | |
158 | (for-each (lambda (o) | |
159 | (if (not (equal? (cdr (assoc 'peer o)) peer)) | |
160 | (set! newlist (cons o newlist)) | |
161 | (set! numrem (+ numrem 1)))) | |
162 | (cdr g)) | |
163 | (if (not (eq? newlist '())) | |
164 | (set! newglist (cons (cons (car g) (sort newlist sr-less?)) newglist))))) | |
165 | sr) | |
166 | (set! sr (sort newglist srg-less?)) | |
167 | (logf (string-append "disabled " (cadr peer) " and removed " (number->string numrem) " search results"))) | |
168 | (let* ((dpa (assoc peer dpeers)) (dp (and (pair? dpa) (cdr dpa)))) | |
169 | (if dp | |
170 | (set-cdr! (assoc 'time dp) (current-time)) | |
171 | (set! dpeers (cons (cons peer (list (cons 'time (current-time)) | |
172 | (cons 'peer peer))) | |
173 | dpeers)))) | |
174 | ) | |
175 | ||
176 | (define (checktrans) | |
177 | (let ((time (current-time))) | |
178 | (for-each (lambda (o) | |
179 | (if (and (memq (cdr (assoc 'state (cdr o))) '(wait hs)) | |
180 | (> (- time (cdr (assoc 'lastprog (cdr o)))) (max 10 (* (cdr (assoc 'resptime (cdr o))) 2)))) | |
181 | (begin (logf (string-append "transfer " (number->string (car o)) " timing out")) | |
182 | (dc-ecmd-assert 200 "cancel" (car o)) | |
183 | (disablepeer (cdr (assoc 'peer (cdr o)))) | |
ecd6e23d | 184 | (set! trans (assq-remove! trans (car o))) |
185 | (write-info-file))) | |
d3372da9 | 186 | (if (and (eq? (cdr (assoc 'state (cdr o))) 'main) |
187 | (> (- time (cdr (assoc 'lastprog (cdr o)))) 60)) | |
188 | (begin (logf (string-append "transfer " (number->string (car o)) " seems to have stalled")) | |
189 | (dc-ecmd-assert 200 "cancel" (car o)) | |
ecd6e23d | 190 | (set! trans (assq-remove! trans (car o))) |
191 | (write-info-file))) | |
d3372da9 | 192 | (if (and (eq? (cdr (assoc 'state (cdr o))) 'main) |
193 | (> (- (cdr (assoc 'lastprog (cdr o))) (cdr (assoc 'lasttime (cdr o)))) 20)) | |
194 | (begin (set-cdr! (assoc 'curspeed (cdr o)) | |
195 | (/ (- (cdr (assoc 'curpos (cdr o))) (cdr (assoc 'lastpos (cdr o)))) | |
196 | (- (cdr (assoc 'lastprog (cdr o))) (cdr (assoc 'lasttime (cdr o)))))) | |
197 | (set-cdr! (assoc 'lastpos (cdr o)) (cdr (assoc 'curpos (cdr o)))) | |
ecd6e23d | 198 | (set-cdr! (assoc 'lasttime (cdr o)) (cdr (assoc 'lastprog (cdr o)))) |
199 | (write-info-file)))) | |
d3372da9 | 200 | trans)) |
201 | ) | |
202 | ||
203 | (define (write-info-file) | |
204 | (if (assoc 'info-file session) | |
205 | (let ((op (open-output-file (cdr (assoc 'info-file session))))) | |
206 | (write (list (cons 'numdl (length trans)) | |
207 | (cons 'lastdl lastdl) | |
208 | (cons 'availsr info-numavail) | |
209 | (cons 'realsr info-numreal) | |
210 | (cons 'totalsr info-numtotal) | |
211 | (cons 'lastsrch lastsearch) | |
212 | (cons 'srcheta info-searcheta)) | |
213 | op) | |
214 | (newline op) | |
215 | (close-port op)))) | |
216 | ||
217 | (define (parseresults) | |
218 | (logf (string-append "entering parseresults with " | |
219 | (number->string | |
220 | (apply + (map (lambda (o) (length (cdr o))) sr))) | |
221 | " results in " | |
222 | (number->string (length sr)) | |
223 | " sizes")) | |
224 | (let ((retval #t) (numreal 0) (numtotal 0) (numavail 0)) | |
225 | (catch 'ret | |
226 | (lambda () | |
227 | (and (eq? sr '()) (throw 'ret #f)) | |
228 | (let ((numrem 0) (countrem 0) (newglist '())) | |
229 | (for-each (lambda (g) | |
230 | (let ((newlist '())) | |
231 | (for-each (lambda (o) | |
232 | (if (< (- (current-time) (cdr (assoc 'recvtime o))) 300) | |
233 | (set! newlist (cons o newlist)) | |
234 | (set! countrem (+ countrem 1)))) | |
235 | (cdr g)) | |
236 | (if (> (length newlist) 0) | |
237 | (set! newglist (cons (cons (car g) (sort newlist sr-less?)) newglist)) | |
238 | (set! numrem (+ numrem 1))))) | |
239 | sr) | |
240 | (set! sr (sort newglist srg-less?)) | |
241 | (if (> countrem 0) | |
242 | (logf (string-append "removed " (number->string countrem) " time-outed results and " (number->string numrem) " entire sizes")))) | |
243 | (let ((numrem 0) (newlist '())) | |
244 | (for-each (lambda (o) | |
245 | (if (> (- (current-time) (cdr (assoc 'time o))) 1800) | |
246 | (set! numrem (+ numrem 1)) | |
247 | (set! newlist (cons o newlist)))) | |
248 | dpeers) | |
249 | (set! dpeers newlist) | |
250 | (logf (string-append "re-enabled " (number->string numrem) " disabled users"))) | |
251 | (let ((numrem 0) (countrem 0) (newglist '())) | |
252 | (for-each (lambda (g) | |
253 | (let ((newlist '())) | |
254 | (for-each (lambda (o) | |
255 | (if (not (assoc (cdr (assoc 'peer o)) dpeers)) | |
256 | (set! newlist (cons o newlist)) | |
257 | (set! countrem (+ countrem 1)))) | |
258 | (cdr g)) | |
259 | (if (> (length newlist) 0) | |
260 | (set! newglist (cons (cons (car g) (sort newlist sr-less?)) newglist)) | |
261 | (set! numrem (+ numrem 1))))) | |
262 | sr) | |
263 | (set! sr (sort newglist srg-less?)) | |
264 | (if (> countrem 0) | |
265 | (logf (string-append "removed " (number->string countrem) " results with disabled peers and " (number->string numrem) " entire sizes")))) | |
266 | (and (eq? sr '()) (throw 'ret #f)) | |
267 | (set! numtotal (apply + (map (lambda (o) (length (cdr o))) sr))) | |
268 | (let* ((maxsize (apply max (map (lambda (o) (length (cdr o))) sr))) | |
269 | (minsize (/ maxsize 3))) | |
270 | (let ((numrem 0) (countrem 0)) | |
271 | (for-each (lambda (o) (if (< (length (cdr o)) minsize) | |
272 | (begin (set! countrem (+ countrem (length (cdr o)))) | |
273 | (set! numrem (+ numrem 1))))) | |
274 | sr) | |
275 | (if (> countrem 0) | |
276 | (logf (string-append "will disregard " (number->string countrem) " results from " (number->string numrem) " sizes due to popularity lack"))) | |
277 | (set! numreal (- numtotal countrem))) | |
278 | (let ((numrem 0) (numrrem 0)) | |
279 | (for-each (lambda (g) | |
280 | (for-each (lambda (o) | |
281 | (if (< (cdr (assoc 'slots o)) 1) | |
282 | (begin (set! numrem (+ numrem 1)) | |
283 | (if (>= (length (cdr g)) minsize) | |
284 | (set! numrrem (+ numrrem 1)))))) | |
285 | (cdr g))) | |
286 | sr) | |
287 | (if (> numrem 0) | |
288 | (logf (string-append (number->string numrem) " results had no slots"))) | |
289 | (set! numavail (- numreal numrrem))) | |
290 | (for-each (lambda (g) | |
291 | (if (>= (length (cdr g)) minsize) | |
292 | (catch 'found | |
293 | (lambda () | |
294 | (for-each (lambda (o) | |
295 | (and (> (cdr (assoc 'slots o)) 0) | |
296 | (throw 'found o))) | |
297 | (cdr g))) | |
298 | (lambda (sig sr) | |
299 | (let ((tr (gettrbysize (cdr (assoc 'size sr))))) | |
300 | (if (not tr) | |
301 | (if (< (length trans) (cdr (assoc 'maxtrans session))) | |
302 | (download sr)) | |
303 | (if (and (cdr (assoc 'curspeed tr)) | |
304 | (not (equal? (cdr (assoc 'peer sr)) (cdr (assoc 'peer tr)))) | |
305 | (> (- (or (cdr (assoc 'speed sr)) (defspeed sr)) (cdr (assoc 'curspeed tr))) 10000)) | |
306 | (begin (logf (string-append "abandoning transfer " | |
307 | (number->string (cdr (assoc 'id tr))) | |
308 | " for possible faster sender")) | |
309 | (dc-ecmd-assert 200 "cancel" (cdr (assoc 'id tr))) | |
310 | (set! trans (assq-remove! trans (cdr (assoc 'id tr)))) | |
311 | (download sr))))))))) | |
312 | sr) | |
313 | ) | |
314 | ) | |
315 | (lambda (sig ret) | |
316 | (set! retval ret) | |
317 | )) | |
318 | (set! info-numavail numavail) | |
319 | (set! info-numreal numreal) | |
320 | (set! info-numtotal numtotal) | |
ecd6e23d | 321 | (write-info-file) |
d3372da9 | 322 | retval) |
323 | ) | |
324 | ||
3b0de0fd | 325 | (define (handlesr filename fnet peer size slots resptime hash) |
d3372da9 | 326 | (let ((cl (or (assoc size sr) |
327 | (let ((newp (cons size '()))) (set! sr (append sr (list newp))) newp))) | |
328 | (newsr (list | |
329 | (cons 'filename filename) | |
330 | (cons 'peer (list fnet peer)) | |
331 | (cons 'size size) | |
332 | (cons 'slots slots) | |
333 | (cons 'resptime resptime) | |
334 | (cons 'speed (getspeed peer)) | |
3b0de0fd | 335 | (cons 'hash hash) |
d3372da9 | 336 | (cons 'recvtime (current-time)) |
337 | (cons 'dis #f))) | |
338 | (newlist '())) | |
339 | (for-each (lambda (o) (if (not (and (equal? (cdr (assoc 'filename o)) filename) | |
340 | (equal? (cdr (assoc 'peer o)) (list fnet peer)))) | |
341 | (set! newlist (cons o newlist)))) | |
342 | (cdr cl)) | |
343 | (set-cdr! cl (sort (cons newsr newlist) sr-less?)) | |
344 | ) | |
345 | ) | |
346 | ||
347 | ; XXX: Redefine to go through the server, once that is implemented | |
348 | (define (getspeed username) | |
349 | (catch 'system-error | |
350 | (lambda () | |
351 | (let* ((port (open-input-file (string-append (getenv "HOME") "/dc/users/" username))) (avg 0) (numdls (string->number (read-line port))) (max (string->number (read-line port))) (numents (string->number (read-line port)))) | |
352 | (do ((i 0 (+ i 1))) ((= i numents) (close-port port) (/ avg numents)) (set! avg (+ avg (string->number (read-line port))))) | |
353 | )) | |
354 | (lambda args | |
355 | #f | |
356 | ) | |
357 | ) | |
358 | ) | |
359 | ||
360 | (define (validate-session session) | |
361 | (catch 'wrong-type-arg | |
362 | (lambda () | |
363 | (and | |
364 | (assoc 'sexpr session) | |
365 | (assoc 'prio session) | |
366 | (assoc 'maxtrans session) | |
367 | #t | |
368 | ) | |
369 | ) | |
370 | (lambda (key . args) | |
371 | (display "Session data is not an a-list\n" (current-error-port)) | |
372 | #f) | |
373 | ) | |
374 | ) | |
375 | ||
376 | (define (autodl-main args) | |
377 | (let ((dc-server #f) (done #f) (retval 0)) | |
378 | (let ((getopt (make-getopt (cdr args) "hs:S:e:p:t:a:I:")) (arg #f)) | |
379 | (do ((arg (getopt) (getopt))) ((not (and (pair? arg) (char? (car arg)))) (set! args arg)) | |
380 | (cond ((eq? (car arg) #\h) | |
381 | (begin (display "usage: autodl [-s server] -S sessfile\n" (current-error-port)) | |
382 | (display " autodl [-s server] -e search-expression [-p prio] [-t tag] [-a userarg]\n" (current-error-port)) | |
383 | (display " autodl [-s server]\n" (current-error-port)) | |
384 | (display " autodl -h\n" (current-error-port)) | |
385 | (exit 0))) | |
386 | ((eq? (car arg) #\s) | |
387 | (set! dc-server (cdr arg))) | |
388 | ((eq? (car arg) #\S) | |
389 | (let ((port (open-file (cdr arg)))) (set! session (read port)) (close-port port))) | |
390 | ((eq? (car arg) #\p) | |
391 | (let ((c (assoc 'prio session))) | |
392 | (if c (set-cdr! c (cdr arg)) | |
393 | (set! session (cons (cons 'prio (cdr arg)) session))))) | |
394 | ((eq? (car arg) #\t) | |
395 | (let ((c (assoc 'tag session))) | |
396 | (if c (set-cdr! c (cdr arg)) | |
397 | (set! session (cons (cons 'tag (cdr arg)) session))))) | |
398 | ((eq? (car arg) #\a) | |
399 | (let ((c (assoc 'uarg session))) | |
400 | (if c (set-cdr! c (cdr arg)) | |
401 | (set! session (cons (cons 'uarg (cdr arg)) session))))) | |
402 | ((eq? (car arg) #\I) | |
403 | (let ((c (assoc 'info-file session))) | |
404 | (if c (set-cdr! c (cdr arg)) | |
405 | (set! session (cons (cons 'info-file (cdr arg)) session))))) | |
406 | ((eq? (car arg) #\e) | |
407 | (set! session (cons (cons 'sexpr (dc-lexsexpr (cdr arg))) session))) | |
408 | ) | |
409 | ) | |
410 | ) | |
411 | (if (eq? session '()) (begin (if (isatty? (current-input-port)) (display "Enter session data (s-expr):\n" (current-error-port))) (set! session (read)))) | |
412 | (if (not (assoc 'prio session)) | |
413 | (set! session (cons '(prio . 10) session))) | |
414 | (if (not (assoc 'maxtrans session)) | |
415 | (set! session (cons '(maxtrans . 1) session))) | |
416 | (if (not (validate-session session)) (begin (display "Invalid session!\n" (current-error-port)) (exit 1))) | |
417 | (if (not dc-server) (set! dc-server (getenv "DCSERVER"))) | |
418 | (if (not dc-server) (set! dc-server "localhost")) | |
419 | (catch 'system-error | |
420 | (lambda () | |
421 | (dc-c&l #t dc-server #t)) | |
422 | (lambda (key . args) | |
423 | (logf (string-append "could not connect to server: " (apply format #f (cadr args) (caddr args)))) | |
424 | (exit 2))) | |
425 | (dc-ecmd-assert 200 "notify" "all" "on") | |
426 | (for-each (lambda (sig) (sigaction sig (lambda (sig) (throw 'sig sig)))) (list SIGINT SIGTERM SIGHUP)) | |
427 | (catch 'sig | |
428 | (lambda () | |
429 | (while #t | |
430 | (if (and (not (= lastsearch -1)) (wanttosearch)) | |
431 | (begin | |
432 | (if (not (= srchid -1)) | |
433 | (dc-ecmd "cansrch" srchid)) | |
434 | (let* ((resp (apply dc-ecmd-assert (append (list '(200 501 509) "search" "prio" (number->string (cdr (assoc 'prio session))) "all") (cdr (assoc 'sexpr session))))) | |
435 | (ires (dc-intresp resp)) | |
436 | (eres (dc-extract resp))) | |
437 | (case (cdr (assoc 'code eres)) | |
438 | ((200) | |
439 | (begin (set! srchid (car ires)) | |
440 | (logf (string-append "search scheduled in " (number->string (cadr ires)) " seconds (id " (number->string srchid) ")")) | |
441 | (set! info-searcheta (+ (current-time) (cadr ires))) | |
ecd6e23d | 442 | (set! lastsearch -1) |
443 | (write-info-file))) | |
d3372da9 | 444 | ((501) |
445 | (begin (set! srchid -1) | |
446 | (logf (string-append "no fnetnodes available to search on")) | |
447 | (set! lastsearch (current-time)))) | |
448 | ((509) | |
449 | (begin (logf "illegal search expression") | |
450 | (set! done #t) | |
451 | (set! retval 3) | |
452 | (throw 'sig 0))))))) | |
453 | (checktrans) | |
454 | (if (> (- (current-time) lastparse) 20) | |
455 | (begin (parseresults) | |
456 | (set! lastparse (current-time)))) | |
d3372da9 | 457 | (dc-select 10000) |
458 | (while (let ((resp (dc-getresp))) | |
459 | (if resp | |
460 | (begin | |
461 | (let* ((er (dc-extract resp)) (code (cdr (assoc 'code er))) (cmd (cdr (assoc 'cmd er)))) | |
462 | (cond | |
463 | ((equal? cmd ".notify") | |
464 | (case code | |
465 | ((611) ; Transfer state change | |
466 | (let ((ires (dc-intresp resp)) (tr #f)) | |
467 | (if (and ires (assoc (car ires) trans)) | |
468 | (begin (set! tr (cdr (assoc (car ires) trans))) | |
469 | (set-cdr! (assoc 'state tr) | |
470 | (cdr (assoc (cadr ires) '((0 . wait) (1 . hs) (2 . main) (3 . done))))) | |
471 | (set-cdr! (assoc 'lastprog tr) (current-time)))))) | |
472 | ((614) ; Transfer error | |
473 | (let ((ires (dc-intresp resp))) | |
474 | (if (and ires (assoc (car ires) trans)) | |
475 | (begin (logf (string-append "transfer " (number->string (car ires)) " encountered error " (number->string (cadr ires)))) | |
476 | (dc-ecmd-assert 200 "cancel" (car ires)) | |
477 | (let ((tr (cdr (assoc (car ires) trans)))) | |
478 | (disablepeer (cdr (assoc 'peer tr)))) | |
479 | (set! trans (assq-remove! trans (car ires))))))) | |
480 | ((615) ; Transfer progress | |
481 | (let ((ires (dc-intresp resp)) (tr #f)) | |
482 | (if (and ires (assoc (car ires) trans)) | |
483 | (begin (set! tr (cdr (assoc (car ires) trans))) | |
484 | (set-cdr! (assoc 'curpos tr) (cadr ires)) | |
485 | (set-cdr! (assoc 'lastprog tr) (current-time)))))) | |
486 | ((617) ; Transfer destroyed | |
487 | (let* ((ires (dc-intresp resp)) (tr (and ires (assoc (car ires) trans)))) | |
488 | (if tr | |
489 | (begin (if (eq? (cdr (assoc 'state (cdr tr))) 'done) | |
490 | (begin (logf (string-append "transfer " (number->string (car ires)) " done")) | |
491 | (set! trans (assq-remove! trans (car ires))) | |
492 | (set! done #t) | |
493 | (throw 'sig 0)) | |
494 | (begin (logf (string-append "transfer " (number->string (car ires)) " disappeared")) | |
495 | (set! trans (assq-remove! trans (car ires))))))))) | |
496 | ((620) ; Search rescheduled | |
497 | (let ((ires (dc-intresp resp))) | |
498 | (if (and ires (= (car ires) srchid)) | |
499 | (begin (set! info-searcheta (+ (current-time) (cadr ires))) | |
500 | (logf (string-append "search rescheduled to T+" (number->string (cadr ires)))))))) | |
501 | ((621) ; Search committed | |
502 | (let ((ires (dc-intresp resp))) | |
503 | (if (and ires (= (car ires) srchid)) | |
504 | (begin (logf "search committed") | |
505 | (set! info-searcheta 0) | |
506 | (set! lastsearch (current-time)))))) | |
507 | ((622) ; Search result | |
508 | (let ((ires (list->vector (dc-intresp resp)))) | |
3b0de0fd | 509 | (if (and ires (= (vector-ref ires 0) srchid)) (apply handlesr (map (lambda (n) (vector-ref ires n)) '(1 2 3 4 5 7 8)))))) |
d3372da9 | 510 | |
511 | ) | |
512 | ) | |
513 | ||
514 | ) | |
515 | ) | |
516 | #t) | |
517 | #f) | |
518 | ) | |
519 | #t | |
520 | ) | |
521 | ) | |
522 | ) | |
523 | (lambda (key sig) | |
524 | (logf (string-append "interrupted by signal " (number->string sig))) | |
525 | (if (not done) | |
526 | (set! retval 1))) | |
527 | ) | |
528 | (logf "quitting...") | |
529 | (catch 'sig | |
530 | (lambda () | |
531 | (if (dc-connected) | |
532 | (begin (for-each (lambda (o) | |
533 | (dc-qcmd (list "cancel" (car o)))) | |
534 | trans) | |
535 | (if (assoc 'info-file session) | |
536 | (catch 'system-error | |
537 | (lambda () | |
538 | (delete-file (cdr (assoc 'info-file session)))) | |
539 | (lambda (key . args) #t))) | |
540 | (if (and done (assoc 'tag session)) | |
541 | (dc-qcmd (list "filtercmd" "rmtag" (cdr (assoc 'tag session))))) | |
542 | (if (not (= srchid -1)) | |
543 | (dc-qcmd (list "cansrch" srchid))) | |
544 | (dc-qcmd '("quit")) | |
545 | (while (dc-connected) (dc-select)) | |
546 | ))) | |
547 | (lambda (key sig) | |
548 | (logf "forcing quit"))) | |
549 | (exit retval) | |
550 | ) | |
551 | ) | |
552 | ||
553 | (setlocale LC_ALL "") | |
554 | (autodl-main (command-line)) |