1 /* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
3 * soup-auth-domain-basic.c: HTTP Basic Authentication (server-side)
5 * Copyright (C) 2007 Novell, Inc.
14 #include "soup-auth-domain-basic.h"
16 #include "soup-marshal.h"
19 * SECTION:soup-auth-domain-basic
20 * @short_description: Server-side "Basic" authentication
22 * #SoupAuthDomainBasic handles the server side of HTTP "Basic" (ie,
23 * cleartext password) authentication.
36 SoupAuthDomainBasicAuthCallback auth_callback;
38 GDestroyNotify auth_dnotify;
39 } SoupAuthDomainBasicPrivate;
41 #define SOUP_AUTH_DOMAIN_BASIC_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), SOUP_TYPE_AUTH_DOMAIN_BASIC, SoupAuthDomainBasicPrivate))
43 G_DEFINE_TYPE (SoupAuthDomainBasic, soup_auth_domain_basic, SOUP_TYPE_AUTH_DOMAIN)
46 soup_auth_domain_basic_init (SoupAuthDomainBasic *basic)
51 soup_auth_domain_basic_finalize (GObject *object)
53 SoupAuthDomainBasicPrivate *priv =
54 SOUP_AUTH_DOMAIN_BASIC_GET_PRIVATE (object);
56 if (priv->auth_dnotify)
57 priv->auth_dnotify (priv->auth_data);
59 G_OBJECT_CLASS (soup_auth_domain_basic_parent_class)->finalize (object);
63 soup_auth_domain_basic_set_property (GObject *object, guint prop_id,
64 const GValue *value, GParamSpec *pspec)
66 SoupAuthDomainBasicPrivate *priv =
67 SOUP_AUTH_DOMAIN_BASIC_GET_PRIVATE (object);
70 case PROP_AUTH_CALLBACK:
71 priv->auth_callback = g_value_get_pointer (value);
74 if (priv->auth_dnotify) {
75 priv->auth_dnotify (priv->auth_data);
76 priv->auth_dnotify = NULL;
78 priv->auth_data = g_value_get_pointer (value);
81 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
87 soup_auth_domain_basic_get_property (GObject *object, guint prop_id,
88 GValue *value, GParamSpec *pspec)
90 SoupAuthDomainBasicPrivate *priv =
91 SOUP_AUTH_DOMAIN_BASIC_GET_PRIVATE (object);
94 case PROP_AUTH_CALLBACK:
95 g_value_set_pointer (value, priv->auth_callback);
98 g_value_set_pointer (value, priv->auth_data);
101 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
107 * soup_auth_domain_basic_new:
108 * @optname1: name of first option, or %NULL
109 * @...: option name/value pairs
111 * Creates a #SoupAuthDomainBasic. You must set the
112 * %SOUP_AUTH_DOMAIN_REALM parameter, to indicate the realm name to be
113 * returned with the authentication challenge to the client. Other
114 * parameters are optional.
116 * Return value: the new #SoupAuthDomain
119 soup_auth_domain_basic_new (const char *optname1, ...)
121 SoupAuthDomain *domain;
124 va_start (ap, optname1);
125 domain = (SoupAuthDomain *)g_object_new_valist (SOUP_TYPE_AUTH_DOMAIN_BASIC,
129 g_return_val_if_fail (soup_auth_domain_get_realm (domain) != NULL, NULL);
135 * SoupAuthDomainBasicAuthCallback:
136 * @domain: the domain
137 * @msg: the message being authenticated
138 * @username: the username provided by the client
139 * @password: the password provided by the client
140 * @user_data: the data passed to soup_auth_domain_basic_set_auth_callback()
142 * Callback used by #SoupAuthDomainBasic for authentication purposes.
143 * The application should verify that @username and @password and valid
144 * and return %TRUE or %FALSE.
146 * If you are maintaining your own password database (rather than
147 * using the password to authenticate against some other system like
148 * PAM or a remote server), you should make sure you know what you are
149 * doing. In particular, don't store cleartext passwords, or
150 * easily-computed hashes of cleartext passwords, even if you don't
151 * care that much about the security of your server, because users
152 * will frequently use the same password for multiple sites, and so
153 * compromising any site with a cleartext (or easily-cracked) password
154 * database may give attackers access to other more-interesting sites
157 * Return value: %TRUE if @username and @password are valid
161 * soup_auth_domain_basic_set_auth_callback:
162 * @domain: the domain
163 * @callback: the callback
164 * @user_data: data to pass to @auth_callback
165 * @dnotify: destroy notifier to free @user_data when @domain
168 * Sets the callback that @domain will use to authenticate incoming
169 * requests. For each request containing authorization, @domain will
170 * invoke the callback, and then either accept or reject the request
171 * based on @callback's return value.
173 * You can also set the auth callback by setting the
174 * %SOUP_AUTH_DOMAIN_BASIC_AUTH_CALLBACK and
175 * %SOUP_AUTH_DOMAIN_BASIC_AUTH_DATA properties, which can also be
176 * used to set the callback at construct time.
179 soup_auth_domain_basic_set_auth_callback (SoupAuthDomain *domain,
180 SoupAuthDomainBasicAuthCallback callback,
182 GDestroyNotify dnotify)
184 SoupAuthDomainBasicPrivate *priv =
185 SOUP_AUTH_DOMAIN_BASIC_GET_PRIVATE (domain);
187 if (priv->auth_dnotify)
188 priv->auth_dnotify (priv->auth_data);
190 priv->auth_callback = callback;
191 priv->auth_data = user_data;
192 priv->auth_dnotify = dnotify;
194 g_object_notify (G_OBJECT (domain), SOUP_AUTH_DOMAIN_BASIC_AUTH_CALLBACK);
195 g_object_notify (G_OBJECT (domain), SOUP_AUTH_DOMAIN_BASIC_AUTH_DATA);
201 memset (pw, 0, strlen (pw));
206 parse_basic (SoupMessage *msg, const char *header,
207 char **username, char **password)
209 char *decoded, *colon;
212 if (!header || (strncmp (header, "Basic ", 6) != 0))
215 decoded = (char *)g_base64_decode (header + 6, &len);
219 colon = memchr (decoded, ':', len);
225 plen = len - (colon - decoded) - 1;
227 *password = g_strndup (colon + 1, plen);
228 memset (colon + 1, 0, plen);
234 soup_auth_domain_basic_accepts (SoupAuthDomain *domain, SoupMessage *msg,
237 SoupAuthDomainBasicPrivate *priv =
238 SOUP_AUTH_DOMAIN_BASIC_GET_PRIVATE (domain);
239 char *username, *password;
242 if (!parse_basic (msg, header, &username, &password))
245 if (priv->auth_callback) {
246 ok = priv->auth_callback (domain, msg, username, password,
249 ok = soup_auth_domain_try_generic_auth_callback (
250 domain, msg, username);
264 soup_auth_domain_basic_challenge (SoupAuthDomain *domain, SoupMessage *msg)
268 challenge = g_string_new ("Basic ");
269 soup_header_g_string_append_param (challenge, "realm", soup_auth_domain_get_realm (domain));
270 return g_string_free (challenge, FALSE);
274 soup_auth_domain_basic_check_password (SoupAuthDomain *domain,
276 const char *username,
277 const char *password)
280 char *msg_username, *msg_password;
283 header = soup_message_headers_get_one (msg->request_headers,
285 if (!parse_basic (msg, header, &msg_username, &msg_password))
288 ok = (!strcmp (username, msg_username) &&
289 !strcmp (password, msg_password));
290 g_free (msg_username);
291 pw_free (msg_password);
297 soup_auth_domain_basic_class_init (SoupAuthDomainBasicClass *basic_class)
299 SoupAuthDomainClass *auth_domain_class =
300 SOUP_AUTH_DOMAIN_CLASS (basic_class);
301 GObjectClass *object_class = G_OBJECT_CLASS (basic_class);
303 g_type_class_add_private (basic_class, sizeof (SoupAuthDomainBasicPrivate));
305 auth_domain_class->accepts = soup_auth_domain_basic_accepts;
306 auth_domain_class->challenge = soup_auth_domain_basic_challenge;
307 auth_domain_class->check_password = soup_auth_domain_basic_check_password;
309 object_class->finalize = soup_auth_domain_basic_finalize;
310 object_class->set_property = soup_auth_domain_basic_set_property;
311 object_class->get_property = soup_auth_domain_basic_get_property;
314 * SOUP_AUTH_DOMAIN_BASIC_AUTH_CALLBACK:
316 * Alias for the #SoupAuthDomainBasic:auth-callback property.
317 * (The #SoupAuthDomainBasicAuthCallback.)
319 g_object_class_install_property (
320 object_class, PROP_AUTH_CALLBACK,
321 g_param_spec_pointer (SOUP_AUTH_DOMAIN_BASIC_AUTH_CALLBACK,
322 "Authentication callback",
323 "Password-checking callback",
326 * SOUP_AUTH_DOMAIN_BASIC_AUTH_DATA:
328 * Alias for the #SoupAuthDomainBasic:auth-data property.
329 * (The data to pass to the #SoupAuthDomainBasicAuthCallback.)
331 g_object_class_install_property (
332 object_class, PROP_AUTH_DATA,
333 g_param_spec_pointer (SOUP_AUTH_DOMAIN_BASIC_AUTH_DATA,
334 "Authentication callback data",
335 "Data to pass to authentication callback",