2 * This module implements a notifier for Funambol push email.
3 * Based on bits of serv_spam, serv_smtp
6 #define FUNAMBOL_WS "/funambol/services/admin"
15 #include <sys/types.h>
17 #if TIME_WITH_SYS_TIME
18 # include <sys/time.h>
22 # include <sys/time.h>
31 #include <sys/socket.h>
34 #include "citserver.h"
44 #include "internet_addressing.h"
46 #include "clientsocket.h"
47 #include "serv_funambol.h"
51 #include "ctdl_module.h"
55 * Create the notify message queue
57 void create_notify_queue(void) {
58 struct ctdlroom qrbuf;
60 create_room(FNBL_QUEUE_ROOM, 3, "", 0, 1, 0, VIEW_MAILBOX);
63 * Make sure it's set to be a "system room" so it doesn't show up
64 * in the <K>nown rooms list for Aides.
66 if (lgetroom(&qrbuf, FNBL_QUEUE_ROOM) == 0) {
67 qrbuf.QRflags2 |= QR2_SYSTEM;
71 void do_notify_queue(void) {
72 static int doing_queue = 0;
75 * This is a simple concurrency check to make sure only one queue run
76 * is done at a time. We could do this with a mutex, but since we
77 * don't really require extremely fine granularity here, we'll do it
78 * with a static variable instead.
80 if (doing_queue) return;
84 * Go ahead and run the queue
86 lprintf(CTDL_INFO, "serv_funambol: processing notify queue\n");
88 if (getroom(&CC->room, FNBL_QUEUE_ROOM) != 0) {
89 lprintf(CTDL_ERR, "Cannot find room <%s>\n", FNBL_QUEUE_ROOM);
92 CtdlForEachMessage(MSGS_ALL, 0L, NULL,
93 SPOOLMIME, NULL, notify_funambol, NULL);
95 lprintf(CTDL_INFO, "serv_funambol: queue run completed\n");
100 * Connect to the Funambol server and scan a message.
102 void notify_funambol(long msgnum, void *userdata) {
103 struct CtdlMessage *msg;
106 char SOAPHeader[SIZ];
109 /* W means 'Wireless'... */
110 msg = CtdlFetchMessage(msgnum, 1);
111 if ( msg->cm_fields['W'] == NULL) {
114 /* Are we allowed to push? */
115 if ( strlen(config.c_funambol_host) == 0) {
118 lprintf(CTDL_INFO, "Push enabled\n");
121 sprintf(port, "%d", config.c_funambol_port);
122 lprintf(CTDL_INFO, "Connecting to Funambol at <%s>\n", config.c_funambol_host);
123 sock = sock_connect(config.c_funambol_host, port, "tcp");
124 if (sock >= 0) lprintf(CTDL_DEBUG, "Connected!\n");
127 /* If the service isn't running, pass for now */
131 /* Build a SOAP message, delicately, by hand */
132 sprintf(SOAPData, "<?xml version=\"1.0\" encoding=\"UTF-8\"?><soapenv:Envelope xmlns:soapenv=\"http://schemas.xmlsoap.org/soap/envelope/\" xmlns:xsd=\"http://www.w3.org/2001/XMLSchema\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\">");
133 strcat(SOAPData, "<soapenv:Body><sendNotificationMessages soapenv:encodingStyle=\"http://schemas.xmlsoap.org/soap/encoding/\">");
134 strcat(SOAPData, "<arg0 xsi:type=\"soapenc:string\" xmlns:soapenc=\"http://schemas.xmlsoap.org/soap/encoding/\">");
135 strcat(SOAPData, msg->cm_fields['W']);
136 strcat(SOAPData, "</arg0>");
137 strcat(SOAPData, "<arg1 xsi:type=\"soapenc:string\" xmlns:soapenc=\"http://schemas.xmlsoap.org/soap/encoding/\"><?xml version="1.0" encoding="UTF-8"?>\r\n");
138 strcat(SOAPData, "<java version="1.5.0_10" class="java.beans.XMLDecoder"> \r\n");
139 strcat(SOAPData, " <array class="com.funambol.framework.core.Alert" length="1">\r\n");
140 strcat(SOAPData, " <void index="0">\r\n");
141 strcat(SOAPData, " <object class="com.funambol.framework.core.Alert">\r\n");
142 strcat(SOAPData, " <void property="cmdID">\r\n");
143 strcat(SOAPData, " <object class="com.funambol.framework.core.CmdID"/>\r\n");
144 strcat(SOAPData, " </void>");
145 strcat(SOAPData, " <void property="data">\r\n");
146 strcat(SOAPData, " <int>210</int>\r\n");
147 strcat(SOAPData, " </void>\r\n");
148 strcat(SOAPData, " <void property="items">\r\n");
149 strcat(SOAPData, " <void method="add">\r\n");
150 strcat(SOAPData, " <object class="com.funambol.framework.core.Item">\r\n");
151 strcat(SOAPData, " <void property="meta">\r\n");
152 strcat(SOAPData, " <object class="com.funambol.framework.core.Meta">\r\n");
153 strcat(SOAPData, " <void property="metInf">\r\n");
154 strcat(SOAPData, " <void property="type">\r\n");
155 strcat(SOAPData, " <string>application/vnd.omads-email+xml</string>\r\n");
156 strcat(SOAPData, " </void>\r\n");
157 strcat(SOAPData, " </void>\r\n");
158 strcat(SOAPData, " </object>\r\n");
159 strcat(SOAPData, " </void>\r\n");
160 strcat(SOAPData, " <void property="target">\r\n");
161 strcat(SOAPData, " <object class="com.funambol.framework.core.Target">\r\n");
162 strcat(SOAPData, " <void property="locURI">\r\n");
163 strcat(SOAPData, " <string>");
164 strcat(SOAPData, config.c_funambol_source);
165 strcat(SOAPData, "</string>\r\n");
166 strcat(SOAPData, " </void>\r\n");
167 strcat(SOAPData, " </object>\r\n");
168 strcat(SOAPData, " </void>\r\n");
169 strcat(SOAPData, " </object>\r\n");
170 strcat(SOAPData, " </void>\r\n");
171 strcat(SOAPData, " </void>\r\n");
172 strcat(SOAPData, " </object>\r\n");
173 strcat(SOAPData, " </void>\r\n");
174 strcat(SOAPData, " </array>\r\n");
175 strcat(SOAPData, "</java>");
176 strcat(SOAPData,"</arg1><arg2 href=\"#id0\"/></sendNotificationMessages><multiRef id=\"id0\" soapenc:root=\"0\"\r\n");
177 strcat(SOAPData,"soapenv:encodingStyle=\"http://schemas.xmlsoap.org/soap/encoding/\" xsi:type=\"soapenc:int\" xmlns:soapenc=\"http://schemas.xmlsoap.org/soap/encoding/\">1</multiRef></soapenv:Body></soapenv:Envelope>");
180 lprintf(CTDL_DEBUG, "Transmitting command\n");
181 sprintf(SOAPHeader, "POST %s HTTP/1.0\r\nContent-type: text/xml; charset=utf-8\r\n",
183 strcat(SOAPHeader,"Accept: application/soap+xml, application/dime, multipart/related, text/*\r\n");
184 sprintf(buf, "User-Agent: %s/%d\r\nHost: %s:%d\r\nCache-control: no-cache\r\n",
187 config.c_funambol_host,
188 config.c_funambol_port
190 strcat(SOAPHeader,buf);
191 strcat(SOAPHeader,"Pragma: no-cache\r\nSOAPAction: \"\"\r\n");
192 sprintf(buf, "Content-Length: %d\r\n",
194 strcat(SOAPHeader, buf);
195 sprintf(buf, "Authorization: Basic %s\r\n\r\n",
196 config.c_funambol_auth);
197 strcat(SOAPHeader, buf);
199 sock_write(sock, SOAPHeader, strlen(SOAPHeader));
200 sock_write(sock, SOAPData, strlen(SOAPData));
201 sock_shutdown(sock, SHUT_WR);
204 lprintf(CTDL_DEBUG, "Awaiting response\n");
205 if (sock_gets(sock, buf) < 0) {
208 lprintf(CTDL_DEBUG, "<%s\n", buf);
209 if (strncasecmp(buf, "HTTP/1.1 200 OK", strlen("HTTP/1.1 200 OK"))) {
213 lprintf(CTDL_DEBUG, "Funambol notified\n");
214 /* We should allow retries here but for now purge after one go */
218 CtdlFreeMessage(msg);
220 todelete[0] = msgnum;
221 CtdlDeleteMessages(FNBL_QUEUE_ROOM, todelete, 1, "");
226 CTDL_MODULE_INIT(funambol)
228 create_notify_queue();
229 CtdlRegisterSessionHook(do_notify_queue, EVT_TIMER);
231 /* return our Subversion id for the Log */