2017-05-10 19:29:35 +02:00
package scraperService
import (
"encoding/binary"
"encoding/hex"
"net"
"time"
2017-05-17 07:58:40 +02:00
"github.com/NyaaPantsu/nyaa/config"
"github.com/NyaaPantsu/nyaa/db"
"github.com/NyaaPantsu/nyaa/model"
"github.com/NyaaPantsu/nyaa/util/log"
2017-05-10 19:29:35 +02:00
)
2017-05-11 13:40:50 +02:00
// TransactionTimeout 30 second timeout for transactions
const TransactionTimeout = time . Second * 30
2017-05-10 19:29:35 +02:00
const stateSendID = 0
const stateRecvID = 1
const stateTransact = 2
const actionError = 3
const actionScrape = 2
const actionAnnounce = 1
const actionConnect = 0
// Transaction a scrape transaction on a udp tracker
type Transaction struct {
TransactionID uint32
ConnectionID uint64
bucket * Bucket
state uint8
swarms [ ] model . Torrent
2017-05-11 13:40:50 +02:00
lastData time . Time
2017-05-10 19:29:35 +02:00
}
// Done marks this transaction as done and removes it from parent
func ( t * Transaction ) Done ( ) {
2017-05-11 13:40:50 +02:00
t . bucket . Forget ( t . TransactionID )
2017-05-10 19:29:35 +02:00
}
func ( t * Transaction ) handleScrapeReply ( data [ ] byte ) {
data = data [ 8 : ]
2017-05-11 00:06:21 +02:00
now := time . Now ( )
2017-05-10 19:29:35 +02:00
for idx := range t . swarms {
2017-05-11 00:06:21 +02:00
t . swarms [ idx ] . Seeders = binary . BigEndian . Uint32 ( data )
data = data [ 4 : ]
t . swarms [ idx ] . Completed = binary . BigEndian . Uint32 ( data )
data = data [ 4 : ]
t . swarms [ idx ] . Leechers = binary . BigEndian . Uint32 ( data )
data = data [ 4 : ]
2017-05-10 19:29:35 +02:00
t . swarms [ idx ] . LastScrape = now
2017-05-11 00:06:21 +02:00
idx ++
2017-05-10 19:29:35 +02:00
}
}
2017-05-31 04:21:57 +02:00
var pgQuery = "UPDATE " + config . Conf . Models . TorrentsTableName + " SET seeders = $1 , leechers = $2 , completed = $3 , last_scrape = $4 WHERE torrent_id = $5"
var sqliteQuery = "UPDATE " + config . Conf . Models . TorrentsTableName + " SET seeders = ? , leechers = ? , completed = ? , last_scrape = ? WHERE torrent_id = ?"
2017-05-11 21:06:47 +02:00
2017-05-10 19:29:35 +02:00
// Sync syncs models with database
func ( t * Transaction ) Sync ( ) ( err error ) {
2017-05-11 21:06:47 +02:00
q := pgQuery
if db . IsSqlite {
q = sqliteQuery
}
tx , e := db . ORM . DB ( ) . Begin ( )
err = e
if err == nil {
for idx := range t . swarms {
_ , err = tx . Exec ( q , t . swarms [ idx ] . Seeders , t . swarms [ idx ] . Leechers , t . swarms [ idx ] . Completed , t . swarms [ idx ] . LastScrape , t . swarms [ idx ] . ID )
2017-05-11 00:06:21 +02:00
}
2017-05-11 21:06:47 +02:00
tx . Commit ( )
2017-05-11 00:06:21 +02:00
}
2017-05-10 19:29:35 +02:00
return
}
// create send event
func ( t * Transaction ) SendEvent ( to net . Addr ) ( ev * SendEvent ) {
ev = & SendEvent {
To : to ,
}
if t . state == stateRecvID {
l := len ( t . swarms ) * 20
l += 16
ev . Data = make ( [ ] byte , l )
binary . BigEndian . PutUint64 ( ev . Data [ : ] , t . ConnectionID )
binary . BigEndian . PutUint32 ( ev . Data [ 8 : ] , 2 )
binary . BigEndian . PutUint32 ( ev . Data [ 12 : ] , t . TransactionID )
for idx := range t . swarms {
ih , err := hex . DecodeString ( t . swarms [ idx ] . Hash )
if err == nil && len ( ih ) == 20 {
copy ( ev . Data [ 16 + ( idx * 20 ) : ] , ih )
}
}
2017-05-11 00:06:21 +02:00
t . state = stateTransact
2017-05-10 19:29:35 +02:00
} else if t . state == stateSendID {
ev . Data = make ( [ ] byte , 16 )
binary . BigEndian . PutUint64 ( ev . Data , InitialConnectionID )
binary . BigEndian . PutUint32 ( ev . Data [ 8 : ] , 0 )
binary . BigEndian . PutUint32 ( ev . Data [ 12 : ] , t . TransactionID )
t . state = stateRecvID
}
2017-05-11 13:40:50 +02:00
t . lastData = time . Now ( )
2017-05-10 19:29:35 +02:00
return
}
func ( t * Transaction ) handleError ( msg string ) {
log . Infof ( "scrape failed: %s" , msg )
}
// handle data for transaction
func ( t * Transaction ) GotData ( data [ ] byte ) ( done bool ) {
2017-05-11 13:40:50 +02:00
t . lastData = time . Now ( )
2017-05-10 19:29:35 +02:00
if len ( data ) > 4 {
cmd := binary . BigEndian . Uint32 ( data )
switch cmd {
case actionConnect :
if len ( data ) == 16 {
if t . state == stateRecvID {
t . ConnectionID = binary . BigEndian . Uint64 ( data [ 8 : ] )
}
}
break
case actionScrape :
if len ( data ) == ( 12 * len ( t . swarms ) ) + 8 && t . state == stateTransact {
2017-05-11 00:06:21 +02:00
t . handleScrapeReply ( data )
2017-05-10 19:29:35 +02:00
}
done = true
break
case actionError :
if len ( data ) == 12 {
t . handleError ( string ( data [ 4 : 12 ] ) )
}
default :
done = true
}
}
return
}
2017-05-11 13:40:50 +02:00
func ( t * Transaction ) IsTimedOut ( ) bool {
return t . lastData . Add ( TransactionTimeout ) . Before ( time . Now ( ) )
}