|
Packit |
8f70b4 |
/*
|
|
Packit |
8f70b4 |
* lftp - file transfer program
|
|
Packit |
8f70b4 |
*
|
|
Packit |
8f70b4 |
* Copyright (c) 1996-2014 by Alexander V. Lukyanov (lav@yars.free.net)
|
|
Packit |
8f70b4 |
*
|
|
Packit |
8f70b4 |
* This program is free software; you can redistribute it and/or modify
|
|
Packit |
8f70b4 |
* it under the terms of the GNU General Public License as published by
|
|
Packit |
8f70b4 |
* the Free Software Foundation; either version 3 of the License, or
|
|
Packit |
8f70b4 |
* (at your option) any later version.
|
|
Packit |
8f70b4 |
*
|
|
Packit |
8f70b4 |
* This program is distributed in the hope that it will be useful,
|
|
Packit |
8f70b4 |
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
Packit |
8f70b4 |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
Packit |
8f70b4 |
* GNU General Public License for more details.
|
|
Packit |
8f70b4 |
*
|
|
Packit |
8f70b4 |
* You should have received a copy of the GNU General Public License
|
|
Packit |
8f70b4 |
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
Packit |
8f70b4 |
*/
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
#ifndef TORRENTTRACKER_H
|
|
Packit |
8f70b4 |
#define TORRENTTRACKER_H
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
#include "url.h"
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
class TrackerBackend;
|
|
Packit |
8f70b4 |
class TorrentTracker : public SMTask, protected ProtoLog
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
friend class Torrent;
|
|
Packit |
8f70b4 |
friend class TrackerBackend;
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
Torrent *parent;
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
xarray_p<xstring> tracker_urls;
|
|
Packit |
8f70b4 |
int current_tracker;
|
|
Packit |
8f70b4 |
SMTaskRef<TrackerBackend> backend;
|
|
Packit |
8f70b4 |
Timer tracker_timer;
|
|
Packit |
8f70b4 |
Timer tracker_timeout_timer;
|
|
Packit |
8f70b4 |
xstring tracker_id;
|
|
Packit |
8f70b4 |
bool started;
|
|
Packit |
8f70b4 |
Ref<Error> error;
|
|
Packit |
8f70b4 |
int tracker_no;
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
TorrentTracker(Torrent *p,const char *url);
|
|
Packit |
8f70b4 |
void AddURL(const char *url);
|
|
Packit |
8f70b4 |
int Do();
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
void Start();
|
|
Packit |
8f70b4 |
void Shutdown();
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
void SendTrackerRequest(const char *event);
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
void SetError(const char *e);
|
|
Packit |
8f70b4 |
bool Failed() const { return error!=0 || tracker_urls.count()==0; }
|
|
Packit |
8f70b4 |
const char *ErrorText() const { return error->Text(); }
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
void NextTracker();
|
|
Packit |
8f70b4 |
void CreateTrackerBackend();
|
|
Packit |
8f70b4 |
const char *GetLogContext() { return GetURL(); }
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
public:
|
|
Packit |
8f70b4 |
~TorrentTracker() {}
|
|
Packit |
8f70b4 |
const char *NextRequestIn() const {
|
|
Packit |
8f70b4 |
return tracker_timer.TimeLeft().toString(
|
|
Packit |
8f70b4 |
TimeInterval::TO_STR_TRANSLATE|TimeInterval::TO_STR_TERSE);
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
const char *GetURL() const {
|
|
Packit |
8f70b4 |
return tracker_urls[current_tracker]->get();
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
const char *Status() const;
|
|
Packit |
8f70b4 |
bool IsActive() const;
|
|
Packit |
8f70b4 |
void TrackerRequestFinished() { tracker_timer.Reset(); }
|
|
Packit |
8f70b4 |
void SetInterval(unsigned i) {
|
|
Packit |
8f70b4 |
if(i<30)
|
|
Packit |
8f70b4 |
i=30;
|
|
Packit |
8f70b4 |
tracker_timer.Set(i);
|
|
Packit |
8f70b4 |
LogNote(4,"Tracker interval is %u",i);
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
void SetTrackerID(const xstring& id) {
|
|
Packit |
8f70b4 |
if(id)
|
|
Packit |
8f70b4 |
tracker_id.set(id);
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
bool AddPeerCompact(const char *a,int len) const;
|
|
Packit |
8f70b4 |
bool AddPeer(const xstring& addr,int port) const;
|
|
Packit |
8f70b4 |
bool ShuttingDown();
|
|
Packit |
8f70b4 |
};
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
class TrackerBackend : public SMTask, protected ProtoLog
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
protected:
|
|
Packit |
8f70b4 |
TorrentTracker *master;
|
|
Packit |
8f70b4 |
void SetError(const char *e) { master->SetError(e); }
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
const char *GetURL() const { return master->GetURL(); }
|
|
Packit |
8f70b4 |
const xstring& GetInfoHash() const;
|
|
Packit |
8f70b4 |
const xstring& GetMyPeerId() const;
|
|
Packit |
8f70b4 |
int GetPort() const;
|
|
Packit |
8f70b4 |
unsigned long long GetTotalSent() const;
|
|
Packit |
8f70b4 |
unsigned long long GetTotalRecv() const;
|
|
Packit |
8f70b4 |
unsigned long long GetTotalLeft() const;
|
|
Packit |
8f70b4 |
bool HasMetadata() const;
|
|
Packit |
8f70b4 |
bool Complete() const;
|
|
Packit |
8f70b4 |
int GetWantedPeersCount() const;
|
|
Packit |
8f70b4 |
const xstring& GetMyKey() const;
|
|
Packit |
8f70b4 |
unsigned GetMyKeyNum() const;
|
|
Packit |
8f70b4 |
const char *GetTrackerId() const;
|
|
Packit |
8f70b4 |
void SetTrackerID(const xstring& id) const { master->SetTrackerID(id); }
|
|
Packit |
8f70b4 |
void SetInterval(unsigned i) const { master->SetInterval(i); }
|
|
Packit |
8f70b4 |
bool AddPeerCompact(const char *a,int len) const { return master->AddPeerCompact(a,len); }
|
|
Packit |
8f70b4 |
bool AddPeer(const xstring& addr,int port) const { return master->AddPeer(addr,port); }
|
|
Packit |
8f70b4 |
void NextTracker() const { master->NextTracker(); }
|
|
Packit |
8f70b4 |
bool ShuttingDown() const;
|
|
Packit |
8f70b4 |
void Started() const;
|
|
Packit |
8f70b4 |
void TrackerRequestFinished() const;
|
|
Packit |
8f70b4 |
const char *GetLogContext() { return master->GetLogContext(); }
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
public:
|
|
Packit |
8f70b4 |
TrackerBackend(TorrentTracker *m) : master(m) {}
|
|
Packit |
8f70b4 |
virtual ~TrackerBackend() {}
|
|
Packit |
8f70b4 |
virtual bool IsActive() const = 0;
|
|
Packit |
8f70b4 |
virtual void SendTrackerRequest(const char *event) = 0;
|
|
Packit |
8f70b4 |
virtual const char *Status() const = 0;
|
|
Packit |
8f70b4 |
};
|
|
Packit |
8f70b4 |
class HttpTracker : public TrackerBackend
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
FileAccessRef t_session;
|
|
Packit |
8f70b4 |
SMTaskRef<IOBuffer> tracker_reply;
|
|
Packit |
8f70b4 |
int HandleTrackerReply();
|
|
Packit |
8f70b4 |
public:
|
|
Packit |
8f70b4 |
bool IsActive() const { return tracker_reply!=0; }
|
|
Packit |
8f70b4 |
void SendTrackerRequest(const char *event);
|
|
Packit |
8f70b4 |
HttpTracker(TorrentTracker *m,ParsedURL *u)
|
|
Packit |
8f70b4 |
: TrackerBackend(m), t_session(FileAccess::New(u)) {}
|
|
Packit |
8f70b4 |
~HttpTracker() {}
|
|
Packit |
8f70b4 |
int Do();
|
|
Packit |
8f70b4 |
const char *Status() const { return t_session->CurrentStatus(); }
|
|
Packit |
8f70b4 |
};
|
|
Packit |
8f70b4 |
class UdpTracker : public TrackerBackend, protected Networker
|
|
Packit |
8f70b4 |
{
|
|
Packit |
8f70b4 |
xstring_c hostname;
|
|
Packit |
8f70b4 |
xstring_c portname;
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
SMTaskRef<Resolver> resolver;
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
xarray<sockaddr_u> peer;
|
|
Packit |
8f70b4 |
int peer_curr;
|
|
Packit |
8f70b4 |
void NextPeer();
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
int sock; // udp socket for packet exchange
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
Timer timeout_timer;
|
|
Packit |
8f70b4 |
int try_number; // timeout = 60 * 2^try_number
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
bool has_connection_id;
|
|
Packit |
8f70b4 |
unsigned long long connection_id;
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
enum action_t {
|
|
Packit |
8f70b4 |
a_none=-1,
|
|
Packit |
8f70b4 |
a_connect=0,
|
|
Packit |
8f70b4 |
a_announce=1,
|
|
Packit |
8f70b4 |
a_scrape=2,
|
|
Packit |
8f70b4 |
a_error=3,
|
|
Packit |
8f70b4 |
a_announce6=4,
|
|
Packit |
8f70b4 |
};
|
|
Packit |
8f70b4 |
enum event_t {
|
|
Packit |
8f70b4 |
ev_idle=-1,
|
|
Packit |
8f70b4 |
ev_none=0,
|
|
Packit |
8f70b4 |
ev_completed=1,
|
|
Packit |
8f70b4 |
ev_started=2,
|
|
Packit |
8f70b4 |
ev_stopped=3,
|
|
Packit |
8f70b4 |
};
|
|
Packit |
8f70b4 |
enum magic_t {
|
|
Packit |
8f70b4 |
connect_magic=0x41727101980ULL,
|
|
Packit |
8f70b4 |
};
|
|
Packit |
8f70b4 |
static const char *EventToString(event_t e);
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
unsigned transaction_id;
|
|
Packit |
8f70b4 |
action_t current_action;
|
|
Packit |
8f70b4 |
event_t current_event;
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
bool SendPacket(Buffer& req);
|
|
Packit |
8f70b4 |
bool SendConnectRequest();
|
|
Packit |
8f70b4 |
bool SendEventRequest();
|
|
Packit |
8f70b4 |
bool RecvReply();
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
unsigned NewTransactionId() { return transaction_id=random(); }
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
public:
|
|
Packit |
8f70b4 |
UdpTracker(TorrentTracker *m,ParsedURL *u)
|
|
Packit |
8f70b4 |
: TrackerBackend(m),
|
|
Packit |
8f70b4 |
hostname(u->host.get()), portname(u->port.get()),
|
|
Packit |
8f70b4 |
peer_curr(0), sock(-1), timeout_timer(60), try_number(0),
|
|
Packit |
8f70b4 |
has_connection_id(false), connection_id(0),
|
|
Packit |
8f70b4 |
current_action(a_none), current_event(ev_idle) {}
|
|
Packit |
8f70b4 |
~UdpTracker() {
|
|
Packit |
8f70b4 |
if(sock!=-1)
|
|
Packit |
8f70b4 |
close(sock);
|
|
Packit |
8f70b4 |
}
|
|
Packit |
8f70b4 |
int Do();
|
|
Packit |
8f70b4 |
bool IsActive() const { return current_event!=ev_idle; }
|
|
Packit |
8f70b4 |
void SendTrackerRequest(const char *event);
|
|
Packit |
8f70b4 |
const char *Status() const;
|
|
Packit |
8f70b4 |
};
|
|
Packit |
8f70b4 |
|
|
Packit |
8f70b4 |
#endif // TORRENTTRACKER_H
|