mirror of
https://github.com/Dispatcharr/Dispatcharr.git
synced 2026-01-23 10:45:27 +00:00
155 lines
5.9 KiB
Python
155 lines
5.9 KiB
Python
"""Shared configuration between proxy types"""
|
|
import time
|
|
from django.db import connection
|
|
|
|
class BaseConfig:
|
|
DEFAULT_USER_AGENT = 'VLC/3.0.20 LibVLC/3.0.20' # Will only be used if connection to settings fail
|
|
CHUNK_SIZE = 8192
|
|
CLIENT_POLL_INTERVAL = 0.1
|
|
MAX_RETRIES = 3
|
|
RETRY_WAIT_INTERVAL = 0.5 # seconds to wait between retries
|
|
CONNECTION_TIMEOUT = 10 # seconds to wait for initial connection
|
|
MAX_STREAM_SWITCHES = 10 # Maximum number of stream switch attempts before giving up
|
|
BUFFER_CHUNK_SIZE = 188 * 1361 # ~256KB
|
|
BUFFERING_TIMEOUT = 15 # Seconds to wait for buffering before switching streams
|
|
BUFFER_SPEED = 1 # What speed to condsider the stream buffering, 1x is normal speed, 2x is double speed, etc.
|
|
|
|
# Cache for proxy settings (class-level, shared across all instances)
|
|
_proxy_settings_cache = None
|
|
_proxy_settings_cache_time = 0
|
|
_proxy_settings_cache_ttl = 10 # Cache for 10 seconds
|
|
|
|
@classmethod
|
|
def get_proxy_settings(cls):
|
|
"""Get proxy settings from CoreSettings JSON data with fallback to defaults (cached)"""
|
|
# Check if cache is still valid
|
|
now = time.time()
|
|
if cls._proxy_settings_cache is not None and (now - cls._proxy_settings_cache_time) < cls._proxy_settings_cache_ttl:
|
|
return cls._proxy_settings_cache
|
|
|
|
# Cache miss or expired - fetch from database
|
|
try:
|
|
from core.models import CoreSettings
|
|
settings = CoreSettings.get_proxy_settings()
|
|
cls._proxy_settings_cache = settings
|
|
cls._proxy_settings_cache_time = now
|
|
return settings
|
|
|
|
except Exception:
|
|
# Return defaults if database query fails
|
|
return {
|
|
"buffering_timeout": 15,
|
|
"buffering_speed": 1.0,
|
|
"redis_chunk_ttl": 60,
|
|
"channel_shutdown_delay": 0,
|
|
"channel_init_grace_period": 5,
|
|
}
|
|
|
|
finally:
|
|
# Always close the connection after reading settings
|
|
try:
|
|
connection.close()
|
|
except Exception:
|
|
pass
|
|
|
|
@classmethod
|
|
def get_redis_chunk_ttl(cls):
|
|
"""Get Redis chunk TTL from database or default"""
|
|
settings = cls.get_proxy_settings()
|
|
return settings.get("redis_chunk_ttl", 60)
|
|
|
|
@property
|
|
def REDIS_CHUNK_TTL(self):
|
|
return self.get_redis_chunk_ttl()
|
|
|
|
class HLSConfig(BaseConfig):
|
|
MIN_SEGMENTS = 12
|
|
MAX_SEGMENTS = 16
|
|
WINDOW_SIZE = 12
|
|
INITIAL_SEGMENTS = 3
|
|
INITIAL_CONNECTION_WINDOW = 10
|
|
CLIENT_TIMEOUT_FACTOR = 1.5
|
|
CLIENT_CLEANUP_INTERVAL = 10
|
|
FIRST_SEGMENT_TIMEOUT = 5.0
|
|
INITIAL_BUFFER_SECONDS = 25.0
|
|
MAX_INITIAL_SEGMENTS = 10
|
|
BUFFER_READY_TIMEOUT = 30.0
|
|
|
|
class TSConfig(BaseConfig):
|
|
"""Configuration settings for TS proxy"""
|
|
|
|
# Buffer settings
|
|
INITIAL_BEHIND_CHUNKS = 4 # How many chunks behind to start a client (4 chunks = ~1MB)
|
|
CHUNK_BATCH_SIZE = 5 # How many chunks to fetch in one batch
|
|
KEEPALIVE_INTERVAL = 0.5 # Seconds between keepalive packets when at buffer head
|
|
# Chunk read timeout
|
|
CHUNK_TIMEOUT = 5 # Seconds to wait for each chunk read
|
|
|
|
# Streaming settings
|
|
TARGET_BITRATE = 8000000 # Target bitrate (8 Mbps)
|
|
STREAM_TIMEOUT = 20 # Disconnect after this many seconds of no data
|
|
HEALTH_CHECK_INTERVAL = 5 # Check stream health every N seconds
|
|
|
|
# Resource management
|
|
CLEANUP_INTERVAL = 60 # Check for inactive channels every 60 seconds
|
|
|
|
# Client tracking settings
|
|
CLIENT_RECORD_TTL = 60 # How long client records persist in Redis (seconds). Client will be considered MIA after this time.
|
|
CLEANUP_CHECK_INTERVAL = 1 # How often to check for disconnected clients (seconds)
|
|
CLIENT_HEARTBEAT_INTERVAL = 5 # How often to send client heartbeats (seconds)
|
|
GHOST_CLIENT_MULTIPLIER = 6.0 # How many heartbeat intervals before client considered ghost (6 would mean 36 seconds if heartbeat interval is 6)
|
|
CLIENT_WAIT_TIMEOUT = 30 # Seconds to wait for client to connect
|
|
|
|
# Stream health and recovery settings
|
|
MAX_HEALTH_RECOVERY_ATTEMPTS = 2 # Maximum times to attempt recovery for a single stream
|
|
MAX_RECONNECT_ATTEMPTS = 3 # Maximum reconnects to try before switching streams
|
|
MIN_STABLE_TIME_BEFORE_RECONNECT = 30 # Minimum seconds a stream must be stable to try reconnect
|
|
FAILOVER_GRACE_PERIOD = 20 # Extra time (seconds) to allow for stream switching before disconnecting clients
|
|
URL_SWITCH_TIMEOUT = 20 # Max time allowed for a stream switch operation
|
|
|
|
|
|
|
|
# Database-dependent settings with fallbacks
|
|
@classmethod
|
|
def get_channel_shutdown_delay(cls):
|
|
"""Get channel shutdown delay from database or default"""
|
|
settings = cls.get_proxy_settings()
|
|
return settings.get("channel_shutdown_delay", 0)
|
|
|
|
@classmethod
|
|
def get_buffering_timeout(cls):
|
|
"""Get buffering timeout from database or default"""
|
|
settings = cls.get_proxy_settings()
|
|
return settings.get("buffering_timeout", 15)
|
|
|
|
@classmethod
|
|
def get_buffering_speed(cls):
|
|
"""Get buffering speed threshold from database or default"""
|
|
settings = cls.get_proxy_settings()
|
|
return settings.get("buffering_speed", 1.0)
|
|
|
|
@classmethod
|
|
def get_channel_init_grace_period(cls):
|
|
"""Get channel init grace period from database or default"""
|
|
settings = cls.get_proxy_settings()
|
|
return settings.get("channel_init_grace_period", 5)
|
|
|
|
# Dynamic property access for these settings
|
|
@property
|
|
def CHANNEL_SHUTDOWN_DELAY(self):
|
|
return self.get_channel_shutdown_delay()
|
|
|
|
@property
|
|
def BUFFERING_TIMEOUT(self):
|
|
return self.get_buffering_timeout()
|
|
|
|
@property
|
|
def BUFFERING_SPEED(self):
|
|
return self.get_buffering_speed()
|
|
|
|
@property
|
|
def CHANNEL_INIT_GRACE_PERIOD(self):
|
|
return self.get_channel_init_grace_period()
|
|
|
|
|
|
|