1 ;;; riece-ndcc.el --- elisp native DCC add-on
2 ;; Copyright (C) 1998-2003 Daiki Ueno
4 ;; Author: Daiki Ueno <ueno@unixuser.org>
5 ;; Keywords: IRC, riece
7 ;; This file is part of Riece.
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)
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.
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.
28 (defgroup riece-ndcc nil
29 "Elisp native DCC implementation"
33 (defcustom riece-ndcc-server-address "127.0.0.1"
34 "Local address of the DCC server.
35 Only used for sending files."
39 (defvar riece-ndcc-requests nil)
41 (defvar riece-ndcc-request-user nil)
42 (defvar riece-ndcc-request-size nil)
44 (defun riece-ndcc-encode-address (address)
46 "^\\([0-9]+\\)\\.\\([0-9]+\\)\\.\\([0-9]+\\)\\.\\([0-9]+\\)$"
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)))))
56 (defun riece-ndcc-decode-address (address)
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))))))
67 (defun riece-ndcc-server-sentinel (process status)
68 (when (string-match "^open from " status)
70 (if (string-match " <[^>]+>$" (process-name process))
71 (substring (process-name process) 0 (match-beginning 0)))))
73 (set-buffer (process-buffer (get-process parent-name)))
74 (delete-process parent-name)
75 (goto-char (point-min))
77 (process-send-region process
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"
84 (kill-buffer (process-buffer process))
85 (delete-process process)))
87 (defun riece-command-dcc-send (user file)
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
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)
99 jka-compr-compression-info-list
100 (buffer (find-file-noselect file))
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
108 :sentinel 'riece-ndcc-server-sentinel))
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))))))
116 (defun riece-ndcc-filter (process input)
118 (set-buffer (process-buffer process))
119 (goto-char (point-max))
121 (message "Receiving %s from %s...(%d/%d)"
122 (file-name-nondirectory buffer-file-name)
123 riece-ndcc-request-user
125 riece-ndcc-request-size)))
127 (defun riece-ndcc-sentinel (process status)
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))
138 (defun riece-command-dcc-receive (request file)
141 (unless riece-ndcc-requests
142 (error "No request"))
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)
150 (princ (format "%2d: %s %s (%d bytes)\n"
153 (nth 1 (car requests))
154 (nth 4 (car requests))))
155 (setq requests (cdr requests)))))
156 (let ((number (read-string "Request#: ")))
157 (unless (string-match "^[0-9]+$" number)
158 (error "Not a number"))
159 (if (or (> (setq number (string-to-number number))
160 (length riece-ndcc-requests))
162 (error "Invalid number"))
163 (nth (1- number) riece-ndcc-requests)))
164 (expand-file-name (read-file-name "Save as: ")))))
165 (let* (selective-display
166 (coding-system-for-read 'binary)
167 (coding-system-for-write 'binary)
168 (process (open-network-stream
170 (riece-ndcc-decode-address (nth 2 request))
172 (with-current-buffer (process-buffer process)
173 (set-buffer-multibyte nil)
174 (buffer-disable-undo)
175 (setq buffer-file-name file)
176 (make-local-variable 'riece-ndcc-request-user)
177 (setq riece-ndcc-request-user (car request))
178 (make-local-variable 'riece-ndcc-request-size)
179 (setq riece-ndcc-request-size (nth 4 request)))
180 (set-process-filter process #'riece-ndcc-filter)
181 (set-process-sentinel process #'riece-ndcc-sentinel)))
183 (defun riece-handle-dcc-request (prefix target message)
184 (let ((case-fold-search t))
186 "SEND \\([^ ]+\\) \\([^ ]+\\) \\([^ ]+\\) \\([^ ]+\\)"
188 (let ((file (match-string 1 message))
189 (address (match-string 2 message))
190 (port (string-to-number (match-string 3 message)))
191 (size (string-to-number (match-string 4 message)))
192 (buffer (if (riece-channel-p target)
193 (cdr (riece-identity-assoc-no-server
194 (riece-make-identity target)
195 riece-channel-buffer-alist))))
196 (user (riece-prefix-nickname prefix)))
197 (setq riece-ndcc-requests
198 (cons (list user file address port size)
199 riece-ndcc-requests))
200 (riece-insert-change buffer (format "DCC SEND from %s\n" user))
202 (if (and riece-channel-buffer-mode
203 (not (eq buffer riece-channel-buffer)))
204 (list riece-dialogue-buffer riece-others-buffer)
205 riece-dialogue-buffer)
207 (riece-concat-server-name
208 (format "DCC SEND from %s (%s) to %s"
210 (riece-strip-user-at-host
211 (riece-prefix-user-at-host prefix))
216 (defun riece-ndcc-requires ()
219 (defvar riece-dialogue-mode-map)
220 (defun riece-ndcc-insinuate ()
221 (add-hook 'riece-ctcp-dcc-request-hook 'riece-handle-dcc-request)
222 (define-key riece-dialogue-mode-map "\C-ds" 'riece-command-dcc-send)
223 (define-key riece-dialogue-mode-map "\C-dr" 'riece-command-dcc-receive))
225 (provide 'riece-ndcc)
227 ;;; riece-ndcc.el ends here