a5ba60846840f6c0efef35d6f8bd3a3f66d51fc1
[riece] / lisp / riece-ndcc.el
1 ;;; riece-ndcc.el --- elisp native DCC add-on
2 ;; Copyright (C) 1998-2003 Daiki Ueno
3
4 ;; Author: Daiki Ueno <ueno@unixuser.org>
5 ;; Keywords: IRC, riece
6
7 ;; This file is part of Riece.
8
9 ;; This program is free software; you can redistribute it and/or modify
10 ;; it under the terms of the GNU General Public License as published by
11 ;; the Free Software Foundation; either version 2, or (at your option)
12 ;; any later version.
13
14 ;; This program is distributed in the hope that it will be useful,
15 ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
16 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17 ;; GNU General Public License for more details.
18
19 ;; You should have received a copy of the GNU General Public License
20 ;; along with GNU Emacs; see the file COPYING.  If not, write to the
21 ;; Free Software Foundation, Inc., 59 Temple Place - Suite 330,
22 ;; Boston, MA 02111-1307, USA.
23
24 ;;; Code:
25
26 (require 'calc)
27
28 (defgroup riece-ndcc nil
29   "Elisp native DCC implementation"
30   :prefix "riece-"
31   :group 'riece)
32
33 (defcustom riece-ndcc-server-address "127.0.0.1"
34   "Local address of the DCC server.
35 Only used for sending files."
36   :type 'vector
37   :group 'riece-ndcc)
38
39 (defvar riece-ndcc-requests nil)
40
41 (defvar riece-ndcc-request-user nil)
42 (defvar riece-ndcc-request-size nil)
43
44 (defun riece-ndcc-encode-address (address)
45   (unless (string-match
46            "^\\([0-9]+\\)\\.\\([0-9]+\\)\\.\\([0-9]+\\)\\.\\([0-9]+\\)$"
47            address)
48     (error "% is not an IP address" address))
49   (let ((calc-number-radix 10))
50     (calc-eval (format "%s * (2 ** 24) + %s * (2 **16) + %s * (2 ** 8) + %s"
51                        (match-string 1 address)
52                        (match-string 2 address)
53                        (match-string 3 address)
54                        (match-string 4 address)))))
55
56 (defun riece-ndcc-decode-address (address)
57   (format "%d.%d.%d.%d"
58           (floor (string-to-number
59                   (calc-eval (format "(%s / (2 ** 24)) %% 256" address))))
60           (floor (string-to-number
61                   (calc-eval (format "(%s / (2 ** 16)) %% 256" address))))
62           (floor (string-to-number
63                   (calc-eval (format "(%s / (2 ** 8)) %% 256" address))))
64           (floor (string-to-number
65                   (calc-eval (format "%s %% 256" address))))))
66
67 (defun riece-ndcc-server-sentinel (process status)
68   (when (string-match "^open from " status)
69     (let ((parent-name
70            (if (string-match " <[^>]+>$" (process-name process))
71                (substring (process-name process) 0 (match-beginning 0)))))
72       (save-excursion
73         (set-buffer (process-buffer (get-process parent-name)))
74         (delete-process parent-name)
75         (goto-char (point-min))
76         (while (not (eobp))
77           (process-send-region process
78                                (point)
79                                (goto-char (min (+ 1024 (point)) (point-max))))
80           (message "Sending %s...(%d/%d)"
81                    (buffer-file-name) (1- (point)) (buffer-size)))
82         (message "Sending %s...done"
83                  (buffer-file-name))))
84     (kill-buffer (process-buffer process))
85     (delete-process process)))
86
87 (defun riece-command-dcc-send (user file)
88   (interactive
89    (let ((completion-ignore-case t))
90      (unless riece-ndcc-server-address
91        (error "Set riece-ndcc-server-address to your host"))
92      (list (completing-read
93             "User: "
94             (mapcar #'list (riece-get-users-on-server)))
95            (expand-file-name (read-file-name "File: ")))))
96   (let* (selective-display
97          (coding-system-for-read 'binary)
98          format-alist
99          jka-compr-compression-info-list
100          (buffer (find-file-noselect file))
101          process)
102     (with-current-buffer buffer         ;To throw an error when the
103       (setq buffer-read-only t))        ;process has input.
104     (setq process (make-network-process :name "DCC" :buffer buffer
105                                         :host riece-ndcc-server-address
106                                         :server t :service t
107                                         :coding 'binary
108                                         :sentinel 'riece-ndcc-server-sentinel))
109     (riece-send-string
110      (format "PRIVMSG %s :\1DCC SEND %s %s %d %d\1\r\n"
111              user (file-name-nondirectory file)
112              (riece-ndcc-encode-address riece-ndcc-server-address)
113              (nth 1 (process-contact process))
114              (nth 7 (file-attributes file))))))
115
116 (defun riece-ndcc-filter (process input)
117   (save-excursion
118     (set-buffer (process-buffer process))
119     (goto-char (point-max))
120     (insert input)
121     (message "Receiving %s from %s...(%d/%d)"
122              (file-name-nondirectory buffer-file-name)
123              riece-ndcc-request-user
124              (1- (point))
125              riece-ndcc-request-size)))
126
127 (defun riece-ndcc-sentinel (process status)
128   (save-excursion
129     (set-buffer (process-buffer process))
130     (unless (= (buffer-size) riece-ndcc-request-size)
131       (error "Premature end of file"))
132     (message "Receiving %s from %s...done"
133              (file-name-nondirectory buffer-file-name)
134              riece-ndcc-request-user)
135     (let ((coding-system-for-write 'binary))
136       (save-buffer))))
137
138 (defun riece-command-dcc-receive (request file)
139   (interactive
140    (progn
141      (unless riece-ndcc-requests
142        (error "No request"))
143      (list
144       (if (= (length riece-ndcc-requests) 1)
145           (car riece-ndcc-requests)
146         (with-output-to-temp-buffer "*Help*"
147           (let ((requests riece-ndcc-requests)
148                 (index 1))
149             (while requests
150               (princ (format "%2d: %s %s (%d bytes)\n"
151                              index
152                              (car (car requests))
153                              (nth 1 (car requests))
154                              (nth 4 (car requests))))
155               (setq index (1+ index)
156                     requests (cdr requests)))))
157         (let ((number (read-string "Request#: ")))
158           (unless (string-match "^[0-9]+$" number)
159             (error "Not a number"))
160           (if (or (> (setq number (string-to-number number))
161                      (length riece-ndcc-requests))
162                   (< number 1))
163               (error "Invalid number"))
164           (nth (1- number) riece-ndcc-requests)))
165       (expand-file-name (read-file-name "Save as: ")))))
166   (let* (selective-display
167          (coding-system-for-read 'binary)
168          (coding-system-for-write 'binary)
169          (process (open-network-stream
170                    "DCC" " *DCC*"
171                    (riece-ndcc-decode-address (nth 2 request))
172                    (nth 3 request))))
173     (setq riece-rdcc-requests (delq request riece-rdcc-requests))
174     (with-current-buffer (process-buffer process)
175       (set-buffer-multibyte nil)
176       (buffer-disable-undo)
177       (setq buffer-file-name file)
178       (make-local-variable 'riece-ndcc-request-user)
179       (setq riece-ndcc-request-user (car request))
180       (make-local-variable 'riece-ndcc-request-size)
181       (setq riece-ndcc-request-size (nth 4 request)))
182     (set-process-filter process #'riece-ndcc-filter)
183     (set-process-sentinel process #'riece-ndcc-sentinel)))
184
185 (defun riece-handle-dcc-request (prefix target message)
186   (let ((case-fold-search t))
187     (when (string-match
188            "SEND \\([^ ]+\\) \\([^ ]+\\) \\([^ ]+\\) \\([^ ]+\\)"
189            message)
190       (let ((file (match-string 1 message))
191             (address (match-string 2 message))
192             (port (string-to-number (match-string 3 message)))
193             (size (string-to-number (match-string 4 message)))
194             (buffer (if (riece-channel-p target)
195                         (cdr (riece-identity-assoc-no-server
196                               (riece-make-identity target)
197                               riece-channel-buffer-alist))))
198             (user (riece-prefix-nickname prefix)))
199         (setq riece-ndcc-requests
200               (cons (list user file address port size)
201                     riece-ndcc-requests))
202         (riece-insert-change buffer (format "DCC SEND from %s\n" user))
203         (riece-insert-change
204          (if (and riece-channel-buffer-mode
205                   (not (eq buffer riece-channel-buffer)))
206              (list riece-dialogue-buffer riece-others-buffer)
207            riece-dialogue-buffer)
208          (concat
209           (riece-concat-server-name
210            (format "DCC SEND from %s (%s) to %s"
211                    user
212                    (riece-strip-user-at-host
213                     (riece-prefix-user-at-host prefix))
214                    target))
215           "\n")))
216       t)))
217
218 (defun riece-ndcc-requires ()
219   '(riece-ctcp))
220
221 (defvar riece-dialogue-mode-map)
222 (defun riece-ndcc-insinuate ()
223   (add-hook 'riece-ctcp-dcc-request-hook 'riece-handle-dcc-request)
224   (define-key riece-dialogue-mode-map "\C-ds" 'riece-command-dcc-send)
225   (define-key riece-dialogue-mode-map "\C-dr" 'riece-command-dcc-receive))
226
227 (provide 'riece-ndcc)
228
229 ;;; riece-ndcc.el ends here