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