Merge from emacs-24; up to 2014-05-29T17:16:00Z!dmantipov@yandex.ru
[bpt/emacs.git] / lisp / gnus / gravatar.el
CommitLineData
61b1af82
G
1;;; gravatar.el --- Get Gravatars
2
ba318903 3;; Copyright (C) 2010-2014 Free Software Foundation, Inc.
61b1af82
G
4
5;; Author: Julien Danjou <julien@danjou.info>
6;; Keywords: news
7
8;; This file is part of GNU Emacs.
9
10;; GNU Emacs is free software: you can redistribute it and/or modify
11;; it under the terms of the GNU General Public License as published by
12;; the Free Software Foundation, either version 3 of the License, or
13;; (at your option) any later version.
14
15;; GNU Emacs is distributed in the hope that it will be useful,
16;; but WITHOUT ANY WARRANTY; without even the implied warranty of
17;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18;; GNU General Public License for more details.
19
20;; You should have received a copy of the GNU General Public License
21;; along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
22
23;;; Commentary:
24
25;;; Code:
26
61b1af82
G
27(require 'url)
28(require 'url-cache)
29
30(defgroup gravatar nil
31 "Gravatar."
2bed3f04 32 :version "24.1"
61b1af82
G
33 :group 'comm)
34
35(defcustom gravatar-automatic-caching t
9c5a5c77
GM
36 "Whether to cache retrieved gravatars."
37 :type 'boolean
61b1af82
G
38 :group 'gravatar)
39
9c5a5c77 40;; FIXME a time value is not the nicest format for a custom variable.
61b1af82
G
41(defcustom gravatar-cache-ttl (days-to-time 30)
42 "Time to live for gravatar cache entries."
9c5a5c77 43 :type '(repeat integer)
61b1af82
G
44 :group 'gravatar)
45
9c5a5c77 46;; FIXME Doc is tautological. What are the options?
61b1af82
G
47(defcustom gravatar-rating "g"
48 "Default rating for gravatar."
9c5a5c77 49 :type 'string
61b1af82
G
50 :group 'gravatar)
51
52(defcustom gravatar-size 32
53 "Default size in pixels for gravatars."
9c5a5c77 54 :type 'integer
61b1af82
G
55 :group 'gravatar)
56
57(defconst gravatar-base-url
58 "http://www.gravatar.com/avatar"
59 "Base URL for getting gravatars.")
60
61(defun gravatar-hash (mail-address)
62 "Create an hash from MAIL-ADDRESS."
63 (md5 (downcase mail-address)))
64
65(defun gravatar-build-url (mail-address)
66 "Return an URL to retrieve MAIL-ADDRESS gravatar."
67 (format "%s/%s?d=404&r=%s&s=%d"
68 gravatar-base-url
69 (gravatar-hash mail-address)
70 gravatar-rating
71 gravatar-size))
72
73(defun gravatar-cache-expired (url)
74 "Check if URL is cached for more than `gravatar-cache-ttl'."
75 (cond (url-standalone-mode
76 (not (file-exists-p (url-cache-create-filename url))))
77 (t (let ((cache-time (url-is-cached url)))
78 (if cache-time
79 (time-less-p
80 (time-add
81 cache-time
82 gravatar-cache-ttl)
83 (current-time))
84 t)))))
85
86(defun gravatar-get-data ()
87 "Get data from current buffer."
70041e9a
G
88 (save-excursion
89 (goto-char (point-min))
90 (when (re-search-forward "^HTTP/.+ 200 OK$" nil (line-end-position))
91 (when (search-forward "\n\n" nil t)
92 (buffer-substring (point) (point-max))))))
61b1af82 93
4b36c6d4
KY
94(eval-and-compile
95 (cond ((featurep 'xemacs)
96 (require 'gnus-xmas)
97 (defalias 'gravatar-create-image 'gnus-xmas-create-image))
98 ((featurep 'gnus-ems)
99 (defalias 'gravatar-create-image 'gnus-create-image))
100 (t
101 (require 'image)
102 (defalias 'gravatar-create-image 'create-image))))
103
61b1af82
G
104(defun gravatar-data->image ()
105 "Get data of current buffer and return an image.
106If no image available, return 'error."
107 (let ((data (gravatar-get-data)))
108 (if data
70041e9a 109 (gravatar-create-image data nil t)
61b1af82
G
110 'error)))
111
cf6a9685
GM
112(autoload 'help-function-arglist "help-fns")
113
61b1af82
G
114;;;###autoload
115(defun gravatar-retrieve (mail-address cb &optional cbargs)
116 "Retrieve MAIL-ADDRESS gravatar and call CB on retrieval.
117You can provide a list of argument to pass to CB in CBARGS."
118 (let ((url (gravatar-build-url mail-address)))
119 (if (gravatar-cache-expired url)
cb51ba08
LI
120 (let ((args (list url
121 'gravatar-retrieved
122 (list cb (when cbargs cbargs)))))
123 (when (> (length (if (featurep 'xemacs)
124 (cdr (split-string (function-arglist 'url-retrieve)))
125 (help-function-arglist 'url-retrieve)))
126 4)
127 (setq args (nconc args (list t))))
128 (apply #'url-retrieve args))
61b1af82
G
129 (apply cb
130 (with-temp-buffer
131 (mm-disable-multibyte)
132 (url-cache-extract (url-cache-create-filename url))
133 (gravatar-data->image))
134 cbargs))))
135
70041e9a
G
136;;;###autoload
137(defun gravatar-retrieve-synchronously (mail-address)
138 "Retrieve MAIL-ADDRESS gravatar and returns it."
139 (let ((url (gravatar-build-url mail-address)))
140 (if (gravatar-cache-expired url)
4d2226bf 141 (with-current-buffer (url-retrieve-synchronously url)
1518e4f0 142 (when gravatar-automatic-caching
70041e9a
G
143 (url-store-in-cache (current-buffer)))
144 (let ((data (gravatar-data->image)))
145 (kill-buffer (current-buffer))
146 data))
147 (with-temp-buffer
148 (mm-disable-multibyte)
149 (url-cache-extract (url-cache-create-filename url))
150 (gravatar-data->image)))))
151
152
61b1af82
G
153(defun gravatar-retrieved (status cb &optional cbargs)
154 "Callback function used by `gravatar-retrieve'."
155 ;; Store gravatar?
156 (when gravatar-automatic-caching
157 (url-store-in-cache (current-buffer)))
158 (if (plist-get status :error)
159 ;; Error happened.
160 (apply cb 'error cbargs)
71e691a5
G
161 (apply cb (gravatar-data->image) cbargs))
162 (kill-buffer (current-buffer)))
61b1af82
G
163
164(provide 'gravatar)
165
166;;; gravatar.el ends here