Don't start thread in constructor; tidy up thread in destructor (UpdateChecker).
[dcpomatic.git] / src / lib / update_checker.h
1 /*
2     Copyright (C) 2014 Carl Hetherington <cth@carlh.net>
3
4     This program is free software; you can redistribute it and/or modify
5     it under the terms of the GNU General Public License as published by
6     the Free Software Foundation; either version 2 of the License, or
7     (at your option) any later version.
8
9     This program is distributed in the hope that it will be useful,
10     but WITHOUT ANY WARRANTY; without even the implied warranty of
11     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12     GNU General Public License for more details.
13
14     You should have received a copy of the GNU General Public License
15     along with this program; if not, write to the Free Software
16     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17
18 */
19
20 /** @file  src/lib/update.h
21  *  @brief UpdateChecker class.
22  */
23
24 #include "signaller.h"
25 #include <curl/curl.h>
26 #include <boost/signals2.hpp>
27 #include <boost/thread/mutex.hpp>
28 #include <boost/thread/condition.hpp>
29 #include <boost/thread.hpp>
30
31 struct update_checker_test;
32
33 /** Class to check for the existance of an update for DCP-o-matic on a remote server */
34 class UpdateChecker : public Signaller, public boost::noncopyable
35 {
36 public:
37         ~UpdateChecker ();
38
39         void run ();
40
41         enum State {
42                 YES,    ///< there is an update
43                 FAILED, ///< the check failed, so we don't know
44                 NO,     ///< there is no update
45                 NOT_RUN ///< the check has not been run (yet)
46         };
47
48         /** @return state of the checker */
49         State state () {
50                 boost::mutex::scoped_lock lm (_data_mutex);
51                 return _state;
52         }
53
54         /** @return new stable version, if there is one */
55         boost::optional<std::string> stable () {
56                 boost::mutex::scoped_lock lm (_data_mutex);
57                 return _stable;
58         }
59
60         /** @return new test version, if there is one and Config is set to look for it */
61         boost::optional<std::string> test () {
62                 boost::mutex::scoped_lock lm (_data_mutex);
63                 return _test;
64         }
65
66         /** @return true if the last signal emission was the first */
67         bool last_emit_was_first () const {
68                 boost::mutex::scoped_lock lm (_data_mutex);
69                 return _emits == 1;
70         }
71
72         size_t write_callback (void *, size_t, size_t);
73
74         boost::signals2::signal<void (void)> StateChanged;
75
76         static UpdateChecker* instance ();
77
78 private:
79         friend struct update_checker_test;
80
81         static UpdateChecker* _instance;
82
83         static bool version_less_than (std::string const & a, std::string const & b);
84
85         UpdateChecker ();
86         void start ();
87         void set_state (State);
88         void thread ();
89
90         char* _buffer;
91         int _offset;
92         CURL* _curl;
93
94         /** mutex to protect _state, _stable, _test and _emits */
95         mutable boost::mutex _data_mutex;
96         State _state;
97         boost::optional<std::string> _stable;
98         boost::optional<std::string> _test;
99         int _emits;
100
101         boost::thread* _thread;
102         boost::mutex _process_mutex;
103         boost::condition _condition;
104         int _to_do;
105         bool _terminate;
106 };