Initial commit: docker compose config
Release Docker multi arch / docker (push) Has been cancelled
Test Install Script / Test Script Syntax (push) Has been cancelled
Test Install Script / Test on almalinux-10 (default) (push) Has been cancelled
Test Install Script / Test on almalinux-10 (root) (push) Has been cancelled
Test Install Script / Test on almalinux-8 (default) (push) Has been cancelled
Test Install Script / Test on almalinux-8 (root) (push) Has been cancelled
Test Install Script / Test on almalinux-9 (default) (push) Has been cancelled
Test Install Script / Test on almalinux-9 (root) (push) Has been cancelled
Test Install Script / Test on amazonlinux-2 (default) (push) Has been cancelled
Test Install Script / Test on amazonlinux-2 (root) (push) Has been cancelled
Test Install Script / Test on debian-11 (default) (push) Has been cancelled
Test Install Script / Test on debian-11 (root) (push) Has been cancelled
Test Install Script / Test on debian-12 (default) (push) Has been cancelled
Test Install Script / Test on debian-12 (root) (push) Has been cancelled
Test Install Script / Test on debian-13 (default) (push) Has been cancelled
Test Install Script / Test on debian-13 (root) (push) Has been cancelled
Test Install Script / Test on fedora-latest (default) (push) Has been cancelled
Test Install Script / Test on fedora-latest (root) (push) Has been cancelled
Test Install Script / Test on rocky-10 (default) (push) Has been cancelled
Test Install Script / Test on rocky-10 (root) (push) Has been cancelled
Test Install Script / Test on rocky-8 (default) (push) Has been cancelled
Test Install Script / Test on rocky-8 (root) (push) Has been cancelled
Test Install Script / Test on rocky-9 (default) (push) Has been cancelled
Test Install Script / Test on rocky-9 (root) (push) Has been cancelled
Test Install Script / Test on ubuntu-22.04 (default) (push) Has been cancelled
Test Install Script / Test on ubuntu-22.04 (root) (push) Has been cancelled
Test Install Script / Test on ubuntu-24.04 (default) (push) Has been cancelled
Test Install Script / Test on ubuntu-24.04 (root) (push) Has been cancelled

This commit is contained in:
2026-05-30 12:07:11 +00:00
commit 616c6b1c62
381 changed files with 55145 additions and 0 deletions
+170
View File
@@ -0,0 +1,170 @@
package torr
import (
// "context"
"encoding/hex"
"errors"
"fmt"
"log"
"net"
"net/http"
"sync/atomic"
"time"
"github.com/anacrolix/dms/dlna"
"github.com/anacrolix/missinggo/v2/httptoo"
"github.com/anacrolix/torrent"
mt "server/mimetype"
sets "server/settings"
"server/torr/state"
)
// Add atomic counter for concurrent streams
var activeStreams int32
// type contextResponseWriter struct {
// http.ResponseWriter
// ctx context.Context
// }
// func (w *contextResponseWriter) Write(p []byte) (n int, err error) {
// // Check context before each write
// select {
// case <-w.ctx.Done():
// return 0, w.ctx.Err()
// default:
// return w.ResponseWriter.Write(p)
// }
// }
func (t *Torrent) Stream(fileID int, req *http.Request, resp http.ResponseWriter) error {
// Increment active streams counter
streamID := atomic.AddInt32(&activeStreams, 1)
defer atomic.AddInt32(&activeStreams, -1)
// Stream disconnect timeout (same as torrent)
streamTimeout := sets.BTsets.TorrentDisconnectTimeout
if !t.GotInfo() {
http.NotFound(resp, req)
return errors.New("torrent doesn't have info yet")
}
// Get file information
st := t.Status()
var stFile *state.TorrentFileStat
for _, fileStat := range st.FileStats {
if fileStat.Id == fileID {
stFile = fileStat
break
}
}
if stFile == nil {
return fmt.Errorf("file with id %v not found", fileID)
}
// Find the actual torrent file
files := t.Files()
var file *torrent.File
for _, tfile := range files {
if tfile.Path() == stFile.Path {
file = tfile
break
}
}
if file == nil {
return fmt.Errorf("file with id %v not found", fileID)
}
// Check file size limit
if int64(sets.MaxSize) > 0 && file.Length() > int64(sets.MaxSize) {
err := fmt.Errorf("file size exceeded max allowed %d bytes", sets.MaxSize)
log.Printf("File %s size (%d) exceeded max allowed %d bytes", file.DisplayPath(), file.Length(), sets.MaxSize)
http.Error(resp, err.Error(), http.StatusForbidden)
return err
}
// Create reader with context for timeout
reader := t.NewReader(file)
if reader == nil {
return errors.New("cannot create torrent reader")
}
// Ensure reader is always closed
defer t.CloseReader(reader)
if sets.BTsets.ResponsiveMode {
reader.SetResponsive()
}
// Log connection
host, port, clerr := net.SplitHostPort(req.RemoteAddr)
if sets.BTsets.EnableDebug {
if clerr != nil {
log.Printf("[Stream:%d] Connect client (Active streams: %d)", streamID, atomic.LoadInt32(&activeStreams))
} else {
log.Printf("[Stream:%d] Connect client %s:%s (Active streams: %d)",
streamID, host, port, atomic.LoadInt32(&activeStreams))
}
}
// Mark as viewed
sets.SetViewed(&sets.Viewed{
Hash: t.Hash().HexString(),
FileIndex: fileID,
})
// Set response headers
resp.Header().Set("Connection", "close")
// Add timeout header if configured
if streamTimeout > 0 {
resp.Header().Set("X-Stream-Timeout", fmt.Sprintf("%d", streamTimeout))
}
// Add ETag
etag := hex.EncodeToString([]byte(fmt.Sprintf("%s/%s", t.Hash().HexString(), file.Path())))
resp.Header().Set("ETag", httptoo.EncodeQuotedString(etag))
// DLNA headers
resp.Header().Set("transferMode.dlna.org", "Streaming")
// add MimeType
mime, err := mt.MimeTypeByPath(file.Path())
if err == nil && mime.IsMedia() {
resp.Header().Set("content-type", mime.String())
}
// DLNA Seek
if req.Header.Get("getContentFeatures.dlna.org") != "" {
resp.Header().Set("contentFeatures.dlna.org", dlna.ContentFeatures{
SupportRange: true,
SupportTimeSeek: true,
}.String())
}
// Add support for range requests
if req.Header.Get("Range") != "" {
resp.Header().Set("Accept-Ranges", "bytes")
}
// // Create a context with timeout if configured
// ctx := req.Context()
// if streamTimeout > 0 {
// var cancel context.CancelFunc
// ctx, cancel = context.WithTimeout(ctx, time.Duration(streamTimeout)*time.Second)
// defer cancel()
// }
// // Update request with new context
// req = req.WithContext(ctx)
// // Handle client disconnections better
// wrappedResp := &contextResponseWriter{
// ResponseWriter: resp,
// ctx: ctx,
// }
// http.ServeContent(wrappedResp, req, file.Path(), time.Unix(t.Timestamp, 0), reader)
http.ServeContent(resp, req, file.Path(), time.Unix(t.Timestamp, 0), reader)
if sets.BTsets.EnableDebug {
if clerr != nil {
log.Printf("[Stream:%d] Disconnect client", streamID)
} else {
log.Printf("[Stream:%d] Disconnect client %s:%s", streamID, host, port)
}
}
return nil
}
// GetActiveStreams returns number of currently active streams
func GetActiveStreams() int32 {
return atomic.LoadInt32(&activeStreams)
}