2013-12-10 23:30:53 +00:00
|
|
|
/*
|
2015-02-17 21:11:34 +00:00
|
|
|
* ZeroTier One - Network Virtualization Everywhere
|
|
|
|
* Copyright (C) 2011-2015 ZeroTier, Inc.
|
2013-12-10 23:30:53 +00:00
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* --
|
|
|
|
*
|
|
|
|
* ZeroTier may be used and distributed under the terms of the GPLv3, which
|
|
|
|
* are available at: http://www.gnu.org/licenses/gpl-3.0.html
|
|
|
|
*
|
|
|
|
* If you would like to embed ZeroTier into a commercial application or
|
|
|
|
* redistribute it in a modified binary form, please contact ZeroTier Networks
|
|
|
|
* LLC. Start here: http://www.zerotier.com/
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef ZT_SOFTWAREUPDATER_HPP
|
|
|
|
#define ZT_SOFTWAREUPDATER_HPP
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
|
2014-01-06 00:24:12 +00:00
|
|
|
#include <string>
|
|
|
|
|
2013-12-10 23:30:53 +00:00
|
|
|
#include "Constants.hpp"
|
|
|
|
#include "Mutex.hpp"
|
|
|
|
#include "Utils.hpp"
|
|
|
|
#include "Defaults.hpp"
|
2014-01-06 00:24:12 +00:00
|
|
|
#include "Address.hpp"
|
2013-12-10 23:30:53 +00:00
|
|
|
|
2015-04-01 00:53:34 +00:00
|
|
|
/**
|
|
|
|
* Delay between fetches of the root topology update URL
|
|
|
|
*
|
|
|
|
* 86400000 = check once every 24 hours (this doesn't change often)
|
|
|
|
*/
|
|
|
|
#define ZT_UPDATE_ROOT_TOPOLOGY_CHECK_INTERVAL 86400000
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Minimum interval between attempts to do a software update
|
|
|
|
*/
|
|
|
|
#define ZT_UPDATE_MIN_INTERVAL 120000
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Maximum interval between checks for new versions
|
|
|
|
*/
|
|
|
|
#define ZT_UPDATE_MAX_INTERVAL 7200000
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Software update HTTP timeout in seconds
|
|
|
|
*/
|
|
|
|
#define ZT_UPDATE_HTTP_TIMEOUT 120
|
|
|
|
|
2013-12-10 23:30:53 +00:00
|
|
|
namespace ZeroTier {
|
|
|
|
|
|
|
|
class RuntimeEnvironment;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Software updater
|
|
|
|
*/
|
|
|
|
class SoftwareUpdater
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
SoftwareUpdater(const RuntimeEnvironment *renv);
|
|
|
|
~SoftwareUpdater();
|
|
|
|
|
2014-02-07 17:13:08 +00:00
|
|
|
/**
|
|
|
|
* Remove old updates in updates.d
|
|
|
|
*/
|
|
|
|
void cleanOldUpdates();
|
|
|
|
|
2013-12-10 23:30:53 +00:00
|
|
|
/**
|
|
|
|
* Called on each version message from a peer
|
|
|
|
*
|
|
|
|
* If a peer has a newer version, that causes an update to be started.
|
|
|
|
*
|
|
|
|
* @param vmaj Peer's major version
|
|
|
|
* @param vmin Peer's minor version
|
|
|
|
* @param rev Peer's revision
|
|
|
|
*/
|
2014-08-16 16:08:52 +00:00
|
|
|
void sawRemoteVersion(unsigned int vmaj,unsigned int vmin,unsigned int rev);
|
2013-12-10 23:30:53 +00:00
|
|
|
|
2013-12-11 21:00:18 +00:00
|
|
|
/**
|
|
|
|
* Check for updates now regardless of last check time or version
|
2013-12-27 04:57:17 +00:00
|
|
|
*
|
|
|
|
* This only starts a check if one is not in progress. Otherwise it does
|
|
|
|
* nothing.
|
2013-12-11 21:00:18 +00:00
|
|
|
*/
|
2014-08-16 16:08:52 +00:00
|
|
|
void checkNow();
|
2013-12-11 21:00:18 +00:00
|
|
|
|
2013-12-27 04:57:17 +00:00
|
|
|
/**
|
|
|
|
* Check for updates now if it's been longer than ZT_UPDATE_MAX_INTERVAL
|
|
|
|
*
|
|
|
|
* This is called periodically from the main loop.
|
|
|
|
*/
|
|
|
|
inline void checkIfMaxIntervalExceeded(uint64_t now)
|
|
|
|
{
|
|
|
|
if ((now - _lastUpdateAttempt) >= ZT_UPDATE_MAX_INTERVAL)
|
|
|
|
checkNow();
|
|
|
|
}
|
|
|
|
|
2013-12-10 23:30:53 +00:00
|
|
|
/**
|
|
|
|
* Pack three-component version into a 64-bit integer
|
|
|
|
*
|
|
|
|
* @param vmaj Major version (0..65535)
|
|
|
|
* @param vmin Minor version (0..65535)
|
|
|
|
* @param rev Revision (0..65535)
|
2013-12-11 21:00:18 +00:00
|
|
|
* @return Version packed into an easily comparable 64-bit integer
|
2013-12-10 23:30:53 +00:00
|
|
|
*/
|
|
|
|
static inline uint64_t packVersion(unsigned int vmaj,unsigned int vmin,unsigned int rev)
|
|
|
|
throw()
|
|
|
|
{
|
|
|
|
return ( ((uint64_t)(vmaj & 0xffff) << 32) | ((uint64_t)(vmin & 0xffff) << 16) | (uint64_t)(rev & 0xffff) );
|
|
|
|
}
|
|
|
|
|
2014-01-06 00:24:12 +00:00
|
|
|
/**
|
|
|
|
* Parse NFO data from .nfo file on software update site
|
|
|
|
*
|
|
|
|
* The first argument is the NFO data, and all the remaining arguments are
|
|
|
|
* result parameters to be filled with results. If an error is returned the
|
|
|
|
* results in the parameters should be considered undefined.
|
|
|
|
*
|
|
|
|
* @param nfo NFO data
|
|
|
|
* @param vMajor Result: major version
|
|
|
|
* @param vMinor Result: minor version
|
|
|
|
* @param vRevision Result: revision number
|
|
|
|
* @param signedBy Result: signing identity
|
|
|
|
* @param signature Result: Ed25519 signature data
|
|
|
|
* @param url Result: URL of update binary
|
|
|
|
* @return NULL on success or error message on failure
|
|
|
|
*/
|
|
|
|
static const char *parseNfo(
|
|
|
|
const char *nfoText,
|
|
|
|
unsigned int &vMajor,
|
|
|
|
unsigned int &vMinor,
|
|
|
|
unsigned int &vRevision,
|
|
|
|
Address &signedBy,
|
|
|
|
std::string &signature,
|
|
|
|
std::string &url);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Validate an update once downloaded
|
|
|
|
*
|
|
|
|
* This obtains the identity corresponding to the address from the compiled-in
|
|
|
|
* list of valid signing identities.
|
|
|
|
*
|
|
|
|
* @param data Update data
|
|
|
|
* @param len Length of update data
|
|
|
|
* @param signedBy Signing authority address
|
|
|
|
* @param signature Signing authority signature
|
|
|
|
* @return True on validation success, false if rejected
|
|
|
|
*/
|
|
|
|
static bool validateUpdate(
|
|
|
|
const void *data,
|
|
|
|
unsigned int len,
|
|
|
|
const Address &signedBy,
|
|
|
|
const std::string &signature);
|
|
|
|
|
2013-12-10 23:30:53 +00:00
|
|
|
private:
|
2014-08-16 16:08:52 +00:00
|
|
|
static void _cbHandleGetLatestVersionInfo(void *arg,int code,const std::string &url,const std::string &body);
|
|
|
|
static void _cbHandleGetLatestVersionBinary(void *arg,int code,const std::string &url,const std::string &body);
|
2013-12-10 23:30:53 +00:00
|
|
|
|
2014-09-24 20:53:03 +00:00
|
|
|
const RuntimeEnvironment *RR;
|
2013-12-10 23:30:53 +00:00
|
|
|
const uint64_t _myVersion;
|
|
|
|
volatile uint64_t _lastUpdateAttempt;
|
|
|
|
volatile enum {
|
|
|
|
UPDATE_STATUS_IDLE,
|
|
|
|
UPDATE_STATUS_GETTING_NFO,
|
|
|
|
UPDATE_STATUS_GETTING_FILE
|
|
|
|
} _status;
|
|
|
|
volatile bool _die;
|
|
|
|
Address _signedBy;
|
|
|
|
std::string _signature;
|
|
|
|
Mutex _lock;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace ZeroTier
|
|
|
|
|
|
|
|
#endif
|