0d088d0ce9462f7ecb620955fd9e83851655954b
[platform/core/connectivity/bluetooth-agent.git] / map-agent / bluetooth_map_email.c
1 /*
2  * Bluetooth-agent
3  *
4  * Copyright (c) 2000 - 2011 Samsung Electronics Co., Ltd. All rights reserved.
5  *
6  * Contact:  Hocheol Seo <hocheol.seo@samsung.com>
7  *               Syam Sidhardhan <s.syam@samsung.com>
8  *               Chanyeol Park <chanyeol.park@samsung.com>
9  *
10  * Licensed under the Apache License, Version 2.0 (the "License");
11  * you may not use this file except in compliance with the License.
12  * You may obtain a copy of the License at
13  *
14  *              http://www.apache.org/licenses/LICENSE-2.0
15  *
16  * Unless required by applicable law or agreed to in writing, software
17  * distributed under the License is distributed on an "AS IS" BASIS,
18  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
19  * See the License for the specific language governing permissions and
20  * limitations under the License.
21  *
22  */
23
24 #include "email-types.h"
25 #include "email-api-init.h"
26 #include "email-api-account.h"
27 #include "email-api-mailbox.h"
28 #include "email-api-mail.h"
29 #include "email-api-network.h"
30 #include <email-api.h>
31
32 #include <bluetooth_map_agent.h>
33 #include <map_bmessage.h>
34
35 #include <glib.h>
36 #include <gio/gio.h>
37 #include <stdlib.h>
38 #ifdef ARCH64
39 #include <stdint.h>
40 #endif
41
42 #define BT_MAIL_ID_MAX_LENGTH 50
43 #define BT_MAP_TIMESTAMP_MAX_LEN 16
44 #define BT_MAIL_TEMP_BODY "/tmp/bt_mail.txt"
45 #define BT_MAP_MSG_HANDLE_MAX 21
46 #define BT_EMAIL_STORAGE_INTERFACE "User.Email.StorageChange"
47 #define BT_EMAIL_STORAGE_PATH "/User/Email/StorageChange"
48 #define BT_EMAIL_STORAGE_SIGNAL "email"
49
50 #define BEGIN_BMSEG "BEGIN:BMSG\r\n"
51 #define END_BMSEG "END:BMSG\r\n"
52 #define BMSEG_VERSION "VERSION:1.0\r\n"
53 #define MSEG_STATUS "STATUS:%s\r\n"
54 #define MSEG_TYPE "TYPE:%s\r\n"
55 #define FOLDER_PATH "FOLDER:%s\r\n"
56 #define EMAIL_VCARD "BEGIN:VCARD\r\nVERSION:2.1\r\nN:%s\r\nEMAIL:%s\r\nEND:VCARD\r\n"
57 #define BEGIN_BENV "BEGIN:BENV\r\n"
58 #define END_BENV "END:BENV\r\n"
59 #define BEGIN_BBODY "BEGIN:BBODY\r\n"
60 #define END_BBODY "END:BBODY\r\n"
61 #define ENCODING "ENCODING:%s\r\n"
62 #define CHARSET "CHARSET:%s\r\n"
63 #define LANGUAGE "LANGUAGE:%s\r\n"
64 #define LENGTH "LENGTH:%d\r\n"
65 #define MSG_BODY "BEGIN:MSG\r\n%s\r\nEND:MSG\r\n"
66 #define MSG_BODY_BEGIN "BEGIN:MSG\r\n"
67 #define MSG_BODY_END "\r\nEND:MSG\r\n"
68
69 extern guint64 current_push_map_id;
70
71 /* LCOV_EXCL_START */
72 static void __bt_map_parse_moved_mails(char *inbuf, int *from_box_id,
73                                 int *to_box_id, GList **mail_list)
74 {
75         if (!inbuf)
76                 return;
77
78         DBG("inbuf = %s", inbuf);
79
80         /* notification format: <from_box_id><0x01><to_box_id><0x01><<mail_id><,><mail_id>>*/
81         gchar **outer_tok;
82         char delimiter[2] = { 0x01, 0x00 };
83         outer_tok = g_strsplit_set(inbuf, delimiter, -1);
84         if (outer_tok == NULL) {
85                 ERR("outer_tok == NULL");
86                 return;
87         }
88         if (outer_tok[0] && strlen(outer_tok[0]) > 0)
89                 *from_box_id = atoi(outer_tok[0]);
90         if (outer_tok[1] && strlen(outer_tok[1]) > 0)
91                 *to_box_id = atoi(outer_tok[1]);
92         if (outer_tok[2] && strlen(outer_tok[2]) > 0) {
93                 gchar **inner_tok;
94                 inner_tok = g_strsplit_set(outer_tok[2], ",", -1);
95                 if (g_strv_length(inner_tok) == 1) { // only one mail_id exists without ","
96                         int mail_id = atoi(outer_tok[2]);
97 #ifdef ARCH64
98                         *mail_list = g_list_append(*mail_list, (void *)(uintptr_t) mail_id);
99 #else
100                         *mail_list = g_list_append(*mail_list, (void *) mail_id);
101 #endif
102                 } else {
103                         int i;
104                         for (i = 0; i < g_strv_length(inner_tok); ++i) {
105                                 if (!strcmp(inner_tok[i], "\"")) /* skip the empty token */
106                                         continue;
107                                 else {
108                                         int mail_id = atoi(inner_tok[i]);
109 #ifdef ARCH64
110                                         *mail_list = g_list_prepend(*mail_list, (void *)(uintptr_t) mail_id);
111 #else
112                                         *mail_list = g_list_prepend(*mail_list, (void *) mail_id);
113 #endif
114                                 }
115                         }
116                 }
117                 g_strfreev(inner_tok);
118         }
119         g_strfreev(outer_tok);
120
121         *mail_list = g_list_reverse(*mail_list);
122 }
123
124 char *__bt_email_get_path(int mailboxtype)
125 {
126         switch (mailboxtype) {
127         case EMAIL_MAILBOX_TYPE_INBOX:
128                 return g_strdup("TELECOM/MSG/INBOX");
129         case EMAIL_MAILBOX_TYPE_SENTBOX:
130                 return g_strdup("TELECOM/MSG/SENT");
131         case EMAIL_MAILBOX_TYPE_TRASH:
132                 return g_strdup("TELECOM/MSG/DELETED");
133         case EMAIL_MAILBOX_TYPE_DRAFT:
134                 return g_strdup("TELECOM/MSG/DRAFT");
135         case EMAIL_MAILBOX_TYPE_OUTBOX:
136                 return g_strdup("TELECOM/MSG/OUTBOX");
137         }
138         return g_strdup("");
139 }
140
141 static void __bt_email_subscription_callback(GDBusConnection *connection,
142                 const gchar *sender_name, const gchar *object_path,
143                 const gchar *interface_name, const gchar *signal_name,
144                 GVariant *parameters, gpointer data)
145 {
146         int subtype = 0;
147         int data1 = 0;
148         int data2 = 0;
149         char *data3 = NULL;
150         int data4 = 0;
151
152         g_variant_get(parameters, "(iii&si)", &subtype, &data1,
153                         &data2, &data3, &data4);
154
155         if ((g_strcmp0(object_path, BT_EMAIL_STORAGE_PATH)) ||
156                         (g_strcmp0(signal_name, BT_EMAIL_STORAGE_SIGNAL)))
157                 return;
158
159         if (subtype == NOTI_MAIL_ADD) {
160                 /* Received values from Signal*/
161                 int account_id = data1;
162                 int mailid = data2;
163                 int mailbox_id = atoi(data3);
164                 /* Fetch Values */
165                 int default_account_id = -1;
166                 email_mailbox_t *mailbox = NULL;
167                 guint64 handle;
168
169                 DBG("Mail Added[AccountID: %d, MailID:%d, MailBoxID:%d]",
170                                 account_id, mailid, mailbox_id);
171
172                 if (email_load_default_account_id(&default_account_id)
173                                                 != EMAIL_ERROR_NONE) {
174                         ERR("Could not load default account");
175                         return;
176                 }
177                 DBG("Default account_id: %d", default_account_id);
178                 if (default_account_id != account_id) {
179                         ERR("Event not meant for default email account");
180                         return;
181                 }
182
183                 if (email_get_mailbox_by_mailbox_id(mailbox_id,
184                                 &mailbox) != EMAIL_ERROR_NONE) {
185                         ERR("Could not get mailbox info");
186                         return;
187                 }
188
189                 handle = _bt_add_id(mailid, BT_MAP_ID_EMAIL);
190                 if (mailbox) {
191                         if (mailbox->mailbox_type == EMAIL_MAILBOX_TYPE_INBOX) {
192                                 _bt_mns_client_event_notify("NewMessage", handle,
193                                                 "TELECOM/MSG/INBOX", "",
194                                                 "EMAIL");
195                         }
196                         email_free_mailbox(&mailbox, 1);
197                 }
198
199         } else if (subtype == NOTI_MAIL_MOVE_FINISH) {
200                 /* Received values from Signal*/
201                 /* DATA1[account_id] DATA2[move_type] DATA4[??]
202                  * DATA3[mailbox_id0x01updated_value0x01mail_id] */
203                 int account_id = data1;
204                 int from_mailbox_id = -1;
205                 int to_mailbox_id = -1;
206                 GList *mail_ids = NULL;
207
208                 /* Fetch Values */
209                 int default_account_id = -1;
210                 email_mailbox_t *mailbox_from = NULL;
211                 email_mailbox_t *mailbox_to = NULL;
212                 guint64 handle;
213
214                 __bt_map_parse_moved_mails(data3, &from_mailbox_id,
215                                         &to_mailbox_id, &mail_ids);
216
217                 DBG("Mail Moved[AccountID:%d From:%d, To:%d]", account_id,
218                                 from_mailbox_id, to_mailbox_id);
219
220                 if (email_load_default_account_id(&default_account_id)
221                                                 != EMAIL_ERROR_NONE) {
222                         ERR("Could not load default account");
223                         return;
224                 }
225                 DBG("Default account_id: %d", default_account_id);
226                 if (default_account_id != account_id) {
227                         ERR("Event not meant for default email account");
228                         return;
229                 }
230                 if (email_get_mailbox_by_mailbox_id(from_mailbox_id, &mailbox_from)
231                                                         != EMAIL_ERROR_NONE) {
232                         ERR("Could not get mailbox info");
233                         return;
234                 }
235                 if (email_get_mailbox_by_mailbox_id(to_mailbox_id, &mailbox_to)
236                                                         != EMAIL_ERROR_NONE) {
237                         ERR("Could not get mailbox info");
238                         if (from_mailbox_id)
239                                 email_free_mailbox(&mailbox_from, 1);
240                         return;
241                 }
242
243                 if (mailbox_to->mailbox_type == EMAIL_MAILBOX_TYPE_TRASH) {
244                         while (mail_ids) {
245 #ifdef ARCH64
246                                 int mailid = (int)(uintptr_t)(void*) mail_ids->data;
247 #else
248                                 int mailid = (int) mail_ids->data;
249 #endif
250                                 char *old_folder;
251                                 DBG("Mail ID[%d]", mailid);
252                                 if (mailid == 0)
253                                         break;
254
255                                 old_folder = __bt_email_get_path(mailbox_from->mailbox_type);
256                                 handle = _bt_add_id(mailid, BT_MAP_ID_EMAIL);
257                                 DBG("[MessageDeleted] Handle:%d", handle);
258                                 _bt_mns_client_event_notify("MessageShift", handle,
259                                                 "TELECOM/MSG/DELETED", old_folder, "EMAIL");
260                                 g_free(old_folder);
261                                 mail_ids = g_list_next(mail_ids);
262                         }
263                 } else if (mailbox_to->mailbox_type == EMAIL_MAILBOX_TYPE_SENTBOX
264                                 && mailbox_from->mailbox_type == EMAIL_MAILBOX_TYPE_OUTBOX) {
265                         while (mail_ids) {
266 #ifdef ARCH64
267                                 int mailid = (int)(uintptr_t)(void*) mail_ids->data;
268 #else
269                                 int mailid = (int) mail_ids->data;
270 #endif
271                                 DBG("Mail ID[%d]", mailid);
272                                 if (mailid == 0)
273                                         break;
274
275                                 handle = _bt_add_id(mailid, BT_MAP_ID_EMAIL);
276                                 DBG("[SendingSuccess] Handle:%d", handle);
277
278                                 _bt_mns_client_event_notify("MessageShift", handle,
279                                                 "TELECOM/MSG/SENT", "TELECOM/MSG/OUTBOX", "EMAIL");
280
281                                 _bt_mns_client_event_notify("SendingSuccess", handle,
282                                                 "TELECOM/MSG/SENT", "", "EMAIL");
283                                 mail_ids = g_list_next(mail_ids);
284                         }
285                 }
286
287                 email_free_mailbox(&mailbox_to, 1);
288                 email_free_mailbox(&mailbox_from, 1);
289         }
290 }
291 /* LCOV_EXCL_STOP */
292
293 gboolean _bt_map_start_email_service(void)
294 {
295         int err;
296         GDBusConnection *dbus_conn = NULL;
297         GError *error = NULL;
298         int signal_handler_storage = -1;
299         err = email_service_begin();
300         if (err != EMAIL_ERROR_NONE) {
301                 ERR("email_service_begin fail  error = %d\n", err);
302                 return FALSE;
303         }
304
305         /* LCOV_EXCL_START */
306         dbus_conn = g_bus_get_sync(G_BUS_TYPE_SYSTEM, NULL, &error);
307         if (error) {
308                 ERR("g_bus_get_sync() failed (%s)", error->message);
309                 g_error_free(error);
310                 email_service_end();
311                 return FALSE;
312         }
313
314         signal_handler_storage = g_dbus_connection_signal_subscribe(dbus_conn,
315                         NULL, BT_EMAIL_STORAGE_INTERFACE, BT_EMAIL_STORAGE_SIGNAL,
316                         BT_EMAIL_STORAGE_PATH, NULL, G_DBUS_SIGNAL_FLAGS_NONE,
317                         __bt_email_subscription_callback, NULL, NULL);
318
319         if (signal_handler_storage == -1) {
320                 ERR("Failed to g_dbus_connection_signal_subscribe()");
321                 g_object_unref(dbus_conn);
322                 email_service_end();
323                 return FALSE;
324         }
325
326         /* LCOV_EXCL_STOP */
327         return TRUE;
328 }
329
330 gboolean _bt_map_stop_email_service(void)
331 {
332         int err;
333
334         err = email_service_end();
335         if (err != EMAIL_ERROR_NONE) {
336                 /* LCOV_EXCL_START */
337                 ERR("email_service_end fail  error = %d\n", err);
338                 return FALSE;
339                 /* LCOV_EXCL_STOP */
340         }
341
342         return TRUE;
343 }
344
345 gboolean _bt_map_email_get_supported_folders(gboolean folders[FOLDER_COUNT][MSG_TYPES])
346 {
347         DBG("");
348         int account_id = 0;
349         int mailbox_count = 0;
350         int err;
351         int i;
352         email_mailbox_t *mailbox_list = NULL;
353         email_mailbox_t *temp = NULL;
354
355         err = email_load_default_account_id(&account_id);
356         if (err != EMAIL_ERROR_NONE)
357                 return FALSE;
358
359         /* LCOV_EXCL_START */
360         err = email_get_mailbox_list(account_id, EMAIL_MAILBOX_ALL,
361                                 &mailbox_list, &mailbox_count);
362         if (err != EMAIL_ERROR_NONE)
363                 return FALSE;
364
365         DBG("Count: %d", mailbox_count);
366         for (i = 0, temp = mailbox_list; i < mailbox_count; i++, temp++) {
367                 DBG("Folder:%s", temp->mailbox_name);
368                 if (!g_ascii_strncasecmp(temp->mailbox_name, "SENT", strlen("SENT"))) {
369                         folders[BT_MSG_SENT][BT_MSG_SOURCE_EMAIL] = TRUE;
370                         DBG("SENT");
371                 } else if (!g_ascii_strncasecmp(temp->mailbox_name, "DRAFT", strlen("DRAFT"))) {
372                         folders[BT_MSG_DRAFT][BT_MSG_SOURCE_EMAIL] = TRUE;
373                         DBG("DRAFT");
374                 } else if (!g_ascii_strncasecmp(temp->mailbox_name, "DELETED", strlen("DELETED")) ||
375                                 !g_ascii_strncasecmp(temp->mailbox_name, "TRASH", strlen("TRASH"))) {
376                         folders[BT_MSG_DELETED][BT_MSG_SOURCE_EMAIL] = TRUE;
377                         DBG("DELETED");
378                 } else if (!g_ascii_strncasecmp(temp->mailbox_name, "INBOX", strlen("INBOX"))) {
379                         folders[BT_MSG_INBOX][BT_MSG_SOURCE_EMAIL] = TRUE;
380                         DBG("INBOX");
381                 } else if (!g_ascii_strncasecmp(temp->mailbox_name, "OUTBOX", strlen("OUTBOX"))) {
382                         folders[BT_MSG_OUTBOX][BT_MSG_SOURCE_EMAIL] = TRUE;
383                         DBG("OUTBOX");
384                 } else if (!g_ascii_strncasecmp(temp->mailbox_name, "[gmail]", strlen("[gmail]"))) {
385                         DBG("GMAIL Folder");
386                         if (!g_ascii_strncasecmp(temp->mailbox_name, "[Gmail]/Drafts", strlen("[Gmail]/Drafts"))) {
387                                 folders[BT_MSG_DRAFT][BT_MSG_SOURCE_EMAIL] = TRUE;
388                                 DBG("[Gmail]/DRAFT");
389                         } else if (!g_ascii_strncasecmp(temp->mailbox_name, "[Gmail]/Sent", strlen("[Gmail]/Sent"))) {
390                                 folders[BT_MSG_SENT][BT_MSG_SOURCE_EMAIL] = TRUE;
391                                 DBG("[Gmail]/SENT");
392                         } else if (!g_ascii_strncasecmp(temp->mailbox_name, "[Gmail]/Trash", strlen("[Gmail]/Trash"))) {
393                                 folders[BT_MSG_DELETED][BT_MSG_SOURCE_EMAIL] = TRUE;
394                                 DBG("[Gmail]/Trash");
395                         }
396                 }
397         }
398
399         if (mailbox_list != NULL)
400                 email_free_mailbox(&mailbox_list, mailbox_count);
401
402         /* LCOV_EXCL_STOP */
403         return TRUE;
404 }
405
406 /* LCOV_EXCL_START */
407 static message_info_t *__bt_email_info_get(email_mail_list_item_t *email_struct,
408                                                         guint8 subject_len)
409 {
410         message_info_t *email_info = NULL;
411         email_mail_data_t *mail_data;
412         guint64 uid = 0;
413         time_t dptime;
414         char email_handle[BT_MAP_MSG_HANDLE_MAX] = {0,};
415         char msg_datetime[BT_MAP_TIMESTAMP_MAX_LEN] = {0,};
416         email_info = g_new0(message_info_t, 1);
417
418         uid = _bt_add_id(email_struct->mail_id, BT_MAP_ID_EMAIL);
419 #ifdef ARCH64
420         snprintf(email_handle, sizeof(email_handle), "%lx", uid);
421 #else
422         snprintf(email_handle, sizeof(email_handle), "%llx", uid);
423 #endif
424         DBG("******* MAP ID:%d, MailID:%d **********", uid, email_struct->mail_id);
425         email_info->handle = g_strdup(email_handle);
426
427         dptime = email_struct->date_time;
428         _get_msg_timestamp(&dptime, msg_datetime);
429
430         email_info->sender_name = g_strdup(email_struct->email_address_sender);
431         email_info->sender_addressing = g_strdup(email_struct->email_address_sender);
432         email_info->recipient_name = g_strdup(email_struct->email_address_recipient);
433         email_info->recipient_addressing = g_strdup(email_struct->email_address_recipient);
434
435         email_info->subject = g_strndup(email_struct->subject, subject_len);
436         email_info->datetime = g_strdup(msg_datetime);
437         email_info->time = dptime; // for sorting
438         email_info->type = g_strdup("EMAIL");
439         email_info->size = g_strdup_printf("%d", email_struct->mail_size);
440         email_info->reception_status = g_strdup("complete");
441         email_info->attachment_size = g_strdup("0");
442         email_info->replyto_addressing = g_strdup(
443                         email_struct->email_address_sender);
444
445         DBG("Seen Status: %d", email_struct->flags_seen_field);
446         if (email_struct->flags_seen_field)
447                 email_info->read = TRUE;
448         else
449                 email_info->read = FALSE;
450
451         DBG("Priority: %d", email_struct->priority);
452         if (email_struct->priority == EMAIL_MAIL_PRIORITY_HIGH)
453                 email_info->priority = TRUE;
454         else
455                 email_info->priority = FALSE;
456
457         email_info->text = FALSE;
458         email_info->protect = FALSE;
459
460         if (email_get_mail_data(email_struct->mail_id, &mail_data) != EMAIL_ERROR_NONE) {
461                 ERR("email_get_mail_data failed\n");
462                 return email_info;
463         }
464
465         if (mail_data->alias_sender) {
466                 g_free(email_info->sender_name);
467                 email_info->sender_name = g_strdup(mail_data->alias_sender);
468         }
469
470         if (mail_data->alias_recipient) {
471                 g_free(email_info->recipient_name);
472                 email_info->recipient_name = g_strdup(mail_data->alias_recipient);
473         }
474
475         if (mail_data->email_address_recipient) {
476                 g_free(email_info->recipient_addressing);
477                 email_info->recipient_addressing = g_strdup(mail_data->email_address_recipient);
478         }
479
480         return email_info;
481 }
482
483 static gboolean __bt_map_email_compare_folders(char *alias, char *folder)
484 {
485         DBG("Folder:%s, Alias:%s", folder, alias);
486
487         char *map_folder = NULL;
488
489         if (!g_ascii_strncasecmp(alias, "SENT", strlen("SENT"))) {
490                 map_folder = "SENT";
491         } else if (!g_ascii_strncasecmp(alias, "DRAFT", strlen("DRAFT"))) {
492                 map_folder = "DRAFT";
493         } else if (!g_ascii_strncasecmp(alias, "DELETED", strlen("DELETED")) ||
494                         !g_ascii_strncasecmp(alias, "TRASH", strlen("TRASH"))) {
495                 map_folder = "DELETED";
496         } else if (!g_ascii_strncasecmp(alias, "INBOX", strlen("INBOX"))) {
497                 map_folder = "INBOX";
498         } else if (!g_ascii_strncasecmp(alias, "OUTBOX", strlen("OUTBOX"))) {
499                 map_folder = "OUTBOX";
500         } else if (!g_ascii_strncasecmp(alias, "[gmail]", strlen("[gmail]"))) {
501                 DBG("GMAIL Folders");
502                 if (!g_ascii_strncasecmp(alias, "[Gmail]/Drafts", strlen("[Gmail]/Drafts")))
503                         map_folder = "DRAFT";
504                 else if (!g_ascii_strncasecmp(alias, "[Gmail]/Sent", strlen("[Gmail]/Sent")))
505                         map_folder = "SENT";
506                 else if (!g_ascii_strncasecmp(alias, "[Gmail]/Trash", strlen("[Gmail]/Trash")))
507                         map_folder = "DELETED";
508         }
509
510         DBG("Equivalent MAP Folder for Alias: %s", map_folder);
511         if (map_folder && g_ascii_strcasecmp(map_folder, folder) == 0)
512                 return TRUE;
513
514         return FALSE;
515 }
516
517 gboolean _bt_map_get_email_list(char *folder, int max,
518                 guint8 subject_len, map_msg_filter_t *filter,
519                 GSList **email_list, guint64 *count, gboolean *newmsg)
520 {
521         DBG("");
522         int i;
523         int ret;
524         int total = 0;
525         int account_id = 0;
526         int mailbox_count = 0;
527         int mail_count = 0;
528         int msg_count = 0;
529         char *type = NULL;
530
531         email_mailbox_t *mailbox_list = NULL;
532         email_mail_list_item_t *mail_list = NULL;
533         email_mail_list_item_t *temp = NULL;
534         email_list_filter_t *filter_list = NULL;
535         email_list_sorting_rule_t *sorting_rule_list = NULL;
536         GSList *list = NULL;
537
538         if (max == 0)
539                 max = 1024;
540
541         ret = email_load_default_account_id(&account_id);
542         if (ret != EMAIL_ERROR_NONE)
543                 return FALSE;
544         DBG("Account ID:%d", account_id);
545
546         ret = email_get_mailbox_list(account_id, EMAIL_MAILBOX_ALL,
547                                         &mailbox_list, &mailbox_count);
548         if (ret != EMAIL_ERROR_NONE || mailbox_list == NULL)
549                 return FALSE;
550
551         for (i = 0; i < mailbox_count; i++) {
552                 DBG("mailbox alias = %s \n", mailbox_list[i].alias);
553                 /* Optimize using mailbox_type */
554                 if (__bt_map_email_compare_folders(mailbox_list[i].mailbox_name, folder)) {
555                         total = mailbox_list[i].total_mail_count_on_server;
556                         DBG("Total mail on sever:%d\n", total);
557                         DBG("mailbox name:%s\n", mailbox_list[i].mailbox_name);
558                         DBG("mailbox ID:%d\n", mailbox_list[i].mailbox_id);
559                         break;
560                 }
561         }
562         DBG("");
563         if (total == 0) {
564                 email_free_mailbox(&mailbox_list, mailbox_count);
565                 return FALSE;
566         }
567         DBG("");
568         filter_list = g_new0(email_list_filter_t, 3);
569         filter_list[0].list_filter_item_type = EMAIL_LIST_FILTER_ITEM_RULE;
570         filter_list[0].list_filter_item.rule.target_attribute = EMAIL_MAIL_ATTRIBUTE_ACCOUNT_ID;
571         filter_list[0].list_filter_item.rule.rule_type = EMAIL_LIST_FILTER_RULE_EQUAL;
572         filter_list[0].list_filter_item.rule.key_value.integer_type_value = account_id;
573
574         filter_list[1].list_filter_item_type = EMAIL_LIST_FILTER_ITEM_OPERATOR;
575         filter_list[1].list_filter_item.operator_type = EMAIL_LIST_FILTER_OPERATOR_AND;
576
577         filter_list[2].list_filter_item_type = EMAIL_LIST_FILTER_ITEM_RULE;
578         filter_list[2].list_filter_item.rule.target_attribute = EMAIL_MAIL_ATTRIBUTE_MAILBOX_ID;
579         filter_list[2].list_filter_item.rule.rule_type = EMAIL_LIST_FILTER_RULE_EQUAL;
580         filter_list[2].list_filter_item.rule.key_value.integer_type_value = mailbox_list[i].mailbox_id;
581         DBG("mailbox ID:%d\n", mailbox_list[i].mailbox_id);
582
583         sorting_rule_list = g_new0(email_list_sorting_rule_t, 1);
584         sorting_rule_list[0].target_attribute = EMAIL_MAIL_ATTRIBUTE_DATE_TIME;
585         sorting_rule_list[0].sort_order = EMAIL_SORT_ORDER_DESCEND;
586         sorting_rule_list[0].force_boolean_check = false;
587
588         ret = email_get_mail_list_ex(filter_list, 3, sorting_rule_list, 1, -1,
589                         -1, &mail_list, &mail_count);
590         if (ret != EMAIL_ERROR_NONE) {
591                 DBG("Error Code:%d", ret);
592                 g_free(type);
593                 g_free(filter_list);
594                 g_free(sorting_rule_list);
595                 return FALSE;
596         }
597
598         DBG("Mail Count: %d", mail_count);
599         max = (max > mail_count) ? (mail_count) : max;
600         DBG("Max:%d", max);
601         for (i = 0, temp = mail_list; i < mail_count && msg_count < max; ++i, temp++) {
602                 message_info_t *email_info;
603
604                 email_info = __bt_email_info_get(temp, subject_len);
605
606                 if (!_bt_verify_read_status(email_info, filter->read_status) ||
607                                 !_bt_verify_receiver(email_info, filter->recipient) ||
608                                 !_bt_verify_sender(email_info, filter->originator) ||
609                                 !_bt_verify_time(email_info, filter) ||
610                                 !_bt_filter_priority(email_info, filter->priority) ||
611                                 !_bt_validate_msg_data(email_info)) {
612                         _bt_message_info_free((gpointer)email_info);
613                         continue;
614                 }
615
616                 list = g_slist_append(list, email_info);
617                 msg_count++;
618         }
619
620         *count = (guint64)mail_count;
621         *email_list = list;
622
623         email_free_mailbox(&mailbox_list, mailbox_count);
624
625         if (mail_list)
626                 free(mail_list);
627
628         g_free(filter_list);
629         g_free(sorting_rule_list);
630         g_free(type);
631         DBG("EXIT");
632         return TRUE;
633 }
634 /* LCOV_EXCL_STOP */
635
636 gboolean _bt_map_update_mailbox(char *folder)
637 {
638         int handle;
639         int ret;
640
641         ret = email_sync_header_for_all_account(&handle);
642         if (ret == EMAIL_ERROR_NONE) {
643                 DBG("Handle to stop download = %d \n", handle);
644         } else {
645                 ERR("Message Update failed \n");
646                 return FALSE;
647         }
648
649         return TRUE;
650 }
651
652 gboolean _bt_map_set_email_read_status(int mail_id, int read_status)
653 {
654         int ret;
655         email_mail_data_t *mail_data = NULL;
656
657         ret = email_get_mail_data(mail_id, &mail_data);
658         if (ret != EMAIL_ERROR_NONE) {
659                 ERR("email_get_mail_data failed\n"); /* LCOV_EXCL_LINE */
660                 return FALSE;
661         }
662
663         /* LCOV_EXCL_START */
664         ret = email_set_flags_field(mail_data->account_id, &mail_id, 1,
665                                 EMAIL_FLAGS_SEEN_FIELD, read_status, 0);
666         if (ret != EMAIL_ERROR_NONE) {
667                 email_free_mail_data(&mail_data, 1);
668                 return FALSE;
669         }
670
671         email_free_mail_data(&mail_data, 1);
672         /* LCOV_EXCL_STOP */
673         return TRUE;
674 }
675
676 gboolean _bt_map_set_email_delete_status(int mail_id, int read_status)
677 {
678         int ret;
679         email_mail_data_t *mail_data = NULL;
680
681         ret = email_get_mail_data(mail_id, &mail_data);
682         if (ret != EMAIL_ERROR_NONE)
683                 return FALSE;
684
685         /* LCOV_EXCL_START */
686         ret = email_delete_mail(mail_data->mailbox_id, &mail_id, 1, 1);
687         if (ret != EMAIL_ERROR_NONE) {
688                 email_free_mail_data(&mail_data, 1);
689                 return FALSE;
690         }
691
692         email_free_mail_data(&mail_data, 1);
693         /* LCOV_EXCL_STOP */
694         return TRUE;
695 }
696
697 /* LCOV_EXCL_START */
698 static gchar *__bt_get_email_folder_name(int mailboxtype)
699 {
700         switch (mailboxtype) {
701         case EMAIL_MAILBOX_TYPE_SENTBOX:
702                 return g_strdup("TELECOM/MSG/SENT");
703         case EMAIL_MAILBOX_TYPE_TRASH:
704                 return g_strdup("TELECOM/MSG/DELETED");
705         case EMAIL_MAILBOX_TYPE_OUTBOX:
706                 return g_strdup("TELECOM/MSG/OUTBOX");
707         case EMAIL_MAILBOX_TYPE_DRAFT:
708                 return g_strdup("TELECOM/MSG/DRAFT");
709         default:
710                 return g_strdup("TELECOM/MSG/INBOX");
711         }
712 }
713
714 static char *__bt_prepare_email_bmseg(email_mail_data_t *mail_data)
715 {
716         FN_START;
717         char *folder = NULL;
718         FILE *body_file;
719         long read_size;
720         long email_size;
721         GString *msg;
722         char *buf = NULL;
723
724         msg = g_string_new(BEGIN_BMSEG);
725         g_string_append(msg, BMSEG_VERSION);
726
727         DBG("Seen Flag: %d", mail_data->flags_seen_field);
728         if (mail_data->flags_seen_field)
729                 g_string_append_printf(msg, MSEG_STATUS, "READ");
730         else
731                 g_string_append_printf(msg, MSEG_STATUS, "UNREAD");
732
733         g_string_append_printf(msg, MSEG_TYPE, "EMAIL");
734
735         folder = __bt_get_email_folder_name(mail_data->mailbox_type);
736         g_string_append_printf(msg, FOLDER_PATH, folder);
737         g_free(folder);
738
739         /* List of recepient & sender */
740         DBG("Sender: %d", mail_data->email_address_sender);
741         DBG("Sender Alias: %d", mail_data->alias_sender);
742         g_string_append_printf(msg, EMAIL_VCARD, mail_data->email_address_sender,
743                         mail_data->email_address_sender);
744
745         g_string_append(msg, BEGIN_BENV);
746         g_string_append(msg, BEGIN_BBODY);
747
748
749         g_string_append_printf(msg, CHARSET, "UTF-8");
750         g_string_append_printf(msg, ENCODING, "8BIT");
751         DBG("Plain Message file: %s", mail_data->file_path_plain);
752         DBG("HTML Message file: %s", mail_data->file_path_html);
753         body_file = fopen(mail_data->file_path_plain, "r");
754         if (body_file == NULL) {
755                 DBG("NOT PLAIN TEXT MESSAGE");
756                 body_file = fopen(mail_data->file_path_html, "rb");
757         }
758
759         if (body_file != NULL) {
760                 fseek(body_file, 0, SEEK_END);
761                 email_size = ftell(body_file);
762                 if (email_size <= 0) {
763                         ERR("email_size is not a positive number");
764                         g_string_free(msg, TRUE);
765                         fclose(body_file);
766                         return NULL;
767                 }
768                 rewind(body_file);
769
770                 buf = (char *)g_malloc0(sizeof(char) * (email_size + 1));
771                 read_size = fread(buf, 1, email_size, body_file);
772                 fclose(body_file);
773                 DBG("MESSAGE: [%s]", buf);
774                 if (read_size != email_size) {
775                         ERR("Unequal Read size");
776                         g_string_free(msg, TRUE);
777                         g_free(buf);
778                         return NULL;
779                 }
780         } else {
781                 DBG("BODY of the MESSAGE NOT FOUND");
782                 buf = (char *)g_strdup("");
783         }
784 #ifdef ARCH64
785         g_string_append_printf(msg, LENGTH, (int)(unsigned int)strlen(buf));
786 #else
787         g_string_append_printf(msg, LENGTH, strlen(buf));
788 #endif
789         g_string_append_printf(msg, MSG_BODY, buf);
790
791
792         g_string_append(msg, END_BBODY);
793         g_string_append(msg, END_BENV);
794         g_string_append(msg, END_BMSEG);
795         g_free(buf);
796
797         FN_END;
798         return g_string_free(msg, FALSE);
799 }
800 /* LCOV_EXCL_STOP */
801
802 gboolean _bt_map_get_email_message(int mail_id, gboolean attach,
803                 gboolean transcode, gboolean first_request, gchar **bmseg)
804 {
805         DBG("ENTER==>");
806         int account_id;
807         int ret;
808         email_mail_data_t *mail_data = NULL;
809
810         ret = email_load_default_account_id(&account_id);
811         if (ret != EMAIL_ERROR_NONE)
812                 return FALSE;
813
814         /* LCOV_EXCL_START */
815         ret = email_get_mail_data(mail_id, &mail_data);
816         if (ret != EMAIL_ERROR_NONE)
817                 return FALSE;
818
819         *bmseg = __bt_prepare_email_bmseg(mail_data);
820
821         email_free_mail_data(&mail_data, 1);
822         DBG("EXIT==>");
823         /* LCOV_EXCL_STOP */
824         return TRUE;
825 }
826
827 /* LCOV_EXCL_START */
828 static int __bt_map_save_email_to_outbox(char *subject, char *body,
829                 char *recepients)
830 {
831         int type =  EMAIL_MAILBOX_TYPE_OUTBOX;
832         int account_id;
833         int mail_id = -1;
834         int ret;
835         struct stat st_buf;
836         FILE *body_file;
837
838         email_account_t *account_data = NULL;
839         email_mailbox_t *mailbox_data = NULL;
840         email_mail_data_t *mail_data = NULL;
841
842         DBG("email_mailbox_type_e :%d", type);
843         DBG("Subject: %s", subject);
844         DBG("Body: %s", body);
845         DBG("Recepients: %s", recepients);
846
847         ret = email_load_default_account_id(&account_id);
848         if (ret != EMAIL_ERROR_NONE)
849                 goto fail;
850
851         DBG("account_id %d", account_id);
852         ret = email_get_mailbox_by_mailbox_type(account_id, type,
853                         &mailbox_data);
854         if (ret != EMAIL_ERROR_NONE)
855                 goto fail;
856
857         ret = email_get_account(account_id, EMAIL_ACC_GET_OPT_FULL_DATA,
858                         &account_data);
859         if (ret != EMAIL_ERROR_NONE)
860                 goto fail;
861
862         mail_data = calloc(1, sizeof(email_mail_data_t));
863         if (mail_data == NULL) {
864                 ERR("Allocation Failed");
865                 goto fail;
866         }
867
868         mail_data->account_id = account_id;
869         mail_data->save_status = EMAIL_MAIL_STATUS_SEND_DELAYED;
870         mail_data->body_download_status = 1;
871         mail_data->flags_seen_field = 1;
872         mail_data->report_status = EMAIL_MAIL_REQUEST_DSN |
873                                                 EMAIL_MAIL_REQUEST_MDN;
874         mail_data->remaining_resend_times = 3;
875         mail_data->file_path_plain = g_strdup(BT_MAIL_TEMP_BODY);
876         mail_data->subject = g_strdup(subject);
877         mail_data->full_address_to = g_strdup(recepients);
878
879         /* Get Sender Address  from Account data*/
880         mail_data->full_address_from = g_strdup(account_data->user_email_address);
881
882         /* Get MailboxID and Type from mailbox data */
883         mail_data->mailbox_id = mailbox_data->mailbox_id;
884         mail_data->mailbox_type = mailbox_data->mailbox_type;
885
886         /* Save Body to a File */
887         body_file = fopen(BT_MAIL_TEMP_BODY, "w");
888         if (body_file == NULL) {
889                 ERR("fopen [%s]failed", BT_MAIL_TEMP_BODY);
890                 goto fail;
891         }
892
893         ret = fprintf(body_file, "%s", body);
894         fflush(body_file);
895         fclose(body_file);
896
897         /* Save Email */
898         ret = email_add_mail(mail_data, NULL, 0, NULL, 0);
899         if (ret != EMAIL_ERROR_NONE) {
900                 DBG("email_add_mail failed. [%d]\n", ret);
901                 if (!stat(mail_data->file_path_plain, &st_buf))
902                         remove(mail_data->file_path_plain);
903
904                 goto fail;
905         }
906
907         DBG("saved mail id = [%d]\n", mail_data->mail_id);
908         mail_id = mail_data->mail_id;
909
910 fail:
911         if (mailbox_data)
912                 email_free_mailbox(&mailbox_data, 1);
913         if (account_data)
914                 email_free_account(&account_data, 1);
915         if (mail_data)
916                 email_free_mail_data(&mail_data, 1);
917
918         return mail_id;
919 }
920
921 static int __bt_map_save_email_to_draft(char *subject,
922                                                 char *body, char *recepients)
923 {
924         int type =  EMAIL_MAILBOX_TYPE_DRAFT;
925         int account_id;
926         int mail_id = -1;
927         int ret;
928         struct stat st_buf;
929         FILE *body_file;
930
931         email_account_t *account_data = NULL;
932         email_mailbox_t *mailbox_data = NULL;
933         email_mail_data_t *mail_data = NULL;
934
935         DBG("email_mailbox_type_e :%d", type);
936         DBG("Subject: %s", subject);
937         DBG("Body: %s", body);
938         DBG("Recepients: %s", recepients);
939
940         ret = email_load_default_account_id(&account_id);
941         if (ret != EMAIL_ERROR_NONE)
942                 goto fail;
943
944         DBG("account_id %d", account_id);
945         ret = email_get_mailbox_by_mailbox_type(account_id, type,
946                                                 &mailbox_data);
947         if (ret != EMAIL_ERROR_NONE)
948                 goto fail;
949
950         ret = email_get_account(account_id, EMAIL_ACC_GET_OPT_FULL_DATA,
951                         &account_data);
952         if (ret != EMAIL_ERROR_NONE)
953                 goto fail;
954
955         mail_data = calloc(1, sizeof(email_mail_data_t));
956         if (mail_data == NULL) {
957                 ERR("Allocation Failed");
958                 goto fail;
959         }
960
961         mail_data->account_id = account_id;
962         mail_data->body_download_status = 1;
963         mail_data->flags_seen_field = 1;
964         mail_data->flags_draft_field = 1;
965         mail_data->report_status = EMAIL_MAIL_REPORT_NONE;
966         mail_data->remaining_resend_times = -1;
967         mail_data->subject = g_strdup(subject);
968         mail_data->full_address_to = g_strdup(recepients);
969
970         /* Get Sender Address  from Account data*/
971         mail_data->full_address_from = g_strdup(account_data->user_email_address);
972         email_free_account(&account_data, 1);
973
974         /* Get MailboxID and Type from mailbox data */
975         mail_data->mailbox_id = mailbox_data->mailbox_id;
976         mail_data->mailbox_type = mailbox_data->mailbox_type;
977         email_free_mailbox(&mailbox_data, 1);
978
979         /* Save Body to a File */
980         mail_data->file_path_plain = g_strdup(BT_MAIL_TEMP_BODY);
981
982         body_file = fopen(BT_MAIL_TEMP_BODY, "w");
983         if (body_file == NULL) {
984                 ERR("fopen [%s]failed", BT_MAIL_TEMP_BODY);
985                 goto fail;
986         }
987
988         ret = fprintf(body_file, "%s", body);
989         fflush(body_file);
990         fclose(body_file);
991
992         /* Save Email */
993         ret = email_add_mail(mail_data, NULL, 0, NULL, 0);
994         if (ret != EMAIL_ERROR_NONE) {
995                 DBG("email_add_mail failed. [%d]\n", ret);
996                 if (!stat(mail_data->file_path_plain, &st_buf))
997                         remove(mail_data->file_path_plain);
998
999                 goto fail;
1000         }
1001
1002         DBG("saved mail id = [%d]\n", mail_data->mail_id);
1003         mail_id = mail_data->mail_id;
1004
1005 fail:
1006         if (mailbox_data)
1007                 email_free_mailbox(&mailbox_data, 1);
1008         if (account_data)
1009                 email_free_account(&account_data, 1);
1010         if (mail_data)
1011                 email_free_mail_data(&mail_data, 1);
1012
1013         return mail_id;
1014 }
1015
1016 static int __bt_map_send_email(char *subject, char *body,
1017                 char *recepients, gboolean send)
1018 {
1019         int ret;
1020         int mail_id = -1;
1021         int handle;
1022
1023         if (send) {
1024                 DBG("Send Mail");
1025                 mail_id = __bt_map_save_email_to_outbox(subject,
1026                                         body, recepients);
1027                 if (mail_id) {
1028                         DBG("mail_id = %d\n", mail_id);
1029                         ret = email_send_mail(mail_id, &handle);
1030                         if (ret != EMAIL_ERROR_NONE)
1031                                 DBG("Sending failed[%d]\n", ret);
1032                 }
1033
1034         } else {
1035                 DBG("Save to Draft");
1036                 mail_id = __bt_map_save_email_to_draft(subject,
1037                                         body, recepients);
1038         }
1039
1040         return mail_id;
1041 }
1042
1043 static char *__bt_map_get_email_address(GSList *recepients)
1044 {
1045         GString *mailto = NULL;
1046
1047         while (recepients) {
1048                 char *address = recepients->data;
1049                 DBG("Email: %s", address);
1050                 if (email_verify_email_address(address) == EMAIL_ERROR_NONE) {
1051                         if (mailto == NULL) {
1052                                 mailto = g_string_new(address);
1053                         } else {
1054                                 g_string_append(mailto, "; ");
1055                                 g_string_append(mailto, address);
1056                         }
1057                 }
1058                 recepients = g_slist_next(recepients);
1059         }
1060
1061         return g_string_free(mailto, FALSE);
1062 }
1063
1064 gboolean _bt_map_push_email_data(struct bmsg_data *bmsg_info,
1065                 msg_send_option_t *option, char *folder)
1066 {
1067         FN_START;
1068         int id = -1;
1069         char *message = NULL;
1070         char *body = NULL;
1071         char *subject = NULL;
1072         GSList *recepients = NULL;
1073         gboolean send = FALSE;
1074         char *mailto = NULL;
1075
1076         DBG("Length of Folder String: %d", strlen(bmsg_info->folder));
1077         if (strlen(bmsg_info->folder) == 0) {
1078                 DBG("No Folder Info. Default to OUTBOX");
1079                 bmsg_info->folder = g_strdup(folder);
1080         }
1081
1082         DBG("FOLDER: %s", bmsg_info->folder);
1083         if (!g_ascii_strcasecmp(bmsg_info->folder, "OUTBOX") ||
1084                         !g_ascii_strcasecmp(bmsg_info->folder, "TELECOM/MSG/OUTBOX"))
1085                 send = TRUE;
1086
1087         message = bmsg_get_msg_body(bmsg_info, option->native);
1088         if (message == NULL)
1089                 goto done;
1090         DBG_SECURE("Message: %s", message);
1091
1092         if (!bmsg_parse_msg_body(message, &body, &subject))
1093                 goto done;
1094         DBG_SECURE("SUBJECT: %s", subject);
1095         DBG_SECURE("BODY: %s", body);
1096
1097         recepients = bmsg_get_msg_recepients(bmsg_info, BT_MAP_ID_EMAIL);
1098
1099         mailto = __bt_map_get_email_address(recepients);
1100         DBG("Email IDs: %s", mailto);
1101
1102         /* TODO : Write logic to Get Subject from bmessage
1103          */
1104
1105         id = __bt_map_send_email(subject, body, mailto, send);
1106         if (id == -1)
1107                 goto done;
1108
1109         _bt_update_id(current_push_map_id, id, BT_MAP_ID_EMAIL);
1110
1111 done:
1112         g_free(body);
1113         g_free(subject);
1114         g_free(message);
1115         g_slist_free(recepients);
1116
1117         if (id == -1) {
1118                 FN_END;
1119                 return FALSE;
1120         }
1121
1122         return TRUE;
1123 }
1124 /* LCOV_EXCL_STOP */
1125