mfplat: Read queue subscriber within the critical section.
[wine/zf.git] / dlls / winhttp / cookie.c
blob6247e379ec7e454d2d2d736b747ba8ffb53c59e1
1 /*
2 * Copyright 2008 Hans Leidekker for CodeWeavers
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19 #include <stdarg.h>
20 #include <wchar.h>
22 #include "windef.h"
23 #include "winbase.h"
24 #include "ws2tcpip.h"
25 #include "winhttp.h"
27 #include "wine/debug.h"
28 #include "wine/list.h"
29 #include "winhttp_private.h"
31 WINE_DEFAULT_DEBUG_CHANNEL(winhttp);
33 struct cookie
35 struct list entry;
36 WCHAR *name;
37 WCHAR *value;
38 WCHAR *path;
41 struct domain
43 struct list entry;
44 WCHAR *name;
45 struct list cookies;
48 static struct domain *add_domain( struct session *session, WCHAR *name )
50 struct domain *domain;
52 if (!(domain = heap_alloc_zero( sizeof(struct domain) ))) return NULL;
54 list_init( &domain->entry );
55 list_init( &domain->cookies );
57 domain->name = strdupW( name );
58 list_add_tail( &session->cookie_cache, &domain->entry );
60 TRACE("%s\n", debugstr_w(domain->name));
61 return domain;
64 static struct cookie *find_cookie( struct domain *domain, const WCHAR *path, const WCHAR *name )
66 struct list *item;
67 struct cookie *cookie;
69 LIST_FOR_EACH( item, &domain->cookies )
71 cookie = LIST_ENTRY( item, struct cookie, entry );
72 if (!wcscmp( cookie->path, path ) && !wcscmp( cookie->name, name ))
74 TRACE("found %s=%s\n", debugstr_w(cookie->name), debugstr_w(cookie->value));
75 return cookie;
78 return NULL;
81 static BOOL domain_match( const WCHAR *name, struct domain *domain, BOOL partial )
83 TRACE("comparing %s with %s\n", debugstr_w(name), debugstr_w(domain->name));
85 if (partial && !wcsstr( name, domain->name )) return FALSE;
86 else if (!partial && wcscmp( name, domain->name )) return FALSE;
87 return TRUE;
90 static void free_cookie( struct cookie *cookie )
92 heap_free( cookie->name );
93 heap_free( cookie->value );
94 heap_free( cookie->path );
95 heap_free( cookie );
98 static void delete_cookie( struct cookie *cookie )
100 list_remove( &cookie->entry );
101 free_cookie( cookie );
104 static void delete_domain( struct domain *domain )
106 struct cookie *cookie;
107 struct list *item, *next;
109 LIST_FOR_EACH_SAFE( item, next, &domain->cookies )
111 cookie = LIST_ENTRY( item, struct cookie, entry );
112 delete_cookie( cookie );
115 list_remove( &domain->entry );
116 heap_free( domain->name );
117 heap_free( domain );
120 void destroy_cookies( struct session *session )
122 struct list *item, *next;
123 struct domain *domain;
125 LIST_FOR_EACH_SAFE( item, next, &session->cookie_cache )
127 domain = LIST_ENTRY( item, struct domain, entry );
128 delete_domain( domain );
132 static BOOL add_cookie( struct session *session, struct cookie *cookie, WCHAR *domain_name, WCHAR *path )
134 struct domain *domain = NULL;
135 struct cookie *old_cookie;
136 struct list *item;
138 if (!(cookie->path = strdupW( path ))) return FALSE;
140 EnterCriticalSection( &session->cs );
142 LIST_FOR_EACH( item, &session->cookie_cache )
144 domain = LIST_ENTRY( item, struct domain, entry );
145 if (domain_match( domain_name, domain, FALSE )) break;
146 domain = NULL;
148 if (!domain) domain = add_domain( session, domain_name );
149 else if ((old_cookie = find_cookie( domain, path, cookie->name ))) delete_cookie( old_cookie );
151 if (domain)
153 list_add_head( &domain->cookies, &cookie->entry );
154 TRACE("domain %s path %s <- %s=%s\n", debugstr_w(domain_name), debugstr_w(cookie->path),
155 debugstr_w(cookie->name), debugstr_w(cookie->value));
158 LeaveCriticalSection( &session->cs );
159 return domain != NULL;
162 static struct cookie *parse_cookie( const WCHAR *string )
164 struct cookie *cookie;
165 const WCHAR *p;
166 int len;
168 if (!(p = wcschr( string, '=' ))) p = string + lstrlenW( string );
169 len = p - string;
170 while (len && string[len - 1] == ' ') len--;
171 if (!len) return NULL;
173 if (!(cookie = heap_alloc_zero( sizeof(struct cookie) ))) return NULL;
174 list_init( &cookie->entry );
176 if (!(cookie->name = heap_alloc( (len + 1) * sizeof(WCHAR) )))
178 heap_free( cookie );
179 return NULL;
181 memcpy( cookie->name, string, len * sizeof(WCHAR) );
182 cookie->name[len] = 0;
184 if (*p++ == '=')
186 while (*p == ' ') p++;
187 len = lstrlenW( p );
188 while (len && p[len - 1] == ' ') len--;
190 if (!(cookie->value = heap_alloc( (len + 1) * sizeof(WCHAR) )))
192 free_cookie( cookie );
193 return NULL;
195 memcpy( cookie->value, p, len * sizeof(WCHAR) );
196 cookie->value[len] = 0;
198 return cookie;
201 struct attr
203 WCHAR *name;
204 WCHAR *value;
207 static void free_attr( struct attr *attr )
209 if (!attr) return;
210 heap_free( attr->name );
211 heap_free( attr->value );
212 heap_free( attr );
215 static struct attr *parse_attr( const WCHAR *str, int *used )
217 const WCHAR *p = str, *q;
218 struct attr *attr;
219 int len;
221 while (*p == ' ') p++;
222 q = p;
223 while (*q && *q != ' ' && *q != '=' && *q != ';') q++;
224 len = q - p;
225 if (!len) return NULL;
227 if (!(attr = heap_alloc( sizeof(struct attr) ))) return NULL;
228 if (!(attr->name = heap_alloc( (len + 1) * sizeof(WCHAR) )))
230 heap_free( attr );
231 return NULL;
233 memcpy( attr->name, p, len * sizeof(WCHAR) );
234 attr->name[len] = 0;
235 attr->value = NULL;
237 p = q;
238 while (*p == ' ') p++;
239 if (*p++ == '=')
241 while (*p == ' ') p++;
242 q = p;
243 while (*q && *q != ';') q++;
244 len = q - p;
245 while (len && p[len - 1] == ' ') len--;
247 if (!(attr->value = heap_alloc( (len + 1) * sizeof(WCHAR) )))
249 free_attr( attr );
250 return NULL;
252 memcpy( attr->value, p, len * sizeof(WCHAR) );
253 attr->value[len] = 0;
256 while (*q == ' ') q++;
257 if (*q == ';') q++;
258 *used = q - str;
260 return attr;
263 BOOL set_cookies( struct request *request, const WCHAR *cookies )
265 BOOL ret = FALSE;
266 WCHAR *buffer, *p;
267 WCHAR *cookie_domain = NULL, *cookie_path = NULL;
268 struct attr *attr, *domain = NULL, *path = NULL;
269 struct session *session = request->connect->session;
270 struct cookie *cookie;
271 int len, used;
273 len = lstrlenW( cookies );
274 if (!(buffer = heap_alloc( (len + 1) * sizeof(WCHAR) ))) return FALSE;
275 lstrcpyW( buffer, cookies );
277 p = buffer;
278 while (*p && *p != ';') p++;
279 if (*p == ';') *p++ = 0;
280 if (!(cookie = parse_cookie( buffer )))
282 heap_free( buffer );
283 return FALSE;
285 len = lstrlenW( p );
286 while (len && (attr = parse_attr( p, &used )))
288 if (!wcsicmp( attr->name, L"domain" ))
290 domain = attr;
291 cookie_domain = attr->value;
293 else if (!wcsicmp( attr->name, L"path" ))
295 path = attr;
296 cookie_path = attr->value;
298 else
300 FIXME( "unhandled attribute %s\n", debugstr_w(attr->name) );
301 free_attr( attr );
303 len -= used;
304 p += used;
306 if (!cookie_domain && !(cookie_domain = strdupW( request->connect->servername ))) goto end;
307 if (!cookie_path && !(cookie_path = strdupW( request->path ))) goto end;
309 if ((p = wcsrchr( cookie_path, '/' )) && p != cookie_path) *p = 0;
310 ret = add_cookie( session, cookie, cookie_domain, cookie_path );
312 end:
313 if (!ret) free_cookie( cookie );
314 if (domain) free_attr( domain );
315 else heap_free( cookie_domain );
316 if (path) free_attr( path );
317 else heap_free( cookie_path );
318 heap_free( buffer );
319 return ret;
322 DWORD add_cookie_headers( struct request *request )
324 struct list *domain_cursor;
325 struct session *session = request->connect->session;
326 DWORD ret = ERROR_SUCCESS;
328 EnterCriticalSection( &session->cs );
330 LIST_FOR_EACH( domain_cursor, &session->cookie_cache )
332 struct domain *domain = LIST_ENTRY( domain_cursor, struct domain, entry );
333 if (domain_match( request->connect->servername, domain, TRUE ))
335 struct list *cookie_cursor;
336 TRACE("found domain %s\n", debugstr_w(domain->name));
338 LIST_FOR_EACH( cookie_cursor, &domain->cookies )
340 struct cookie *cookie = LIST_ENTRY( cookie_cursor, struct cookie, entry );
342 TRACE("comparing path %s with %s\n", debugstr_w(request->path), debugstr_w(cookie->path));
344 if (wcsstr( request->path, cookie->path ) == request->path)
346 static const WCHAR cookieW[] = {'C','o','o','k','i','e',':',' '};
347 int len, len_cookie = ARRAY_SIZE( cookieW ), len_name = lstrlenW( cookie->name );
348 WCHAR *header;
350 len = len_cookie + len_name;
351 if (cookie->value) len += lstrlenW( cookie->value ) + 1;
352 if (!(header = heap_alloc( (len + 1) * sizeof(WCHAR) )))
354 LeaveCriticalSection( &session->cs );
355 return ERROR_OUTOFMEMORY;
358 memcpy( header, cookieW, len_cookie * sizeof(WCHAR) );
359 lstrcpyW( header + len_cookie, cookie->name );
360 if (cookie->value)
362 header[len_cookie + len_name] = '=';
363 lstrcpyW( header + len_cookie + len_name + 1, cookie->value );
366 TRACE("%s\n", debugstr_w(header));
367 ret = add_request_headers( request, header, len,
368 WINHTTP_ADDREQ_FLAG_ADD | WINHTTP_ADDREQ_FLAG_COALESCE_WITH_SEMICOLON );
369 heap_free( header );
375 LeaveCriticalSection( &session->cs );
376 return ret;