blob: 754e261601f865075260b5cc781eacfa6e649923 [file] [log] [blame]
Willy Tarreaubaaee002006-06-26 02:48:02 +02001/*
2 * AppSession functions.
3 *
4 * Copyright 2004-2006 Alexander Lazic, Klaus Wagner
Willy Tarreau26c25062009-03-08 09:38:41 +01005 * Copyright 2006-2009 Willy Tarreau
Willy Tarreaubaaee002006-06-26 02:48:02 +02006 *
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License
9 * as published by the Free Software Foundation; either version
10 * 2 of the License, or (at your option) any later version.
11 *
12 */
13
14#include <stdio.h>
Willy Tarreau2dd0d472006-06-29 17:53:05 +020015#include <string.h>
Willy Tarreaubaaee002006-06-26 02:48:02 +020016
Willy Tarreau2dd0d472006-06-29 17:53:05 +020017#include <common/appsession.h>
Willy Tarreaue3ba5f02006-06-29 18:54:54 +020018#include <common/config.h>
Willy Tarreau63963c62007-05-13 21:29:55 +020019#include <common/memory.h>
Willy Tarreau51041c72007-09-09 21:56:53 +020020#include <common/sessionhash.h>
Willy Tarreau0c303ee2008-07-07 00:09:58 +020021#include <common/ticks.h>
Willy Tarreau2dd0d472006-06-29 17:53:05 +020022#include <common/time.h>
Willy Tarreaubaaee002006-06-26 02:48:02 +020023
Willy Tarreaubaaee002006-06-26 02:48:02 +020024#include <types/global.h>
Willy Tarreaubaaee002006-06-26 02:48:02 +020025
Willy Tarreauec6c5df2008-07-15 00:22:45 +020026#include <proto/proxy.h>
Willy Tarreaubaaee002006-06-26 02:48:02 +020027#include <proto/task.h>
28
Krzysztof Piotr Oledzkia643baf2008-05-29 23:53:44 +020029static struct task *appsess_refresh = NULL;
Willy Tarreau63963c62007-05-13 21:29:55 +020030struct pool_head *pool2_appsess;
Willy Tarreaubaaee002006-06-26 02:48:02 +020031struct app_pool apools;
32int have_appsession;
33
Willy Tarreaubaaee002006-06-26 02:48:02 +020034int appsession_init(void)
35{
36 static int initialized = 0;
37 int idlen;
38 struct server *s;
39 struct proxy *p = proxy;
40
41 if (!initialized) {
Willy Tarreau63963c62007-05-13 21:29:55 +020042 pool2_appsess = create_pool("appsess", sizeof(appsess), MEM_F_SHARED);
43 if (pool2_appsess == NULL)
44 return -1;
45
Willy Tarreaubaaee002006-06-26 02:48:02 +020046 if (!appsession_task_init()) {
Willy Tarreau63963c62007-05-13 21:29:55 +020047 int ser_msize, ses_msize;
48
Willy Tarreaubaaee002006-06-26 02:48:02 +020049 apools.sessid = NULL;
50 apools.serverid = NULL;
Willy Tarreau63963c62007-05-13 21:29:55 +020051
52 ser_msize = sizeof(void *);
53 ses_msize = sizeof(void *);
Willy Tarreaubaaee002006-06-26 02:48:02 +020054 while (p) {
55 s = p->srv;
Willy Tarreau63963c62007-05-13 21:29:55 +020056 if (ses_msize < p->appsession_len)
57 ses_msize = p->appsession_len;
Willy Tarreaubaaee002006-06-26 02:48:02 +020058 while (s) {
59 idlen = strlen(s->id);
Willy Tarreau63963c62007-05-13 21:29:55 +020060 if (ser_msize < idlen)
61 ser_msize = idlen;
Willy Tarreaubaaee002006-06-26 02:48:02 +020062 s = s->next;
63 }
64 p = p->next;
65 }
66 /* we use strings, so reserve space for '\0' */
Willy Tarreau63963c62007-05-13 21:29:55 +020067 ser_msize ++;
68 ses_msize ++;
69
70 apools.sessid = create_pool("sessid", ses_msize, MEM_F_SHARED);
71 if (!apools.sessid)
72 return -1;
73 apools.serverid = create_pool("serverid", ser_msize, MEM_F_SHARED);
74 if (!apools.serverid)
75 return -1;
Willy Tarreaubaaee002006-06-26 02:48:02 +020076 }
77 else {
78 fprintf(stderr, "appsession_task_init failed\n");
79 return -1;
80 }
81 initialized ++;
82 }
83 return 0;
84}
85
86int appsession_task_init(void)
87{
88 static int initialized = 0;
Willy Tarreaubaaee002006-06-26 02:48:02 +020089 if (!initialized) {
Willy Tarreaua4613182009-03-21 18:13:21 +010090 if ((appsess_refresh = task_new()) == NULL)
Willy Tarreaubaaee002006-06-26 02:48:02 +020091 return -1;
Krzysztof Piotr Oledzkia643baf2008-05-29 23:53:44 +020092
Krzysztof Piotr Oledzkia643baf2008-05-29 23:53:44 +020093 appsess_refresh->context = NULL;
Willy Tarreau0c303ee2008-07-07 00:09:58 +020094 appsess_refresh->expire = tick_add(now_ms, MS_TO_TICKS(TBLCHKINT));
Krzysztof Piotr Oledzkia643baf2008-05-29 23:53:44 +020095 appsess_refresh->process = appsession_refresh;
96 task_queue(appsess_refresh);
Willy Tarreaubaaee002006-06-26 02:48:02 +020097 initialized ++;
98 }
99 return 0;
100}
101
Willy Tarreau26c25062009-03-08 09:38:41 +0100102struct task *appsession_refresh(struct task *t)
Willy Tarreaubaaee002006-06-26 02:48:02 +0200103{
Willy Tarreau51041c72007-09-09 21:56:53 +0200104 struct proxy *p = proxy;
105 struct appsession_hash *htbl;
106 appsess *element, *back;
107 int i;
Willy Tarreaubaaee002006-06-26 02:48:02 +0200108
109 while (p) {
110 if (p->appsession_name != NULL) {
111 htbl = &p->htbl_proxy;
Willy Tarreau51041c72007-09-09 21:56:53 +0200112 as_hash_for_each_entry_safe(i, element, back, &p->htbl_proxy, hash_list) {
Willy Tarreau0c303ee2008-07-07 00:09:58 +0200113 if (tick_is_expired(element->expire, now_ms)) {
Willy Tarreau51041c72007-09-09 21:56:53 +0200114 if ((global.mode & MODE_DEBUG) &&
115 (!(global.mode & MODE_QUIET) || (global.mode & MODE_VERBOSE))) {
116 int len;
117 /*
118 on Linux NULL pointers are caught by sprintf, on solaris -> segfault
119 */
120 len = sprintf(trash, "appsession_refresh: cleaning up expired Session '%s' on Server %s\n",
121 element->sessid, element->serverid?element->serverid:"(null)");
122 write(1, trash, len);
123 }
124 /* delete the expired element from within the hash table */
125 LIST_DEL(&element->hash_list);
126 htbl->destroy(element);
127 }/* end if (tv_isle(&asession->expire, &now)) */
Willy Tarreaubaaee002006-06-26 02:48:02 +0200128 }
129 }
130 p = p->next;
131 }
Willy Tarreau0c303ee2008-07-07 00:09:58 +0200132 t->expire = tick_add(now_ms, MS_TO_TICKS(TBLCHKINT)); /* check expiration every 5 seconds */
Willy Tarreau26c25062009-03-08 09:38:41 +0100133 return t;
Willy Tarreaubaaee002006-06-26 02:48:02 +0200134} /* end appsession_refresh */
135
136int match_str(const void *key1, const void *key2)
137{
138 appsess *temp1,*temp2;
139 temp1 = (appsess *)key1;
140 temp2 = (appsess *)key2;
141
142 //fprintf(stdout,">>>>>>>>>>>>>>temp1->sessid :%s:\n",temp1->sessid);
143 //fprintf(stdout,">>>>>>>>>>>>>>temp2->sessid :%s:\n",temp2->sessid);
144
145 return (strcmp(temp1->sessid,temp2->sessid) == 0);
146}/* end match_str */
147
Willy Tarreau51041c72007-09-09 21:56:53 +0200148void destroy(appsess *temp1) {
Willy Tarreau63963c62007-05-13 21:29:55 +0200149 pool_free2(apools.sessid, temp1->sessid);
Willy Tarreau63963c62007-05-13 21:29:55 +0200150 pool_free2(apools.serverid, temp1->serverid);
Willy Tarreau48d63db2008-08-03 17:41:33 +0200151 pool_free2(pool2_appsess, temp1);
Willy Tarreaubaaee002006-06-26 02:48:02 +0200152} /* end destroy */
153
154void appsession_cleanup( void )
155{
156 struct proxy *p = proxy;
157
158 while(p) {
Willy Tarreau51041c72007-09-09 21:56:53 +0200159 appsession_hash_destroy(&(p->htbl_proxy));
Willy Tarreaubaaee002006-06-26 02:48:02 +0200160 p = p->next;
161 }
Krzysztof Piotr Oledzkia643baf2008-05-29 23:53:44 +0200162
163 if (appsess_refresh) {
164 task_delete(appsess_refresh);
165 task_free(appsess_refresh);
166 appsess_refresh = NULL;
167 }
168
Willy Tarreaubaaee002006-06-26 02:48:02 +0200169}/* end appsession_cleanup() */
170
171
172
173/*
174 * Local variables:
175 * c-indent-level: 8
176 * c-basic-offset: 8
177 * End:
178 */