3 #include <sys/socket.h>
12 #include <net_connection.h>
14 #include "download-provider-config.h"
15 #include "download-provider-log.h"
16 #include "download-provider-pthread.h"
17 #include "download-provider-notification.h"
18 #include "download-provider-ipc.h"
19 #include "download-provider-db.h"
20 #include "download-provider-utils.h"
22 #include "download-agent-defs.h"
23 #include "download-agent-interface.h"
25 int _init_agent(void);
26 void _deinit_agent(void);
27 static void __downloading_info_cb(user_downloading_info_t *download_info,
29 static void __download_info_cb(user_download_info_t *download_info,
31 static void __notify_cb(user_notify_info_t *notify_info, void *user_data);
32 static int __change_error(int err);
33 static int __change_state(da_state state);
35 void TerminateDaemon(int signo);
37 pthread_attr_t g_download_provider_thread_attr;
38 fd_set g_download_provider_socket_readset;
39 fd_set g_download_provider_socket_exceptset;
41 void *_start_download(void *args)
46 download_clientinfo_slot *clientinfoslot =
47 (download_clientinfo_slot *) args;
48 if (!clientinfoslot) {
49 TRACE_DEBUG_MSG("[NULL-CHECK] download_clientinfo_slot");
52 download_clientinfo *clientinfo =
53 (download_clientinfo *) clientinfoslot->clientinfo;
55 TRACE_DEBUG_MSG("[NULL-CHECK] download_clientinfo");
59 CLIENT_MUTEX_LOCK(&(clientinfo->client_mutex));
60 clientinfo->state = DOWNLOAD_STATE_READY;
61 clientinfo->err = DOWNLOAD_ERROR_NONE;
62 CLIENT_MUTEX_UNLOCK(&(clientinfo->client_mutex));
64 // call start_download() of download-agent
65 if (clientinfo->requestinfo->headers.rows > 0) {
68 char **req_header = NULL;
69 len = clientinfo->requestinfo->headers.rows;
70 req_header = calloc(len, sizeof(char *));
72 TRACE_DEBUG_MSG("fail to calloc");
75 for (i = 0; i < len; i++)
77 strdup(clientinfo->requestinfo->headers.str[i].str);
78 if (clientinfo->requestinfo->install_path.length > 1) {
79 if (clientinfo->requestinfo->filename.length > 1)
81 da_start_download_with_extension(clientinfo->requestinfo->
83 DA_FEATURE_REQUEST_HEADER,
85 DA_FEATURE_INSTALL_PATH,
86 clientinfo->requestinfo->install_path.str,
88 clientinfo->requestinfo->filename.str,
90 (void *)clientinfoslot,
94 da_start_download_with_extension(clientinfo->requestinfo->
96 DA_FEATURE_REQUEST_HEADER,
98 DA_FEATURE_INSTALL_PATH,
99 clientinfo->requestinfo->install_path.str,
100 DA_FEATURE_USER_DATA,
101 (void *)clientinfoslot,
104 if (clientinfo->requestinfo->filename.length > 1)
106 da_start_download_with_extension(clientinfo->requestinfo->
108 DA_FEATURE_REQUEST_HEADER,
110 DA_FEATURE_FILE_NAME,
111 clientinfo->requestinfo->filename.str,
112 DA_FEATURE_USER_DATA,
113 (void *)clientinfoslot,
117 da_start_download_with_extension(clientinfo->requestinfo->
119 DA_FEATURE_REQUEST_HEADER,
121 DA_FEATURE_USER_DATA,
122 (void *)clientinfoslot,
125 for (i = 0; i < len; i++) {
130 if (clientinfo->requestinfo->install_path.length > 1) {
131 if (clientinfo->requestinfo->filename.length > 1)
133 da_start_download_with_extension(clientinfo->requestinfo->
135 DA_FEATURE_INSTALL_PATH,
136 clientinfo->requestinfo->install_path.str,
137 DA_FEATURE_FILE_NAME,
138 clientinfo->requestinfo->filename.str,
139 DA_FEATURE_USER_DATA,
140 (void *)clientinfoslot,
144 da_start_download_with_extension(clientinfo->requestinfo->
146 DA_FEATURE_INSTALL_PATH,
147 clientinfo->requestinfo->install_path.str,
148 DA_FEATURE_USER_DATA,
149 (void *)clientinfoslot,
152 if (clientinfo->requestinfo->filename.length > 1)
154 da_start_download_with_extension(clientinfo->requestinfo->
156 DA_FEATURE_FILE_NAME,
157 clientinfo->requestinfo->filename.str,
158 DA_FEATURE_USER_DATA,
159 (void *)clientinfoslot,
163 da_start_download_with_extension(clientinfo->requestinfo->
165 DA_FEATURE_USER_DATA,
166 (void *)clientinfoslot,
171 // if start_download() return error cause of maximun download limitation, set state to DOWNLOAD_STATE_PENDED.
172 if (da_ret == DA_ERR_ALREADY_MAX_DOWNLOAD) {
173 TRACE_DEBUG_INFO_MSG("change to pended request [%d]", da_ret);
174 CLIENT_MUTEX_LOCK(&(clientinfo->client_mutex));
175 clientinfo->state = DOWNLOAD_STATE_PENDED;
176 clientinfo->err = DOWNLOAD_ERROR_TOO_MANY_DOWNLOADS;
177 download_provider_db_requestinfo_update_column(clientinfo,
179 ipc_send_request_stateinfo(clientinfo);
180 CLIENT_MUTEX_UNLOCK(&(clientinfo->client_mutex));
182 } else if (da_ret != DA_RESULT_OK) {
183 TRACE_DEBUG_INFO_MSG("Fail to request start [%d]", da_ret);
184 /* FIXME : need to seperate in detail according to error return values */
185 CLIENT_MUTEX_LOCK(&(clientinfo->client_mutex));
186 clientinfo->state = DOWNLOAD_STATE_FAILED;
187 clientinfo->err = DOWNLOAD_ERROR_INVALID_PARAMETER;
188 download_provider_db_requestinfo_remove(clientinfo->
189 requestinfo->requestid);
190 ipc_send_request_stateinfo(clientinfo);
191 CLIENT_MUTEX_UNLOCK(&(clientinfo->client_mutex));
195 CLIENT_MUTEX_LOCK(&(clientinfo->client_mutex));
197 TRACE_DEBUG_INFO_MSG("started download [%d]", da_ret);
199 clientinfo->req_id = req_dl_id;
200 clientinfo->state = DOWNLOAD_STATE_DOWNLOADING;
201 clientinfo->err = DOWNLOAD_ERROR_NONE;
203 download_provider_db_requestinfo_update_column(clientinfo,
206 // sync return // client should be alive till this line at least.
207 ipc_send_request_stateinfo(clientinfo);
209 CLIENT_MUTEX_UNLOCK(&(clientinfo->client_mutex));
213 int _handle_new_connection(download_clientinfo_slot *clientinfo_list, download_clientinfo *request_clientinfo)
218 if (!clientinfo_list || !request_clientinfo ) {
219 TRACE_DEBUG_MSG("NULL-CHECK");
223 CLIENT_MUTEX_INIT(&(request_clientinfo->client_mutex), NULL);
227 sizeof(request_clientinfo->credentials);
229 (request_clientinfo->clientfd, SOL_SOCKET,
230 SO_PEERCRED, &request_clientinfo->credentials,
233 ("Client Info : pid=%d, uid=%d, gid=%d\n",
234 request_clientinfo->credentials.pid,
235 request_clientinfo->credentials.uid,
236 request_clientinfo->credentials.gid);
240 download_controls type =
241 ipc_receive_header(request_clientinfo->clientfd);
242 TRACE_DEBUG_INFO_MSG("[ACCEPT] HEADER : [%d] ", type);
243 // first of all, receive requestinfo .
244 if (type <= 0 || ipc_receive_request_msg(request_clientinfo) < 0) {
245 TRACE_DEBUG_MSG("Ignore this connection, Invalid command");
246 clear_clientinfo(request_clientinfo);
250 if (type == DOWNLOAD_CONTROL_STOP
251 || type == DOWNLOAD_CONTROL_GET_STATE_INFO
252 || type == DOWNLOAD_CONTROL_RESUME
253 || type == DOWNLOAD_CONTROL_PAUSE) {
254 // get requestid from socket.
255 if (request_clientinfo->requestinfo
256 && request_clientinfo->requestinfo->requestid > 0) {
257 // search requestid in slots.
258 int searchindex = get_same_request_slot_index
260 request_clientinfo->requestinfo->requestid);
261 if (type == DOWNLOAD_CONTROL_STOP) {
262 TRACE_DEBUG_INFO_MSG("Request : DOWNLOAD_CONTROL_STOP");
263 if (searchindex >= 0) {
264 if (da_cancel_download
265 (clientinfo_list[searchindex].clientinfo->req_id)
267 request_clientinfo->state = DOWNLOAD_STATE_STOPPED;
268 request_clientinfo->err = DOWNLOAD_ERROR_NONE;
269 if (clientinfo_list[searchindex].clientinfo->requestinfo
270 && clientinfo_list[searchindex].clientinfo->requestinfo->notification)
271 set_downloadedinfo_appfw_notification(clientinfo_list[searchindex].clientinfo);
272 download_provider_db_requestinfo_remove(request_clientinfo->requestinfo->requestid);
273 download_provider_db_history_new(clientinfo_list[searchindex].clientinfo);
275 request_clientinfo->state = DOWNLOAD_STATE_FAILED;
276 request_clientinfo->err = DOWNLOAD_ERROR_INVALID_PARAMETER;
279 request_clientinfo->state = DOWNLOAD_STATE_NONE;
280 request_clientinfo->err = DOWNLOAD_ERROR_NONE;
282 ipc_send_stateinfo(request_clientinfo);
283 } else if (type == DOWNLOAD_CONTROL_GET_STATE_INFO) {
284 // search slots/downloading db/history db
285 if (searchindex > 0) { // exist in slots (memory)
286 request_clientinfo->state =
287 clientinfo_list[searchindex].clientinfo->state;
288 request_clientinfo->err =
289 clientinfo_list[searchindex].clientinfo->err;
290 } else { //search downloading db or history db
291 download_dbinfo* dbinfo =
292 download_provider_db_get_info(
293 request_clientinfo->requestinfo->requestid);
294 if (dbinfo) { // found in downloading db..it means crashed job
295 request_clientinfo->state = DOWNLOAD_STATE_PENDED;
296 request_clientinfo->err = DOWNLOAD_ERROR_TOO_MANY_DOWNLOADS;
297 } else { // no exist in downloading db
298 dbinfo = download_provider_db_history_get_info(
299 request_clientinfo->requestinfo->requestid);
300 if (!dbinfo) { // no info anywhere
301 request_clientinfo->state = DOWNLOAD_STATE_NONE;
302 request_clientinfo->err = DOWNLOAD_ERROR_NONE;
303 } else { //history info
304 request_clientinfo->state = dbinfo->state;
305 request_clientinfo->err = DOWNLOAD_ERROR_NONE;
308 download_provider_db_info_free(dbinfo);
311 ipc_send_stateinfo(request_clientinfo);
312 // estabilish the spec of return value.
313 } else if (type == DOWNLOAD_CONTROL_PAUSE) {
314 if (searchindex >= 0) {
315 if (da_suspend_download
316 (clientinfo_list[searchindex].clientinfo->req_id)
318 request_clientinfo->state = DOWNLOAD_STATE_PAUSE_REQUESTED;
319 request_clientinfo->err = DOWNLOAD_ERROR_NONE;
321 request_clientinfo->state = DOWNLOAD_STATE_FAILED;
322 request_clientinfo->err = DOWNLOAD_ERROR_INVALID_PARAMETER;
325 request_clientinfo->state = DOWNLOAD_STATE_NONE;
326 request_clientinfo->err = DOWNLOAD_ERROR_NONE;
328 ipc_send_stateinfo(request_clientinfo);
329 } else if (type == DOWNLOAD_CONTROL_RESUME) {
330 if (searchindex >= 0) {
331 if (da_resume_download
332 (clientinfo_list[searchindex].clientinfo->req_id)
334 request_clientinfo->state = DOWNLOAD_STATE_DOWNLOADING;
335 request_clientinfo->err = DOWNLOAD_ERROR_NONE;
337 request_clientinfo->state = DOWNLOAD_STATE_FAILED;
338 request_clientinfo->err = DOWNLOAD_ERROR_INVALID_PARAMETER;
341 request_clientinfo->state = DOWNLOAD_STATE_NONE;
342 request_clientinfo->err = DOWNLOAD_ERROR_NONE;
344 ipc_send_stateinfo(request_clientinfo);
347 clear_clientinfo(request_clientinfo);
351 if (type != DOWNLOAD_CONTROL_START) {
353 ("Now, DOWNLOAD_CONTROL_START is only supported");
354 clear_clientinfo(request_clientinfo);
358 // check whether requestinfo has requestid or not.
359 if (request_clientinfo->requestinfo
360 && request_clientinfo->requestinfo->requestid > 0) {
361 // search same request id.
362 int searchindex = get_same_request_slot_index(clientinfo_list,
363 request_clientinfo->requestinfo->requestid);
364 if (searchindex < 0) {
365 TRACE_DEBUG_INFO_MSG("Not Found Same Request ID");
367 request_clientinfo->state = DOWNLOAD_STATE_FAILED;
368 request_clientinfo->err = DOWNLOAD_ERROR_INVALID_PARAMETER;
369 ipc_send_request_stateinfo(request_clientinfo);
370 clear_clientinfo(request_clientinfo);
372 } else { // found request id. // how to deal etag ?
374 TRACE_DEBUG_INFO_MSG("Found Same Request ID slot[%d]", searchindex);
375 CLIENT_MUTEX_LOCK(&(request_clientinfo->client_mutex));
376 // close previous socket.
377 if (clientinfo_list[searchindex].clientinfo->clientfd > 0)
378 close(clientinfo_list[searchindex].clientinfo->clientfd);
379 // change to new socket.
380 clientinfo_list[searchindex].clientinfo->clientfd =
381 request_clientinfo->clientfd;
382 ipc_send_request_stateinfo(clientinfo_list[searchindex].clientinfo);
384 clientinfo_list[searchindex].clientinfo->requestinfo->callbackinfo =
385 request_clientinfo->requestinfo->callbackinfo;
386 clientinfo_list[searchindex].clientinfo->requestinfo->notification =
387 request_clientinfo->requestinfo->notification;
388 request_clientinfo->clientfd = 0; // prevent to not be disconnected.
389 CLIENT_MUTEX_UNLOCK(&(request_clientinfo->client_mutex));
390 clear_clientinfo(request_clientinfo);
396 searchslot = get_empty_slot_index(clientinfo_list);
397 if (searchslot < 0) {
398 TRACE_DEBUG_MSG("download-provider is busy, try later");
399 clear_clientinfo(request_clientinfo);
400 sleep(5); // provider need the time of refresh.
403 // create new unique id, and insert info to DB.
404 if (request_clientinfo->requestinfo
405 && request_clientinfo->requestinfo->requestid <= 0) {
406 request_clientinfo->requestinfo->requestid =
407 get_download_request_id();
408 if (download_provider_db_requestinfo_new
409 (request_clientinfo) < 0) {
410 clear_clientinfo(request_clientinfo);
411 sleep(5); // provider need the time of refresh.
416 clientinfo_list[searchslot].clientinfo = request_clientinfo;
418 TRACE_DEBUG_INFO_MSG("New Connection slot [%d] max [%d] max once [%d]",
421 DA_MAX_DOWNLOAD_REQ_AT_ONCE);
423 if (get_downloading_count(clientinfo_list) >=
424 DA_MAX_DOWNLOAD_REQ_AT_ONCE) {
425 CLIENT_MUTEX_LOCK(&(clientinfo_list[searchslot].clientinfo->client_mutex));
426 // deal as pended job.
427 clientinfo_list[searchslot].clientinfo->state = DOWNLOAD_STATE_PENDED;
428 clientinfo_list[searchslot].clientinfo->err = DOWNLOAD_ERROR_TOO_MANY_DOWNLOADS;
429 download_provider_db_requestinfo_update_column
430 (clientinfo_list[searchslot].clientinfo,
432 ipc_send_request_stateinfo(clientinfo_list[searchslot].clientinfo);
433 TRACE_DEBUG_INFO_MSG ("Pended Request is saved to [%d/%d]",
434 searchslot, MAX_CLIENT);
435 CLIENT_MUTEX_UNLOCK(&(clientinfo_list[searchslot].clientinfo->client_mutex));
436 sleep(5); // provider need the time of refresh.
438 // create thread for receiving the reqeust info from client.
439 // and if possible, it will create the thread for listening the event.
441 (&clientinfo_list[searchslot].clientinfo->thread_pid,
442 &g_download_provider_thread_attr, _start_download,
443 &clientinfo_list[searchslot]) != 0) {
444 TRACE_DEBUG_INFO_MSG("failed to call pthread_create for client");
445 TRACE_DEBUG_INFO_MSG("Change to pended job");
446 CLIENT_MUTEX_LOCK(&(clientinfo_list[searchslot].clientinfo->client_mutex));
447 clientinfo_list[searchslot].clientinfo->state =
448 DOWNLOAD_STATE_PENDED;
449 clientinfo_list[searchslot].clientinfo->err =
450 DOWNLOAD_ERROR_TOO_MANY_DOWNLOADS;
451 download_provider_db_requestinfo_update_column
452 (clientinfo_list[searchslot].clientinfo,
454 ipc_send_request_stateinfo(clientinfo_list[searchslot].clientinfo);
455 CLIENT_MUTEX_UNLOCK(&(clientinfo_list[searchslot].clientinfo->client_mutex));
456 sleep(5); // provider need the time of refresh.
462 int _handle_client_request(download_clientinfo* clientinfo)
469 TRACE_DEBUG_MSG("NULL-CHECK");
473 switch (msgType = ipc_receive_header(clientinfo->clientfd)) {
474 case DOWNLOAD_CONTROL_STOP:
475 TRACE_DEBUG_INFO_MSG("DOWNLOAD_CONTROL_STOP");
476 da_ret = da_cancel_download(clientinfo->req_id);
477 CLIENT_MUTEX_LOCK(&(clientinfo->client_mutex));
478 if (da_ret != DA_RESULT_OK) {
479 /* FIXME : need to seperate in detail according to error return values */
480 clientinfo->state = DOWNLOAD_STATE_FAILED;
481 clientinfo->err = DOWNLOAD_ERROR_INVALID_PARAMETER;
482 TRACE_DEBUG_MSG("Fail to request cancel [%d]", da_ret);
484 clientinfo->state = DOWNLOAD_STATE_STOPPED;
485 clientinfo->err = DOWNLOAD_ERROR_NONE;
486 if (clientinfo->requestinfo) {
487 if (clientinfo->requestinfo->notification)
488 set_downloadedinfo_appfw_notification(clientinfo);
489 download_provider_db_requestinfo_remove(clientinfo->
490 requestinfo->requestid);
492 download_provider_db_history_new(clientinfo);
494 ipc_send_stateinfo(clientinfo);
495 CLIENT_MUTEX_UNLOCK(&(clientinfo->client_mutex));
497 case DOWNLOAD_CONTROL_PAUSE:
498 TRACE_DEBUG_INFO_MSG("DOWNLOAD_CONTROL_PAUSE");
499 da_ret = da_suspend_download(clientinfo->req_id);
500 CLIENT_MUTEX_LOCK(&(clientinfo->client_mutex));
501 if (da_ret != DA_RESULT_OK) {
502 /* FIXME : need to seperate in detail according to error return values */
503 clientinfo->state = DOWNLOAD_STATE_FAILED;
504 clientinfo->err = DOWNLOAD_ERROR_INVALID_PARAMETER;
505 TRACE_DEBUG_MSG("Fail to request suspend [%d]", da_ret);
507 clientinfo->state = DOWNLOAD_STATE_PAUSE_REQUESTED;
508 clientinfo->err = DOWNLOAD_ERROR_NONE;
510 ipc_send_stateinfo(clientinfo);
511 CLIENT_MUTEX_UNLOCK(&(clientinfo->client_mutex));
513 case DOWNLOAD_CONTROL_RESUME:
514 TRACE_DEBUG_INFO_MSG("DOWNLOAD_CONTROL_RESUME");
515 da_ret = da_resume_download(clientinfo->req_id);
516 CLIENT_MUTEX_LOCK(&(clientinfo->client_mutex));
517 if (da_ret != DA_RESULT_OK) {
518 /* FIXME : need to seperate in detail according to error return values */
519 clientinfo->state = DOWNLOAD_STATE_FAILED;
520 clientinfo->err = DOWNLOAD_ERROR_INVALID_PARAMETER;
521 TRACE_DEBUG_MSG("Fail to request resume [%d]", da_ret);
523 clientinfo->state = DOWNLOAD_STATE_DOWNLOADING;
524 clientinfo->err = DOWNLOAD_ERROR_NONE;
526 ipc_send_stateinfo(clientinfo);
527 CLIENT_MUTEX_UNLOCK(&(clientinfo->client_mutex));
529 case DOWNLOAD_CONTROL_GET_STATE_INFO: // sync call
530 TRACE_DEBUG_INFO_MSG("DOWNLOAD_CONTROL_GET_STATE_INFO");
531 CLIENT_MUTEX_LOCK(&(clientinfo->client_mutex));
532 ipc_send_stateinfo(clientinfo);
533 CLIENT_MUTEX_UNLOCK(&(clientinfo->client_mutex));
535 case DOWNLOAD_CONTROL_GET_DOWNLOAD_INFO: // sync call
536 TRACE_DEBUG_INFO_MSG("DOWNLOAD_CONTROL_GET_DOWNLOAD_INFO");
537 CLIENT_MUTEX_LOCK(&(clientinfo->client_mutex));
538 ipc_send_downloadinfo(clientinfo);
539 CLIENT_MUTEX_UNLOCK(&(clientinfo->client_mutex));
543 TRACE_DEBUG_MSG("(Closed Socket ) terminate event thread (%d)",
545 // bloken socket... it seems the client is dead or closed by agent thread.
546 // close socket, this will break the loop. and terminate this thread.
547 clear_socket(clientinfo);
550 TRACE_DEBUG_MSG("Unknow message [%d]", msgType);
556 void *run_manage_download_server(void *args)
558 int listenfd = 0; // main socket to be albe to listen the new connection
561 fd_set rset, exceptset;
562 struct timeval timeout;
563 long flexible_timeout;
564 download_clientinfo_slot *clientinfo_list;
566 uint count_downloading_threads = 0;
567 download_clientinfo *request_clientinfo;
572 struct sockaddr_un listenaddr, clientaddr;
574 GMainLoop *mainloop = (GMainLoop *) args;
577 if (ret != DOWNLOAD_ERROR_NONE) {
578 TRACE_DEBUG_MSG("failed to init agent");
579 TerminateDaemon(SIGTERM);
582 clear_downloadinginfo_appfw_notification();
584 if ((listenfd = socket(AF_UNIX, SOCK_STREAM, 0)) < 0) {
585 TRACE_DEBUG_MSG("failed to create socket");
586 TerminateDaemon(SIGTERM);
590 bzero(&listenaddr, sizeof(listenaddr));
591 listenaddr.sun_family = AF_UNIX;
592 strcpy(listenaddr.sun_path, DOWNLOAD_PROVIDER_IPC);
594 if (bind(listenfd, (struct sockaddr *)&listenaddr, sizeof listenaddr) !=
596 TRACE_DEBUG_MSG("failed to call bind");
597 TerminateDaemon(SIGTERM);
601 if (chmod(listenaddr.sun_path, 0777) < 0) {
603 ("failed to change the permission of socket file");
604 TerminateDaemon(SIGTERM);
608 if (listen(listenfd, MAX_CLIENT) != 0) {
609 TRACE_DEBUG_MSG("failed to call listen");
610 TerminateDaemon(SIGTERM);
615 TRACE_DEBUG_INFO_MSG("Ready to listen IPC [%d][%s]", listenfd,
616 DOWNLOAD_PROVIDER_IPC);
618 // allocation the array structure for managing the clients.
620 (download_clientinfo_slot *) calloc(MAX_CLIENT,
621 sizeof(download_clientinfo_slot));
622 if (clientinfo_list == NULL) {
623 TRACE_DEBUG_MSG("failed to allocate the memory for client list");
624 TerminateDaemon(SIGTERM);
628 if (pthread_attr_init(&g_download_provider_thread_attr) != 0) {
629 TRACE_DEBUG_MSG("failed to call pthread_attr_init for client");
630 TerminateDaemon(SIGTERM);
633 if (pthread_attr_setdetachstate(&g_download_provider_thread_attr,
634 PTHREAD_CREATE_DETACHED) != 0) {
635 TRACE_DEBUG_MSG("failed to set detach option");
636 TerminateDaemon(SIGTERM);
640 flexible_timeout = DOWNLOAD_PROVIDER_CARE_CLIENT_MIN_INTERVAL;
642 FD_ZERO(&g_download_provider_socket_readset);
643 FD_ZERO(&g_download_provider_socket_exceptset);
644 FD_SET(listenfd, &g_download_provider_socket_readset);
645 FD_SET(listenfd, &g_download_provider_socket_exceptset);
647 while (g_main_loop_is_running(mainloop)) {
650 for (i=0; i < MAX_CLIENT; i++) {
651 if (!clientinfo_list[i].clientinfo)
654 if (clientinfo_list[i].clientinfo->state >= DOWNLOAD_STATE_FINISHED) {
655 clear_clientinfoslot(&clientinfo_list[i]);
660 rset = g_download_provider_socket_readset;
661 exceptset = g_download_provider_socket_exceptset;
663 memset(&timeout, 0x00, sizeof(struct timeval));
664 timeout.tv_sec = flexible_timeout;
666 if (select((maxfd + 1), &rset, 0, &exceptset, &timeout) < 0) {
668 ("select error, provider can't receive any request from client.");
669 TerminateDaemon(SIGTERM);
673 for (i=0; i < MAX_CLIENT; i++) { // find the socket received the packet.
674 if (!clientinfo_list[i].clientinfo)
676 // ignore it is not started yet.
677 if (clientinfo_list[i].clientinfo->state <= DOWNLOAD_STATE_READY)
679 // ignore if finished
680 if (clientinfo_list[i].clientinfo->state >= DOWNLOAD_STATE_FINISHED)
682 //Even if no socket, downloading should be progressed.
683 if (clientinfo_list[i].clientinfo->clientfd <= 0)
685 if (FD_ISSET(clientinfo_list[i].clientinfo->clientfd, &rset) > 0) {
686 TRACE_DEBUG_INFO_MSG("FD_ISSET [%d] readset slot[%d]",
687 clientinfo_list[i].clientinfo->clientfd, i);
688 _handle_client_request(clientinfo_list[i].clientinfo);
689 } else if (FD_ISSET(clientinfo_list[i].clientinfo->clientfd, &exceptset) > 0) {
690 TRACE_DEBUG_MSG("FD_ISSET [%d] exceptset slot[%d]", clientinfo_list[i].clientinfo->clientfd, i);
691 clear_clientinfoslot(&clientinfo_list[i]);
695 if (FD_ISSET(listenfd, &exceptset) > 0) {
696 TRACE_DEBUG_MSG("meet listenfd Exception of socket");
697 TerminateDaemon(SIGTERM);
699 } else if (FD_ISSET(listenfd, &rset) > 0) { // new connection
700 TRACE_DEBUG_INFO_MSG("FD_ISSET listenfd rset");
703 DOWNLOAD_PROVIDER_CARE_CLIENT_MIN_INTERVAL;
706 (download_clientinfo *) calloc(1,
707 sizeof(download_clientinfo));
708 if (!request_clientinfo) {
710 ("download-provider can't allocate the memory, try later");
711 clientlen = sizeof(clientaddr);
712 int clientfd = accept(listenfd,
713 (struct sockaddr *)&clientaddr, &clientlen);
714 close(clientfd); // disconnect.
715 sleep(5); // provider need the time of refresh.
719 clientlen = sizeof(clientaddr);
720 request_clientinfo->clientfd = accept(listenfd,
721 (struct sockaddr*)&clientaddr,
723 if (request_clientinfo->clientfd < 0) {
724 clear_clientinfo(request_clientinfo);
725 sleep(5); // provider need the time of refresh.
728 if (_handle_new_connection(clientinfo_list, request_clientinfo) < 0) {
732 // after starting the download by DA, event thread will start to get the event from client.
733 if (request_clientinfo && request_clientinfo->clientfd > 0) {
734 FD_SET(request_clientinfo->clientfd, &g_download_provider_socket_readset); // add new descriptor to set
735 FD_SET(request_clientinfo->clientfd, &g_download_provider_socket_exceptset);
736 if (request_clientinfo->clientfd > maxfd )
737 maxfd = request_clientinfo->clientfd; /* for select */
741 if (i >= MAX_CLIENT) { // timeout
742 // If there is better solution to be able to know
743 // the number of downloading threads, replace below rough codes.
744 count_downloading_threads =
745 get_downloading_count(clientinfo_list);
746 // check whether the number of downloading is already maximum.
747 if (count_downloading_threads >=
748 DA_MAX_DOWNLOAD_REQ_AT_ONCE)
751 // search pended request
752 for (searchslot = 0; searchslot < MAX_CLIENT;
754 if (clientinfo_list[searchslot].clientinfo) {
755 if (clientinfo_list[searchslot].clientinfo->state ==
756 DOWNLOAD_STATE_PENDED) {
758 ("Retry Pended Request [%d/%d] state [%d/%d]",
759 searchslot, MAX_CLIENT,
760 count_downloading_threads,
761 DA_MAX_DOWNLOAD_REQ_AT_ONCE);
762 // create thread for restarting the pended download.
764 (&clientinfo_list[searchslot].clientinfo->thread_pid,
765 &g_download_provider_thread_attr,
767 &clientinfo_list[searchslot]) != 0) {
769 ("failed to call pthread_create for client");
771 count_downloading_threads++;
772 usleep(1000); // sleep in busy state.
779 && count_downloading_threads <
780 DA_MAX_DOWNLOAD_REQ_AT_ONCE) {
781 // Auto re-download feature. ethernet may be connected with other downloading items.
782 connection_h network_handle = NULL;
783 if (connection_create(&network_handle) < 0) {
785 ("Failed connection_create");
789 connection_ethernet_state_e system_network_state
790 = CONNECTION_ETHERNET_STATE_DISCONNECTED;
791 if (connection_get_ethernet_state(network_handle,
792 &system_network_state) !=
793 CONNECTION_ERROR_NONE)
795 ("Failed connection_get_ethernet_state");
797 ("ethernet check result : [%d]", (int)system_network_state);
799 connection_cellular_state_e system_cellular_state
800 = CONNECTION_CELLULAR_STATE_OUT_OF_SERVICE;
801 if (connection_get_cellular_state(network_handle,
802 &system_cellular_state) !=
803 CONNECTION_ERROR_NONE)
805 ("Failed connection_get_ethernet_state");
807 ("cellula check result : [%d]", (int)system_cellular_state);
809 connection_wifi_state_e system_wifi_state
810 = CONNECTION_WIFI_STATE_DEACTIVATED;
811 if (connection_get_wifi_state(network_handle,
812 &system_wifi_state) !=
813 CONNECTION_ERROR_NONE)
815 ("Failed connection_get_ethernet_state");
817 ("wifi check result : [%d]", (int)system_wifi_state);
819 if (connection_destroy(network_handle) !=
820 CONNECTION_ERROR_NONE)
822 ("Failed connection_destroy");
824 if (!(system_network_state
825 == CONNECTION_ETHERNET_STATE_CONNECTED
826 || system_cellular_state
827 == CONNECTION_CELLULAR_STATE_AVAILABLE
829 == CONNECTION_WIFI_STATE_CONNECTED))
832 // check auto-retrying list regardless state. pended state is also included to checking list.
834 download_dbinfo_list *db_list =
835 download_provider_db_get_list(DOWNLOAD_STATE_NONE);
836 if (!db_list || db_list->count <= 0) {
838 ("provider does not need to check DB anymore. in this life.");
839 check_retry = 0; // provider does not need to check DB anymore. in this life.
841 download_provider_db_list_free(db_list);
845 count_downloading_threads <
846 DA_MAX_DOWNLOAD_REQ_AT_ONCE
847 && i < db_list->count; i++) {
848 if (db_list->item[i].requestid <= 0)
850 if (get_same_request_slot_index
851 (clientinfo_list,db_list->item[i].requestid) < 0) {
852 // not found requestid in memory
854 ("Retry download [%d]",
855 db_list->item[i].requestid);
856 //search empty slot. copy db info to slot.
858 get_empty_slot_index(clientinfo_list);
859 if (searchslot < 0) {
861 ("download-provider is busy, try later");
863 flexible_timeout * 2;
866 // allocte requestinfo to empty slot.
868 (download_clientinfo *)
869 calloc(1, sizeof(download_clientinfo));
870 if (!request_clientinfo)
872 request_clientinfo->requestinfo
874 download_provider_db_get_requestinfo
876 if (!request_clientinfo->requestinfo) {
877 free(request_clientinfo);
878 request_clientinfo = NULL;
882 CLIENT_MUTEX_INIT(&(request_clientinfo->client_mutex), NULL);
883 request_clientinfo->state = DOWNLOAD_STATE_READY;
884 clientinfo_list[searchslot].clientinfo =
888 ("Retry download [%d/%d][%d/%d]",
889 searchslot, MAX_CLIENT,
890 count_downloading_threads,
891 DA_MAX_DOWNLOAD_REQ_AT_ONCE);
893 (&clientinfo_list[searchslot].clientinfo->thread_pid,
894 &g_download_provider_thread_attr,
896 &clientinfo_list[searchslot])
899 ("failed to call pthread_create for client");
900 clientinfo_list[searchslot].clientinfo->state =
901 DOWNLOAD_STATE_PENDED;
902 clientinfo_list[searchslot].clientinfo->err =
903 DOWNLOAD_ERROR_TOO_MANY_DOWNLOADS;
906 count_downloading_threads++;
907 usleep(1000); // sleep in busy state.
910 if (i >= db_list->count) // do not search anymore.
912 download_provider_db_list_free(db_list);
915 // save system resource (CPU)
916 if (check_retry == 0 && count_downloading_threads == 0
917 && flexible_timeout <
918 DOWNLOAD_PROVIDER_CARE_CLIENT_MAX_INTERVAL)
919 flexible_timeout = flexible_timeout * 2;
920 if (flexible_timeout >
921 DOWNLOAD_PROVIDER_CARE_CLIENT_MAX_INTERVAL)
923 DOWNLOAD_PROVIDER_CARE_CLIENT_MAX_INTERVAL;
924 TRACE_DEBUG_INFO_MSG("Next Timeout after [%ld] sec",
927 } // if (i >= MAX_CLIENT) { // timeout
930 FD_CLR(listenfd, &rset);
931 FD_CLR(listenfd, &exceptset);
932 FD_CLR(listenfd, &g_download_provider_socket_readset);
933 FD_CLR(listenfd, &g_download_provider_socket_exceptset);
935 // close accept socket.
941 // close all sockets for client. ..
942 // client thread will terminate by itself through catching this closing.
943 for (searchslot = 0; searchslot < MAX_CLIENT; searchslot++)
944 if (clientinfo_list[searchslot].clientinfo)
945 clear_clientinfoslot(&clientinfo_list[searchslot]);
948 free(clientinfo_list);
954 void __download_info_cb(user_download_info_t *download_info, void *user_data)
958 TRACE_DEBUG_MSG("[CRITICAL] clientinfoslot is NULL");
961 download_clientinfo_slot *clientinfoslot =
962 (download_clientinfo_slot *) user_data;
963 download_clientinfo *clientinfo =
964 (download_clientinfo *) clientinfoslot->clientinfo;
966 TRACE_DEBUG_MSG("[CRITICAL] clientinfo is NULL");
969 TRACE_DEBUG_INFO_MSG("id[%d],size[%lu]",
970 download_info->da_dl_req_id, download_info->file_size);
972 if (clientinfo->req_id != download_info->da_dl_req_id) {
973 TRACE_DEBUG_MSG("[CRITICAL] req_id[%d] da_dl_req_id[%d}",
975 download_info->da_dl_req_id);
978 CLIENT_MUTEX_LOCK(&(clientinfo->client_mutex));
979 if (!clientinfo->downloadinfo)
980 clientinfo->downloadinfo =
981 (download_content_info *) calloc(1, sizeof(download_content_info));
982 if (clientinfo->downloadinfo)
983 clientinfo->downloadinfo->file_size = download_info->file_size;
984 if (download_info->file_type) {
985 TRACE_DEBUG_INFO_MSG("mime[%s]", download_info->file_type);
987 len = strlen(download_info->file_type);
988 if (len > (DP_MAX_STR_LEN - 1))
989 len = DP_MAX_STR_LEN - 1;
990 if (clientinfo->downloadinfo) {
991 strncpy(clientinfo->downloadinfo->mime_type,
992 download_info->file_type, len);
993 download_provider_db_requestinfo_update_column
994 (clientinfo, DOWNLOAD_DB_MIMETYPE);
997 if (download_info->tmp_saved_path) {
999 TRACE_DEBUG_INFO_MSG("tmp path[%s]", download_info->tmp_saved_path);
1000 clientinfo->tmp_saved_path =
1001 strdup(download_info->tmp_saved_path);
1002 download_provider_db_requestinfo_update_column(clientinfo,
1003 DOWNLOAD_DB_SAVEDPATH);
1004 str = strrchr(download_info->tmp_saved_path, '/');
1008 if (len > (DP_MAX_STR_LEN - 1))
1009 len = DP_MAX_STR_LEN - 1;
1010 if (clientinfo->downloadinfo) {
1011 strncpy(clientinfo->downloadinfo->content_name,
1013 download_provider_db_requestinfo_update_column
1014 (clientinfo, DOWNLOAD_DB_FILENAME);
1015 TRACE_DEBUG_INFO_MSG("content_name[%s]",
1016 clientinfo->downloadinfo->
1022 if (clientinfo->requestinfo->callbackinfo.started)
1023 ipc_send_downloadinfo(clientinfo);
1025 CLIENT_MUTEX_UNLOCK(&(clientinfo->client_mutex));
1028 void __downloading_info_cb(user_downloading_info_t *download_info,
1033 TRACE_DEBUG_MSG("[CRITICAL] clientinfoslot is NULL");
1036 download_clientinfo_slot *clientinfoslot =
1037 (download_clientinfo_slot *) user_data;
1038 download_clientinfo *clientinfo =
1039 (download_clientinfo *) clientinfoslot->clientinfo;
1041 TRACE_DEBUG_MSG("[CRITICAL] clientinfo is NULL");
1044 CLIENT_MUTEX_LOCK(&(clientinfo->client_mutex));
1045 if (clientinfo->req_id != download_info->da_dl_req_id) {
1046 TRACE_DEBUG_MSG("[CRITICAL] req_id[%d] da_dl_req_id[%d}",
1048 download_info->da_dl_req_id);
1049 CLIENT_MUTEX_UNLOCK(&(clientinfo->client_mutex));
1052 if (!clientinfo->downloadinginfo)
1053 clientinfo->downloadinginfo = (downloading_state_info *) calloc(1,
1054 sizeof(downloading_state_info));
1055 if (clientinfo->downloadinginfo)
1056 clientinfo->downloadinginfo->received_size =
1057 download_info->total_received_size;
1058 if (download_info->saved_path) {
1059 TRACE_DEBUG_INFO_MSG("tmp path[%s]", download_info->saved_path);
1060 len = strlen(download_info->saved_path);
1061 if (len > (DP_MAX_PATH_LEN - 1))
1062 len = DP_MAX_PATH_LEN - 1;
1063 if (clientinfo->downloadinginfo)
1064 strncpy(clientinfo->downloadinginfo->saved_path,
1065 download_info->saved_path, len);
1066 /* FIXME : This should be reviewd again after smack rules is applied */
1067 if (chown(clientinfo->downloadinginfo->saved_path,
1068 clientinfo->credentials.uid,
1069 clientinfo->credentials.gid) < 0)
1070 TRACE_DEBUG_INFO_MSG("Fail to chown [%s]", strerror(errno));
1073 static size_t updated_second;
1074 time_t tt = time(NULL);
1075 struct tm *localTime = localtime(&tt);
1077 if (updated_second != localTime->tm_sec || download_info->saved_path) { // every 1 second.
1078 if (clientinfo->requestinfo
1079 && clientinfo->requestinfo->notification)
1080 set_downloadinginfo_appfw_notification(clientinfo);
1081 if (clientinfo->requestinfo->callbackinfo.progress)
1082 ipc_send_downloadinginfo(clientinfo);
1083 updated_second = localTime->tm_sec;
1085 CLIENT_MUTEX_UNLOCK(&(clientinfo->client_mutex));
1088 void __notify_cb(user_notify_info_t *notify_info, void *user_data)
1091 TRACE_DEBUG_MSG("[CRITICAL] clientinfoslot is NULL");
1094 download_clientinfo_slot *clientinfoslot =
1095 (download_clientinfo_slot *) user_data;
1096 download_clientinfo *clientinfo =
1097 (download_clientinfo *) clientinfoslot->clientinfo;
1099 TRACE_DEBUG_MSG("[CRITICAL] clientinfo is NULL");
1103 TRACE_DEBUG_INFO_MSG("id[%d],state[%d],err[%d]",
1104 notify_info->da_dl_req_id,
1105 notify_info->state, notify_info->err);
1106 if (clientinfo->req_id != notify_info->da_dl_req_id) {
1107 TRACE_DEBUG_MSG("[CRITICAL] req_id[%d] da_dl_req_id[%d}",
1108 clientinfo->req_id, notify_info->da_dl_req_id);
1112 CLIENT_MUTEX_LOCK(&(clientinfo->client_mutex));
1114 clientinfo->state = __change_state(notify_info->state);
1115 clientinfo->err = __change_error(notify_info->err);
1116 if (clientinfo->state == DOWNLOAD_STATE_FINISHED ||
1117 clientinfo->state == DOWNLOAD_STATE_FAILED) {
1118 if (clientinfo->requestinfo) {
1119 if (clientinfo->requestinfo->notification)
1120 set_downloadedinfo_appfw_notification(clientinfo);
1121 download_provider_db_requestinfo_remove(clientinfo->
1122 requestinfo->requestid);
1124 download_provider_db_history_new(clientinfo);
1125 TRACE_DEBUG_INFO_MSG("[TEST]Finish clientinfo[%p],fd[%d]",
1126 clientinfo, clientinfo->clientfd);
1129 TRACE_DEBUG_INFO_MSG("state[%d]", clientinfo->state);
1130 ipc_send_stateinfo(clientinfo);
1132 CLIENT_MUTEX_UNLOCK(&(clientinfo->client_mutex));
1135 int __change_state(da_state state)
1137 int ret = DOWNLOAD_STATE_NONE;
1139 case DA_STATE_WAITING:
1140 case DA_STATE_DOWNLOAD_STARTED:
1141 TRACE_DEBUG_INFO_MSG("DA_STATE_DOWNLOAD_STARTED");
1142 ret = DOWNLOAD_STATE_READY;
1144 case DA_STATE_DOWNLOADING:
1145 TRACE_DEBUG_INFO_MSG("DA_STATE_DOWNLOADING");
1146 ret = DOWNLOAD_STATE_DOWNLOADING;
1148 case DA_STATE_DOWNLOAD_COMPLETE:
1149 TRACE_DEBUG_INFO_MSG("DA_STATE_COMPLETE");
1150 ret = DOWNLOAD_STATE_INSTALLING;
1152 case DA_STATE_CANCELED:
1153 TRACE_DEBUG_INFO_MSG("DA_STATE_CANCELED");
1154 ret = DOWNLOAD_STATE_STOPPED;
1156 case DA_STATE_CANCELED_ALL:
1157 TRACE_DEBUG_INFO_MSG("DA_STATE_CANCELED_ALL");
1159 case DA_STATE_SUSPENDED:
1160 TRACE_DEBUG_INFO_MSG("DA_STATE_SUSPENDED");
1161 ret = DOWNLOAD_STATE_PAUSED;
1163 case DA_STATE_SUSPENDED_ALL:
1164 TRACE_DEBUG_INFO_MSG("DA_STATE_SUSPENDED_ALL");
1166 case DA_STATE_RESUMED:
1167 TRACE_DEBUG_INFO_MSG("DA_STATE_RESUMED");
1168 ret = DOWNLOAD_STATE_DOWNLOADING;
1170 case DA_STATE_RESUMED_ALL:
1171 TRACE_DEBUG_INFO_MSG("DA_STATE_RESUMED_ALL");
1173 case DA_STATE_FINISHED:
1174 TRACE_DEBUG_INFO_MSG("DA_STATE_FINISHED");
1175 ret = DOWNLOAD_STATE_FINISHED;
1177 case DA_STATE_FAILED:
1178 TRACE_DEBUG_INFO_MSG("DA_STATE_FAILED");
1179 ret = DOWNLOAD_STATE_FAILED;
1187 int __change_error(int err)
1189 int ret = DOWNLOAD_ERROR_UNKOWN;
1192 ret = DOWNLOAD_ERROR_NONE;
1194 case DA_ERR_INVALID_ARGUMENT:
1195 ret = DOWNLOAD_ERROR_INVALID_PARAMETER;
1197 case DA_ERR_FAIL_TO_MEMALLOC:
1198 ret = DOWNLOAD_ERROR_OUT_OF_MEMORY;
1200 case DA_ERR_UNREACHABLE_SERVER:
1201 ret = DOWNLOAD_ERROR_NETWORK_UNREACHABLE;
1203 case DA_ERR_HTTP_TIMEOUT:
1204 ret = DOWNLOAD_ERROR_CONNECTION_TIMED_OUT;
1206 case DA_ERR_DISK_FULL:
1207 ret = DOWNLOAD_ERROR_NO_SPACE;
1209 case DA_ERR_INVALID_STATE:
1210 ret = DOWNLOAD_ERROR_INVALID_STATE;
1212 case DA_ERR_NETWORK_FAIL:
1213 ret = DOWNLOAD_ERROR_CONNECTION_FAILED;
1215 case DA_ERR_INVALID_URL:
1216 ret = DOWNLOAD_ERROR_INVALID_URL;
1218 case DA_ERR_INVALID_INSTALL_PATH:
1219 ret = DOWNLOAD_ERROR_INVALID_DESTINATION;
1221 case DA_ERR_ALREADY_MAX_DOWNLOAD:
1222 ret = DOWNLOAD_ERROR_TOO_MANY_DOWNLOADS;
1224 case DA_ERR_FAIL_TO_INSTALL_FILE:
1225 ret = DOWNLOAD_ERROR_INSTALL_FAIL;
1227 case DA_ERR_FAIL_TO_CREATE_THREAD:
1228 case DA_ERR_FAIL_TO_OBTAIN_MUTEX:
1229 case DA_ERR_FAIL_TO_ACCESS_FILE:
1230 case DA_ERR_FAIL_TO_GET_CONF_VALUE:
1231 case DA_ERR_FAIL_TO_ACCESS_STORAGE:
1232 case DA_ERR_DLOPEN_FAIL:
1233 ret = DOWNLOAD_ERROR_IO_ERROR;
1242 da_client_cb_t da_cb = {
1245 __downloading_info_cb
1247 da_ret = da_init(&da_cb, DA_DOWNLOAD_MANAGING_METHOD_AUTO);
1248 if (da_ret != DA_RESULT_OK) {
1249 return DOWNLOAD_ERROR_FAIL_INIT_AGENT;
1251 return DOWNLOAD_ERROR_NONE;
1254 void _deinit_agent()