source: trunk/daemon/daemon.c @ 14265

Last change on this file since 14265 was 14265, checked in by jordan, 8 years ago

(trunk, daemon) #5660 'Daemon should use event loop' -- fixed with patch by missionsix

  • Property svn:keywords set to Date Rev Author Id
File size: 22.9 KB
Line 
1/*
2 * This file Copyright (C) 2008-2014 Mnemosyne LLC
3 *
4 * It may be used under the GNU GPL versions 2 or 3
5 * or any future license endorsed by Mnemosyne LLC.
6 *
7 * $Id: daemon.c 14265 2014-04-27 22:01:51Z jordan $
8 */
9
10#include <errno.h>
11#include <stdio.h> /* printf */
12#include <stdlib.h> /* exit, atoi */
13
14#include <fcntl.h> /* open */
15#include <signal.h>
16#ifdef HAVE_SYSLOG
17#include <syslog.h>
18#endif
19#include <unistd.h> /* daemon */
20
21#include <event.h>
22#include <event2/buffer.h>
23#include <event2/event.h>
24
25#include <libtransmission/transmission.h>
26#include <libtransmission/tr-getopt.h>
27#include <libtransmission/log.h>
28#include <libtransmission/utils.h>
29#include <libtransmission/variant.h>
30#include <libtransmission/version.h>
31
32#ifdef USE_SYSTEMD_DAEMON
33 #include <systemd/sd-daemon.h>
34#else
35 static void sd_notify (int status UNUSED, const char * str UNUSED) { }
36 static void sd_notifyf (int status UNUSED, const char * fmt UNUSED, ...) { }
37#endif
38
39#include "watch.h"
40
41#define MY_NAME "transmission-daemon"
42
43#define MEM_K 1024
44#define MEM_K_STR "KiB"
45#define MEM_M_STR "MiB"
46#define MEM_G_STR "GiB"
47#define MEM_T_STR "TiB"
48
49#define DISK_K 1000
50#define DISK_B_STR  "B"
51#define DISK_K_STR "kB"
52#define DISK_M_STR "MB"
53#define DISK_G_STR "GB"
54#define DISK_T_STR "TB"
55
56#define SPEED_K 1000
57#define SPEED_B_STR  "B/s"
58#define SPEED_K_STR "kB/s"
59#define SPEED_M_STR "MB/s"
60#define SPEED_G_STR "GB/s"
61#define SPEED_T_STR "TB/s"
62
63#define LOGFILE_MODE_STR "a+"
64
65static bool paused = false;
66static bool seenHUP = false;
67static const char *logfileName = NULL;
68static FILE *logfile = NULL;
69static tr_session * mySession = NULL;
70static tr_quark key_pidfile = 0;
71static struct event_base *ev_base = NULL;
72
73/***
74****  Config File
75***/
76
77static const char *
78getUsage (void)
79{
80    return "Transmission " LONG_VERSION_STRING
81           "  http://www.transmissionbt.com/\n"
82           "A fast and easy BitTorrent client\n"
83           "\n"
84           MY_NAME " is a headless Transmission session\n"
85           "that can be controlled via transmission-remote\n"
86           "or the web interface.\n"
87           "\n"
88           "Usage: " MY_NAME " [options]";
89}
90
91static const struct tr_option options[] =
92{
93
94    { 'a', "allowed", "Allowed IP addresses. (Default: " TR_DEFAULT_RPC_WHITELIST ")", "a", 1, "<list>" },
95    { 'b', "blocklist", "Enable peer blocklists", "b", 0, NULL },
96    { 'B', "no-blocklist", "Disable peer blocklists", "B", 0, NULL },
97    { 'c', "watch-dir", "Where to watch for new .torrent files", "c", 1, "<directory>" },
98    { 'C', "no-watch-dir", "Disable the watch-dir", "C", 0, NULL },
99    { 941, "incomplete-dir", "Where to store new torrents until they're complete", NULL, 1, "<directory>" },
100    { 942, "no-incomplete-dir", "Don't store incomplete torrents in a different location", NULL, 0, NULL },
101    { 'd', "dump-settings", "Dump the settings and exit", "d", 0, NULL },
102    { 'e', "logfile", "Dump the log messages to this filename", "e", 1, "<filename>" },
103    { 'f', "foreground", "Run in the foreground instead of daemonizing", "f", 0, NULL },
104    { 'g', "config-dir", "Where to look for configuration files", "g", 1, "<path>" },
105    { 'p', "port", "RPC port (Default: " TR_DEFAULT_RPC_PORT_STR ")", "p", 1, "<port>" },
106    { 't', "auth", "Require authentication", "t", 0, NULL },
107    { 'T', "no-auth", "Don't require authentication", "T", 0, NULL },
108    { 'u', "username", "Set username for authentication", "u", 1, "<username>" },
109    { 'v', "password", "Set password for authentication", "v", 1, "<password>" },
110    { 'V', "version", "Show version number and exit", "V", 0, NULL },
111    { 810, "log-error", "Show error messages", NULL, 0, NULL },
112    { 811, "log-info", "Show error and info messages", NULL, 0, NULL },
113    { 812, "log-debug", "Show error, info, and debug messages", NULL, 0, NULL },
114    { 'w', "download-dir", "Where to save downloaded data", "w", 1, "<path>" },
115    { 800, "paused", "Pause all torrents on startup", NULL, 0, NULL },
116    { 'o', "dht", "Enable distributed hash tables (DHT)", "o", 0, NULL },
117    { 'O', "no-dht", "Disable distributed hash tables (DHT)", "O", 0, NULL },
118    { 'y', "lpd", "Enable local peer discovery (LPD)", "y", 0, NULL },
119    { 'Y', "no-lpd", "Disable local peer discovery (LPD)", "Y", 0, NULL },
120    { 830, "utp", "Enable uTP for peer connections", NULL, 0, NULL },
121    { 831, "no-utp", "Disable uTP for peer connections", NULL, 0, NULL },
122    { 'P', "peerport", "Port for incoming peers (Default: " TR_DEFAULT_PEER_PORT_STR ")", "P", 1, "<port>" },
123    { 'm', "portmap", "Enable portmapping via NAT-PMP or UPnP", "m", 0, NULL },
124    { 'M', "no-portmap", "Disable portmapping", "M", 0, NULL },
125    { 'L', "peerlimit-global", "Maximum overall number of peers (Default: " TR_DEFAULT_PEER_LIMIT_GLOBAL_STR ")", "L", 1, "<limit>" },
126    { 'l', "peerlimit-torrent", "Maximum number of peers per torrent (Default: " TR_DEFAULT_PEER_LIMIT_TORRENT_STR ")", "l", 1, "<limit>" },
127    { 910, "encryption-required",  "Encrypt all peer connections", "er", 0, NULL },
128    { 911, "encryption-preferred", "Prefer encrypted peer connections", "ep", 0, NULL },
129    { 912, "encryption-tolerated", "Prefer unencrypted peer connections", "et", 0, NULL },
130    { 'i', "bind-address-ipv4", "Where to listen for peer connections", "i", 1, "<ipv4 addr>" },
131    { 'I', "bind-address-ipv6", "Where to listen for peer connections", "I", 1, "<ipv6 addr>" },
132    { 'r', "rpc-bind-address", "Where to listen for RPC connections", "r", 1, "<ipv4 addr>" },
133    { 953, "global-seedratio", "All torrents, unless overridden by a per-torrent setting, should seed until a specific ratio", "gsr", 1, "ratio" },
134    { 954, "no-global-seedratio", "All torrents, unless overridden by a per-torrent setting, should seed regardless of ratio", "GSR", 0, NULL },
135    { 'x', "pid-file", "Enable PID file", "x", 1, "<pid-file>" },
136    { 0, NULL, NULL, NULL, 0, NULL }
137};
138
139static void
140showUsage (void)
141{
142    tr_getopt_usage (MY_NAME, getUsage (), options);
143    exit (0);
144}
145
146static void
147gotsig (int sig)
148{
149    switch (sig)
150    {
151        case SIGHUP:
152        {
153            if (!mySession)
154            {
155                tr_logAddInfo ("Deferring reload until session is fully started.");
156                seenHUP = true;
157            }
158            else
159            {
160                tr_variant settings;
161                const char * configDir;
162
163                /* reopen the logfile to allow for log rotation */
164                if (logfileName) {
165                    logfile = freopen (logfileName, LOGFILE_MODE_STR, logfile);
166                    if (!logfile)
167                        fprintf (stderr, "Couldn't reopen \"%s\": %s\n", logfileName, tr_strerror (errno));
168                }
169
170                configDir = tr_sessionGetConfigDir (mySession);
171                tr_logAddInfo ("Reloading settings from \"%s\"", configDir);
172                tr_variantInitDict (&settings, 0);
173                tr_variantDictAddBool (&settings, TR_KEY_rpc_enabled, true);
174                tr_sessionLoadSettings (&settings, configDir, MY_NAME);
175                tr_sessionSet (mySession, &settings);
176                tr_variantFree (&settings);
177                tr_sessionReloadBlocklists (mySession);
178            }
179            break;
180        }
181
182        default:
183            tr_logAddError ("Unexpected signal (%d) in daemon, closing.", sig);
184            /* no break */
185
186        case SIGINT:
187        case SIGTERM:
188            event_base_loopexit(ev_base, NULL);
189            break;
190    }
191}
192
193#if defined (WIN32)
194 #define USE_NO_DAEMON
195#elif !defined (HAVE_DAEMON) || defined (__UCLIBC__)
196 #define USE_TR_DAEMON
197#else
198 #define USE_OS_DAEMON
199#endif
200
201static int
202tr_daemon (int nochdir, int noclose)
203{
204#if defined (USE_OS_DAEMON)
205
206    return daemon (nochdir, noclose);
207
208#elif defined (USE_TR_DAEMON)
209
210    /* this is loosely based off of glibc's daemon () implementation
211     * http://sourceware.org/git/?p=glibc.git;a=blob_plain;f=misc/daemon.c */
212
213    switch (fork ()) {
214        case -1: return -1;
215        case 0: break;
216        default: _exit (0);
217    }
218
219    if (setsid () == -1)
220        return -1;
221
222    if (!nochdir)
223        chdir ("/");
224
225    if (!noclose) {
226        int fd = open ("/dev/null", O_RDWR, 0);
227        dup2 (fd, STDIN_FILENO);
228        dup2 (fd, STDOUT_FILENO);
229        dup2 (fd, STDERR_FILENO);
230        close (fd);
231    }
232
233    return 0;
234
235#else /* USE_NO_DAEMON */
236    return 0;
237#endif
238}
239
240static const char*
241getConfigDir (int argc, const char ** argv)
242{
243    int c;
244    const char * configDir = NULL;
245    const char * optarg;
246    const int ind = tr_optind;
247
248    while ((c = tr_getopt (getUsage (), argc, argv, options, &optarg))) {
249        if (c == 'g') {
250            configDir = optarg;
251            break;
252        }
253    }
254
255    tr_optind = ind;
256
257    if (configDir == NULL)
258        configDir = tr_getDefaultConfigDir (MY_NAME);
259
260    return configDir;
261}
262
263static void
264onFileAdded (tr_session * session, const char * dir, const char * file)
265{
266    char * filename = tr_buildPath (dir, file, NULL);
267    tr_ctor * ctor = tr_ctorNew (session);
268    int err = tr_ctorSetMetainfoFromFile (ctor, filename);
269
270    if (!err)
271    {
272        tr_torrentNew (ctor, &err, NULL);
273
274        if (err == TR_PARSE_ERR)
275            tr_logAddError ("Error parsing .torrent file \"%s\"", file);
276        else
277        {
278            bool trash = false;
279            int test = tr_ctorGetDeleteSource (ctor, &trash);
280
281            tr_logAddInfo ("Parsing .torrent file successful \"%s\"", file);
282
283            if (!test && trash)
284            {
285                tr_logAddInfo ("Deleting input .torrent file \"%s\"", file);
286                if (tr_remove (filename))
287                    tr_logAddError ("Error deleting .torrent file: %s", tr_strerror (errno));
288            }
289            else
290            {
291                char * new_filename = tr_strdup_printf ("%s.added", filename);
292                tr_rename (filename, new_filename);
293                tr_free (new_filename);
294            }
295        }
296    }
297
298    tr_ctorFree (ctor);
299    tr_free (filename);
300}
301
302static void
303printMessage (FILE * logfile, int level, const char * name, const char * message, const char * file, int line)
304{
305    if (logfile != NULL)
306    {
307        char timestr[64];
308        tr_logGetTimeStr (timestr, sizeof (timestr));
309        if (name)
310            fprintf (logfile, "[%s] %s %s (%s:%d)\n", timestr, name, message, file, line);
311        else
312            fprintf (logfile, "[%s] %s (%s:%d)\n", timestr, message, file, line);
313    }
314#ifdef HAVE_SYSLOG
315    else /* daemon... write to syslog */
316    {
317        int priority;
318
319        /* figure out the syslog priority */
320        switch (level) {
321            case TR_LOG_ERROR: priority = LOG_ERR; break;
322            case TR_LOG_DEBUG: priority = LOG_DEBUG; break;
323            default:           priority = LOG_INFO; break;
324        }
325
326        if (name)
327            syslog (priority, "%s %s (%s:%d)", name, message, file, line);
328        else
329            syslog (priority, "%s (%s:%d)", message, file, line);
330    }
331#endif
332}
333
334static void
335pumpLogMessages (FILE * logfile)
336{
337    const tr_log_message * l;
338    tr_log_message * list = tr_logGetQueue ();
339
340    for (l=list; l!=NULL; l=l->next)
341        printMessage (logfile, l->level, l->name, l->message, l->file, l->line);
342
343    if (logfile != NULL)
344        fflush (logfile);
345
346    tr_logFreeQueue (list);
347}
348
349static void
350reportStatus (void)
351{
352    const double up = tr_sessionGetRawSpeed_KBps (mySession, TR_UP);
353    const double dn = tr_sessionGetRawSpeed_KBps (mySession, TR_DOWN);
354
355    if (up>0 || dn>0)
356        sd_notifyf (0, "STATUS=Uploading %.2f KBps, Downloading %.2f KBps.\n", up, dn);
357    else
358        sd_notify (0, "STATUS=Idle.\n");
359}
360
361static void
362periodicUpdate (evutil_socket_t fd UNUSED, short what UNUSED, void *watchdir)
363{
364    dtr_watchdir_update (watchdir);
365
366    pumpLogMessages (logfile);
367
368    reportStatus ();
369}
370
371static tr_rpc_callback_status
372on_rpc_callback (tr_session            * session UNUSED,
373                 tr_rpc_callback_type    type,
374                 struct tr_torrent     * tor UNUSED,
375                 void                  * user_data UNUSED)
376{
377    if (type == TR_RPC_SESSION_CLOSE)
378        event_base_loopexit(ev_base, NULL);
379    return TR_RPC_OK;
380}
381
382int
383main (int argc, char ** argv)
384{
385    int c;
386    const char * optarg;
387    tr_variant settings;
388    bool boolVal;
389    bool loaded;
390    bool foreground = false;
391    bool dumpSettings = false;
392    const char * configDir = NULL;
393    const char * pid_filename;
394    dtr_watchdir * watchdir = NULL;
395    bool pidfile_created = false;
396    tr_session * session = NULL;
397    struct event *status_ev;
398
399    key_pidfile = tr_quark_new ("pidfile",  7);
400
401    signal (SIGINT, gotsig);
402    signal (SIGTERM, gotsig);
403#ifndef WIN32
404    signal (SIGHUP, gotsig);
405#endif
406
407    /* load settings from defaults + config file */
408    tr_variantInitDict (&settings, 0);
409    tr_variantDictAddBool (&settings, TR_KEY_rpc_enabled, true);
410    configDir = getConfigDir (argc, (const char**)argv);
411    loaded = tr_sessionLoadSettings (&settings, configDir, MY_NAME);
412
413    /* overwrite settings from the comamndline */
414    tr_optind = 1;
415    while ((c = tr_getopt (getUsage (), argc, (const char**)argv, options, &optarg))) {
416        switch (c) {
417            case 'a': tr_variantDictAddStr  (&settings, TR_KEY_rpc_whitelist, optarg);
418                      tr_variantDictAddBool (&settings, TR_KEY_rpc_whitelist_enabled, true);
419                      break;
420            case 'b': tr_variantDictAddBool (&settings, TR_KEY_blocklist_enabled, true);
421                      break;
422            case 'B': tr_variantDictAddBool (&settings, TR_KEY_blocklist_enabled, false);
423                      break;
424            case 'c': tr_variantDictAddStr  (&settings, TR_KEY_watch_dir, optarg);
425                      tr_variantDictAddBool (&settings, TR_KEY_watch_dir_enabled, true);
426                      break;
427            case 'C': tr_variantDictAddBool (&settings, TR_KEY_watch_dir_enabled, false);
428                      break;
429            case 941: tr_variantDictAddStr  (&settings, TR_KEY_incomplete_dir, optarg);
430                      tr_variantDictAddBool (&settings, TR_KEY_incomplete_dir_enabled, true);
431                      break;
432            case 942: tr_variantDictAddBool (&settings, TR_KEY_incomplete_dir_enabled, false);
433                      break;
434            case 'd': dumpSettings = true;
435                      break;
436            case 'e': logfile = fopen (optarg, LOGFILE_MODE_STR);
437                      if (logfile)
438                          logfileName = optarg;
439                      else
440                          fprintf (stderr, "Couldn't open \"%s\": %s\n", optarg, tr_strerror (errno));
441                      break;
442            case 'f': foreground = true;
443                      break;
444            case 'g': /* handled above */
445                      break;
446            case 'V': /* version */
447                      fprintf (stderr, "%s %s\n", MY_NAME, LONG_VERSION_STRING);
448                      exit (0);
449            case 'o': tr_variantDictAddBool (&settings, TR_KEY_dht_enabled, true);
450                      break;
451            case 'O': tr_variantDictAddBool (&settings, TR_KEY_dht_enabled, false);
452                      break;
453            case 'p': tr_variantDictAddInt (&settings, TR_KEY_rpc_port, atoi (optarg));
454                      break;
455            case 't': tr_variantDictAddBool (&settings, TR_KEY_rpc_authentication_required, true);
456                      break;
457            case 'T': tr_variantDictAddBool (&settings, TR_KEY_rpc_authentication_required, false);
458                      break;
459            case 'u': tr_variantDictAddStr (&settings, TR_KEY_rpc_username, optarg);
460                      break;
461            case 'v': tr_variantDictAddStr (&settings, TR_KEY_rpc_password, optarg);
462                      break;
463            case 'w': tr_variantDictAddStr (&settings, TR_KEY_download_dir, optarg);
464                      break;
465            case 'P': tr_variantDictAddInt (&settings, TR_KEY_peer_port, atoi (optarg));
466                      break;
467            case 'm': tr_variantDictAddBool (&settings, TR_KEY_port_forwarding_enabled, true);
468                      break;
469            case 'M': tr_variantDictAddBool (&settings, TR_KEY_port_forwarding_enabled, false);
470                      break;
471            case 'L': tr_variantDictAddInt (&settings, TR_KEY_peer_limit_global, atoi (optarg));
472                      break;
473            case 'l': tr_variantDictAddInt (&settings, TR_KEY_peer_limit_per_torrent, atoi (optarg));
474                      break;
475            case 800: paused = true;
476                      break;
477            case 910: tr_variantDictAddInt (&settings, TR_KEY_encryption, TR_ENCRYPTION_REQUIRED);
478                      break;
479            case 911: tr_variantDictAddInt (&settings, TR_KEY_encryption, TR_ENCRYPTION_PREFERRED);
480                      break;
481            case 912: tr_variantDictAddInt (&settings, TR_KEY_encryption, TR_CLEAR_PREFERRED);
482                      break;
483            case 'i': tr_variantDictAddStr (&settings, TR_KEY_bind_address_ipv4, optarg);
484                      break;
485            case 'I': tr_variantDictAddStr (&settings, TR_KEY_bind_address_ipv6, optarg);
486                      break;
487            case 'r': tr_variantDictAddStr (&settings, TR_KEY_rpc_bind_address, optarg);
488                      break;
489            case 953: tr_variantDictAddReal (&settings, TR_KEY_ratio_limit, atof (optarg));
490                      tr_variantDictAddBool (&settings, TR_KEY_ratio_limit_enabled, true);
491                      break;
492            case 954: tr_variantDictAddBool (&settings, TR_KEY_ratio_limit_enabled, false);
493                      break;
494            case 'x': tr_variantDictAddStr (&settings, key_pidfile, optarg);
495                      break;
496            case 'y': tr_variantDictAddBool (&settings, TR_KEY_lpd_enabled, true);
497                      break;
498            case 'Y': tr_variantDictAddBool (&settings, TR_KEY_lpd_enabled, false);
499                      break;
500            case 810: tr_variantDictAddInt (&settings,  TR_KEY_message_level, TR_LOG_ERROR);
501                      break;
502            case 811: tr_variantDictAddInt (&settings,  TR_KEY_message_level, TR_LOG_INFO);
503                      break;
504            case 812: tr_variantDictAddInt (&settings,  TR_KEY_message_level, TR_LOG_DEBUG);
505                      break;
506            case 830: tr_variantDictAddBool (&settings, TR_KEY_utp_enabled, true);
507                      break;
508            case 831: tr_variantDictAddBool (&settings, TR_KEY_utp_enabled, false);
509                      break;
510            default:  showUsage ();
511                      break;
512        }
513    }
514
515    if (foreground && !logfile)
516        logfile = stderr;
517
518    if (!loaded)
519    {
520        printMessage (logfile, TR_LOG_ERROR, MY_NAME, "Error loading config file -- exiting.", __FILE__, __LINE__);
521        return -1;
522    }
523
524    if (dumpSettings)
525    {
526        char * str = tr_variantToStr (&settings, TR_VARIANT_FMT_JSON, NULL);
527        fprintf (stderr, "%s", str);
528        tr_free (str);
529        return 0;
530    }
531
532    if (!foreground && tr_daemon (true, false) < 0)
533    {
534        char buf[256];
535        tr_snprintf (buf, sizeof (buf), "Failed to daemonize: %s", tr_strerror (errno));
536        printMessage (logfile, TR_LOG_ERROR, MY_NAME, buf, __FILE__, __LINE__);
537        exit (1);
538    }
539
540    sd_notifyf (0, "MAINPID=%d\n", (int)getpid()); 
541
542    /* setup event state */
543    ev_base = event_base_new();
544    if (ev_base == NULL)
545    {
546        char buf[256];
547        tr_snprintf(buf, sizeof(buf), "Failed to init daemon event state: %s", tr_strerror(errno));
548        printMessage (logfile, TR_LOG_ERROR, MY_NAME, buf, __FILE__, __LINE__);
549        exit (1);
550    }
551
552    /* start the session */
553    tr_formatter_mem_init (MEM_K, MEM_K_STR, MEM_M_STR, MEM_G_STR, MEM_T_STR);
554    tr_formatter_size_init (DISK_K, DISK_K_STR, DISK_M_STR, DISK_G_STR, DISK_T_STR);
555    tr_formatter_speed_init (SPEED_K, SPEED_K_STR, SPEED_M_STR, SPEED_G_STR, SPEED_T_STR);
556    session = tr_sessionInit ("daemon", configDir, true, &settings);
557    tr_sessionSetRPCCallback (session, on_rpc_callback, NULL);
558    tr_logAddNamedInfo (NULL, "Using settings from \"%s\"", configDir);
559    tr_sessionSaveSettings (session, configDir, &settings);
560
561    pid_filename = NULL;
562    tr_variantDictFindStr (&settings, key_pidfile, &pid_filename, NULL);
563    if (pid_filename && *pid_filename)
564    {
565        FILE * fp = fopen (pid_filename, "w+");
566        if (fp != NULL)
567        {
568            fprintf (fp, "%d", (int)getpid ());
569            fclose (fp);
570            tr_logAddInfo ("Saved pidfile \"%s\"", pid_filename);
571            pidfile_created = true;
572        }
573        else
574            tr_logAddError ("Unable to save pidfile \"%s\": %s", pid_filename, tr_strerror (errno));
575    }
576
577    if (tr_variantDictFindBool (&settings, TR_KEY_rpc_authentication_required, &boolVal) && boolVal)
578        tr_logAddNamedInfo (MY_NAME, "requiring authentication");
579
580    mySession = session;
581
582    /* If we got a SIGHUP during startup, process that now. */
583    if (seenHUP)
584        gotsig (SIGHUP);
585
586    /* maybe add a watchdir */
587    {
588        const char * dir;
589
590        if (tr_variantDictFindBool (&settings, TR_KEY_watch_dir_enabled, &boolVal)
591            && boolVal
592            && tr_variantDictFindStr (&settings, TR_KEY_watch_dir, &dir, NULL)
593            && dir
594            && *dir)
595        {
596            tr_logAddInfo ("Watching \"%s\" for new .torrent files", dir);
597            watchdir = dtr_watchdir_new (mySession, dir, onFileAdded);
598        }
599    }
600
601    /* load the torrents */
602    {
603        tr_torrent ** torrents;
604        tr_ctor * ctor = tr_ctorNew (mySession);
605        if (paused)
606            tr_ctorSetPaused (ctor, TR_FORCE, true);
607        torrents = tr_sessionLoadTorrents (mySession, ctor, NULL);
608        tr_free (torrents);
609        tr_ctorFree (ctor);
610    }
611
612#ifdef HAVE_SYSLOG
613    if (!foreground)
614        openlog (MY_NAME, LOG_CONS|LOG_PID, LOG_DAEMON);
615#endif
616
617    /* Create new timer event to report daemon status */
618    {
619        struct timeval one_sec = { 1, 0 };
620        status_ev = event_new(ev_base, -1, EV_PERSIST, &periodicUpdate, watchdir);
621        if (status_ev == NULL)
622        {
623            tr_logAddError("Failed to create status event %s", tr_strerror(errno));
624            goto cleanup;
625        }
626        if (event_add(status_ev, &one_sec) == -1)
627        {
628            tr_logAddError("Failed to add status event %s", tr_strerror(errno));
629            goto cleanup;
630        }
631    }
632
633    sd_notify( 0, "READY=1\n" );
634
635    /* Run daemon event loop */
636    if (event_base_dispatch(ev_base) == -1)
637    {
638        tr_logAddError("Failed to launch daemon event loop: %s", tr_strerror(errno));
639        goto cleanup;
640    }
641
642cleanup:
643    sd_notify( 0, "STATUS=Closing transmission session...\n" );
644    printf ("Closing transmission session...");
645
646    if (status_ev)
647    {
648        event_del(status_ev);
649        event_free(status_ev);
650    }
651    event_base_free(ev_base);
652
653    tr_sessionSaveSettings (mySession, configDir, &settings);
654    dtr_watchdir_free (watchdir);
655    tr_sessionClose (mySession);
656    pumpLogMessages (logfile);
657    printf (" done.\n");
658
659    /* shutdown */
660#if HAVE_SYSLOG
661    if (!foreground)
662    {
663        syslog (LOG_INFO, "%s", "Closing session");
664        closelog ();
665    }
666#endif
667
668    /* cleanup */
669    if (pidfile_created)
670        tr_remove (pid_filename);
671    tr_variantFree (&settings);
672    sd_notify (0, "STATUS=\n");
673    return 0;
674}
Note: See TracBrowser for help on using the repository browser.