Add conection and remove proxy usage
This commit is contained in:
79
database.py
79
database.py
@@ -45,12 +45,17 @@ def get_latest_proxy() -> Optional[Dict]:
|
|||||||
conn.close()
|
conn.close()
|
||||||
|
|
||||||
return dict(proxy) if proxy else None
|
return dict(proxy) if proxy else None
|
||||||
except Exception as e:
|
except Exception:
|
||||||
print(f"Erro ao buscar proxy: {e}")
|
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def get_all_active_proxies() -> list:
|
def get_all_active_proxies() -> list:
|
||||||
"""Retorna todos os proxies ativos do banco"""
|
"""
|
||||||
|
Retorna todos os proxies ativos do banco, priorizando:
|
||||||
|
1. Proxies com sucesso recente
|
||||||
|
2. Proxies com baixo failure_count
|
||||||
|
3. Proxies com boa taxa de sucesso
|
||||||
|
4. Proxies novos ainda não testados
|
||||||
|
"""
|
||||||
try:
|
try:
|
||||||
conn = get_db_connection()
|
conn = get_db_connection()
|
||||||
cursor = conn.cursor()
|
cursor = conn.cursor()
|
||||||
@@ -61,16 +66,23 @@ def get_all_active_proxies() -> list:
|
|||||||
country_code, country_name, city, is_active, is_anonymous,
|
country_code, country_name, city, is_active, is_anonymous,
|
||||||
response_time_ms, last_checked_at, last_successful_at,
|
response_time_ms, last_checked_at, last_successful_at,
|
||||||
failure_count, success_count, usage, source, notes,
|
failure_count, success_count, usage, source, notes,
|
||||||
created_at, updated_at
|
created_at, updated_at,
|
||||||
FROM proxies
|
|
||||||
WHERE is_active = TRUE
|
|
||||||
ORDER BY
|
|
||||||
last_successful_at DESC NULLS LAST,
|
|
||||||
response_time_ms ASC NULLS LAST,
|
|
||||||
(CASE WHEN success_count + failure_count > 0
|
(CASE WHEN success_count + failure_count > 0
|
||||||
THEN CAST(success_count AS FLOAT) / (success_count + failure_count)
|
THEN CAST(success_count AS FLOAT) / (success_count + failure_count)
|
||||||
ELSE 0 END) DESC,
|
ELSE 0.5 END) as success_rate
|
||||||
|
FROM proxies
|
||||||
|
WHERE is_active = TRUE
|
||||||
|
AND failure_count < 8 -- Ignora proxies com muitas falhas
|
||||||
|
ORDER BY
|
||||||
|
-- Prioriza proxies com sucesso recente
|
||||||
|
last_successful_at DESC NULLS LAST,
|
||||||
|
-- Depois por taxa de sucesso
|
||||||
|
success_rate DESC,
|
||||||
|
-- Depois por menos falhas
|
||||||
|
failure_count ASC,
|
||||||
|
-- Por último, proxies novos
|
||||||
created_at DESC
|
created_at DESC
|
||||||
|
LIMIT 50 -- Limita a 50 melhores proxies para não perder tempo
|
||||||
""")
|
""")
|
||||||
|
|
||||||
proxies = cursor.fetchall()
|
proxies = cursor.fetchall()
|
||||||
@@ -78,8 +90,7 @@ def get_all_active_proxies() -> list:
|
|||||||
conn.close()
|
conn.close()
|
||||||
|
|
||||||
return [dict(proxy) for proxy in proxies] if proxies else []
|
return [dict(proxy) for proxy in proxies] if proxies else []
|
||||||
except Exception as e:
|
except Exception:
|
||||||
print(f"Erro ao buscar proxies: {e}")
|
|
||||||
return []
|
return []
|
||||||
|
|
||||||
def delete_proxy(proxy_id: int) -> bool:
|
def delete_proxy(proxy_id: int) -> bool:
|
||||||
@@ -101,10 +112,8 @@ def delete_proxy(proxy_id: int) -> bool:
|
|||||||
cursor.close()
|
cursor.close()
|
||||||
conn.close()
|
conn.close()
|
||||||
|
|
||||||
print(f"Proxy {proxy_id} desativado: {updated}")
|
|
||||||
return updated
|
return updated
|
||||||
except Exception as e:
|
except Exception:
|
||||||
print(f"Erro ao desativar proxy {proxy_id}: {e}")
|
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def format_proxy_url(proxy: Dict) -> str:
|
def format_proxy_url(proxy: Dict) -> str:
|
||||||
@@ -139,8 +148,42 @@ def mark_proxy_success(proxy_id: int) -> bool:
|
|||||||
cursor.close()
|
cursor.close()
|
||||||
conn.close()
|
conn.close()
|
||||||
|
|
||||||
print(f"Proxy (id {proxy_id}) marcado como sucesso")
|
|
||||||
return updated
|
return updated
|
||||||
except Exception as e:
|
except Exception:
|
||||||
print(f"Erro ao marcar proxy {proxy_id} como sucesso: {e}")
|
return False
|
||||||
|
|
||||||
|
def mark_proxy_failure(proxy_id: int, max_failures: int = 10) -> bool:
|
||||||
|
"""
|
||||||
|
Marca proxy como falha e desativa se atingir max_failures consecutivas.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
proxy_id: ID do proxy
|
||||||
|
max_failures: Número máximo de falhas antes de desativar (padrão: 10)
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
conn = get_db_connection()
|
||||||
|
cursor = conn.cursor()
|
||||||
|
|
||||||
|
# Incrementa failure_count e desativa se atingir o limite
|
||||||
|
cursor.execute("""
|
||||||
|
UPDATE proxies
|
||||||
|
SET failure_count = failure_count + 1,
|
||||||
|
last_checked_at = NOW(),
|
||||||
|
updated_at = NOW(),
|
||||||
|
is_active = CASE
|
||||||
|
WHEN failure_count + 1 >= %s THEN FALSE
|
||||||
|
ELSE is_active
|
||||||
|
END
|
||||||
|
WHERE id = %s
|
||||||
|
RETURNING failure_count, is_active
|
||||||
|
""", (max_failures, proxy_id))
|
||||||
|
|
||||||
|
result = cursor.fetchone()
|
||||||
|
conn.commit()
|
||||||
|
cursor.close()
|
||||||
|
conn.close()
|
||||||
|
|
||||||
|
return result is not None
|
||||||
|
|
||||||
|
except Exception:
|
||||||
return False
|
return False
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
services:
|
services:
|
||||||
youtube-api:
|
youtube-api:
|
||||||
|
container_name: youtube-api
|
||||||
build: .
|
build: .
|
||||||
dns: [8.8.8.8, 1.1.1.1]
|
dns: [8.8.8.8, 1.1.1.1]
|
||||||
sysctls:
|
sysctls:
|
||||||
|
|||||||
107
main.py
107
main.py
@@ -105,7 +105,7 @@ def get_video_metadata(
|
|||||||
|
|
||||||
return info
|
return info
|
||||||
|
|
||||||
info = execute_with_proxy_retry(ydl_opts, extract_metadata, retry_per_proxy=2)
|
info = execute_with_proxy_retry(ydl_opts, extract_metadata, retry_per_proxy=1, max_proxies_to_try=3)
|
||||||
|
|
||||||
except ProxyError as e:
|
except ProxyError as e:
|
||||||
error_msg = str(e).replace('\n', ' ').strip()
|
error_msg = str(e).replace('\n', ' ').strip()
|
||||||
@@ -175,16 +175,14 @@ def download_video(
|
|||||||
unique_id = str(uuid.uuid4())
|
unique_id = str(uuid.uuid4())
|
||||||
output_template = os.path.join(videos_dir, f"{unique_id}.%(ext)s")
|
output_template = os.path.join(videos_dir, f"{unique_id}.%(ext)s")
|
||||||
|
|
||||||
ydl_opts = {
|
# Opções base para extração de metadados (operação rápida com proxy)
|
||||||
"format": quality_map[qualidade],
|
metadata_opts = {
|
||||||
"outtmpl": output_template,
|
|
||||||
"quiet": True,
|
"quiet": True,
|
||||||
"noplaylist": True,
|
"no_warnings": True,
|
||||||
"merge_output_format": "mp4",
|
"skip_download": True,
|
||||||
"nocheckcertificate": True,
|
"nocheckcertificate": True,
|
||||||
"socket_timeout": 60,
|
"socket_timeout": 8,
|
||||||
"retries": 0,
|
"retries": 0,
|
||||||
"extractor_retries": 0,
|
|
||||||
"force_ipv4": True,
|
"force_ipv4": True,
|
||||||
"geo_bypass": True,
|
"geo_bypass": True,
|
||||||
"extractor_args": {"youtube": {"player_client": ["android"], "player_skip": ["webpage"]}},
|
"extractor_args": {"youtube": {"player_client": ["android"], "player_skip": ["webpage"]}},
|
||||||
@@ -194,16 +192,51 @@ def download_video(
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Opções para download (operação pesada - tentará com e sem proxy)
|
||||||
|
download_opts = {
|
||||||
|
"format": quality_map[qualidade],
|
||||||
|
"outtmpl": output_template,
|
||||||
|
"quiet": True,
|
||||||
|
"noplaylist": True,
|
||||||
|
"merge_output_format": "mp4",
|
||||||
|
"nocheckcertificate": True,
|
||||||
|
"socket_timeout": 45, # Timeout menor para detectar falha de proxy rapidamente
|
||||||
|
"retries": 0,
|
||||||
|
"extractor_retries": 0,
|
||||||
|
"force_ipv4": True,
|
||||||
|
"geo_bypass": True,
|
||||||
|
"fragment_retries": 3,
|
||||||
|
"file_access_retries": 3,
|
||||||
|
"extractor_args": {"youtube": {"player_client": ["android"], "player_skip": ["webpage"]}},
|
||||||
|
"http_headers": {
|
||||||
|
"Accept-Language": "pt-BR,pt;q=0.9,en-US;q=0.8,en;q=0.7",
|
||||||
|
"User-Agent": "com.google.android.youtube/19.17.36 (Linux; U; Android 13) gzip",
|
||||||
|
},
|
||||||
|
"http_chunk_size": 1048576, # 1MB chunks
|
||||||
|
}
|
||||||
|
|
||||||
try:
|
try:
|
||||||
def download_operation(ydl):
|
# ETAPA 1: Extrair metadados COM PROXY (operação rápida)
|
||||||
base = ydl.extract_info(target, download=False)
|
def extract_metadata_operation(ydl):
|
||||||
title = base.get("title", unique_id)
|
info = ydl.extract_info(target, download=False)
|
||||||
|
if not info or 'title' not in info:
|
||||||
|
raise Exception("Não foi possível extrair metadados do vídeo")
|
||||||
|
return info
|
||||||
|
|
||||||
|
# Tenta apenas 3 proxies para metadados antes de fallback
|
||||||
|
metadata = execute_with_proxy_retry(
|
||||||
|
metadata_opts,
|
||||||
|
extract_metadata_operation,
|
||||||
|
retry_per_proxy=1,
|
||||||
|
max_proxies_to_try=3
|
||||||
|
)
|
||||||
|
|
||||||
|
title = metadata.get("title", unique_id)
|
||||||
clean_title = sanitize_title(title)
|
clean_title = sanitize_title(title)
|
||||||
filename = f"{clean_title}_{qualidade}.mp4"
|
filename = f"{clean_title}_{qualidade}.mp4"
|
||||||
final_path = os.path.join(videos_dir, filename)
|
final_path = os.path.join(videos_dir, filename)
|
||||||
|
|
||||||
print('Info ok')
|
# Verifica cache
|
||||||
|
|
||||||
if os.path.exists(final_path):
|
if os.path.exists(final_path):
|
||||||
return {
|
return {
|
||||||
"videoId": video_id,
|
"videoId": video_id,
|
||||||
@@ -211,10 +244,43 @@ def download_video(
|
|||||||
"cached": True
|
"cached": True
|
||||||
}
|
}
|
||||||
|
|
||||||
print('Lets download')
|
# ETAPA 2: Download COM PROXY (tentativa rápida)
|
||||||
|
def download_with_proxy(ydl):
|
||||||
result = ydl.extract_info(target, download=True)
|
result = ydl.extract_info(target, download=True)
|
||||||
|
return result
|
||||||
|
|
||||||
|
download_success = False
|
||||||
|
result = None
|
||||||
|
|
||||||
|
try:
|
||||||
|
# Tenta apenas 2 proxies para download antes de fallback
|
||||||
|
result = execute_with_proxy_retry(
|
||||||
|
download_opts,
|
||||||
|
download_with_proxy,
|
||||||
|
retry_per_proxy=1,
|
||||||
|
max_proxies_to_try=2
|
||||||
|
)
|
||||||
|
download_success = True
|
||||||
|
except ProxyError as proxy_err:
|
||||||
|
# ETAPA 3: Download SEM PROXY (fallback)
|
||||||
|
|
||||||
|
# Aumenta timeout para download sem proxy
|
||||||
|
download_opts_no_proxy = {**download_opts, "socket_timeout": 180}
|
||||||
|
|
||||||
|
try:
|
||||||
|
with YoutubeDL(download_opts_no_proxy) as ydl:
|
||||||
|
result = ydl.extract_info(target, download=True)
|
||||||
|
download_success = True
|
||||||
|
except Exception as e:
|
||||||
|
raise HTTPException(
|
||||||
|
status_code=500,
|
||||||
|
detail=f"Falha no download com e sem proxy: {str(e)}"
|
||||||
|
)
|
||||||
|
|
||||||
|
if not download_success or not result:
|
||||||
|
raise HTTPException(status_code=500, detail="Erro desconhecido no download")
|
||||||
|
|
||||||
|
# Renomear arquivo para nome final
|
||||||
if "requested_downloads" in result and len(result["requested_downloads"]) > 0:
|
if "requested_downloads" in result and len(result["requested_downloads"]) > 0:
|
||||||
real_file_path = result["requested_downloads"][0]["filepath"]
|
real_file_path = result["requested_downloads"][0]["filepath"]
|
||||||
elif "filepath" in result:
|
elif "filepath" in result:
|
||||||
@@ -222,7 +288,10 @@ def download_video(
|
|||||||
else:
|
else:
|
||||||
real_file_path = output_template.replace("%(ext)s", "mp4")
|
real_file_path = output_template.replace("%(ext)s", "mp4")
|
||||||
|
|
||||||
|
if os.path.exists(real_file_path):
|
||||||
os.rename(real_file_path, final_path)
|
os.rename(real_file_path, final_path)
|
||||||
|
else:
|
||||||
|
raise HTTPException(status_code=500, detail="Arquivo de download não encontrado")
|
||||||
|
|
||||||
return {
|
return {
|
||||||
"videoId": video_id,
|
"videoId": video_id,
|
||||||
@@ -230,8 +299,8 @@ def download_video(
|
|||||||
"cached": False
|
"cached": False
|
||||||
}
|
}
|
||||||
|
|
||||||
return execute_with_proxy_retry(ydl_opts, download_operation, retry_per_proxy=3)
|
except HTTPException:
|
||||||
|
raise
|
||||||
except ProxyError as e:
|
except ProxyError as e:
|
||||||
raise HTTPException(status_code=503, detail=f"Erro com proxies: {e}")
|
raise HTTPException(status_code=503, detail=f"Erro com proxies: {e}")
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
@@ -270,7 +339,7 @@ def search_youtube_yt_dlp(
|
|||||||
})
|
})
|
||||||
return {"results": results}
|
return {"results": results}
|
||||||
|
|
||||||
return execute_with_proxy_retry(ydl_opts, search_operation, retry_per_proxy=2)
|
return execute_with_proxy_retry(ydl_opts, search_operation, retry_per_proxy=1, max_proxies_to_try=3)
|
||||||
|
|
||||||
except ProxyError as e:
|
except ProxyError as e:
|
||||||
raise HTTPException(status_code=503, detail=f"Erro com proxies: {e}")
|
raise HTTPException(status_code=503, detail=f"Erro com proxies: {e}")
|
||||||
@@ -310,7 +379,7 @@ def list_formats(url: str):
|
|||||||
} for f in fmts]
|
} for f in fmts]
|
||||||
return {"total": len(brief), "formats": brief[:60]}
|
return {"total": len(brief), "formats": brief[:60]}
|
||||||
|
|
||||||
return execute_with_proxy_retry(opts, list_formats_operation, retry_per_proxy=2)
|
return execute_with_proxy_retry(opts, list_formats_operation, retry_per_proxy=1, max_proxies_to_try=3)
|
||||||
|
|
||||||
except ProxyError as e:
|
except ProxyError as e:
|
||||||
raise HTTPException(status_code=503, detail=f"Erro com proxies: {e}")
|
raise HTTPException(status_code=503, detail=f"Erro com proxies: {e}")
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
from typing import Callable, Any, Optional
|
from typing import Callable, Any, Optional
|
||||||
from yt_dlp import YoutubeDL
|
from yt_dlp import YoutubeDL
|
||||||
from database import get_all_active_proxies, format_proxy_url, mark_proxy_success
|
from database import get_all_active_proxies, format_proxy_url, mark_proxy_success, mark_proxy_failure
|
||||||
|
|
||||||
class ProxyError(Exception):
|
class ProxyError(Exception):
|
||||||
pass
|
pass
|
||||||
@@ -32,6 +32,11 @@ def is_proxy_error(error_msg: str) -> bool:
|
|||||||
'ssl', # Erros SSL causados por proxy
|
'ssl', # Erros SSL causados por proxy
|
||||||
'certificate', # Erros de certificado causados por proxy
|
'certificate', # Erros de certificado causados por proxy
|
||||||
'certificate_verify_failed', # Verificação de certificado SSL falhou
|
'certificate_verify_failed', # Verificação de certificado SSL falhou
|
||||||
|
'failed to parse json', # Erros de parsing JSON (proxy retornando HTML)
|
||||||
|
'jsondecode', # Erros de decodificação JSON
|
||||||
|
'remote end closed', # Conexão fechada pelo proxy
|
||||||
|
'remotedisconnected', # Desconexão remota
|
||||||
|
'connection/parsing error', # Erros de conexão/parsing do download
|
||||||
]
|
]
|
||||||
|
|
||||||
non_proxy_error_keywords = [
|
non_proxy_error_keywords = [
|
||||||
@@ -53,10 +58,11 @@ def is_proxy_error(error_msg: str) -> bool:
|
|||||||
def execute_with_proxy_retry(
|
def execute_with_proxy_retry(
|
||||||
ydl_opts: dict,
|
ydl_opts: dict,
|
||||||
operation: Callable[[YoutubeDL], Any],
|
operation: Callable[[YoutubeDL], Any],
|
||||||
retry_per_proxy: int = 2
|
retry_per_proxy: int = 2,
|
||||||
|
max_proxies_to_try: int = None
|
||||||
) -> Any:
|
) -> Any:
|
||||||
"""
|
"""
|
||||||
Tenta executar operação com todos os proxies disponíveis.
|
Tenta executar operação com proxies disponíveis.
|
||||||
Cada proxy é tentado N vezes antes de passar para o próximo.
|
Cada proxy é tentado N vezes antes de passar para o próximo.
|
||||||
Proxies NÃO são removidos do banco, apenas pulados.
|
Proxies NÃO são removidos do banco, apenas pulados.
|
||||||
|
|
||||||
@@ -64,34 +70,34 @@ def execute_with_proxy_retry(
|
|||||||
ydl_opts: Opções do YoutubeDL
|
ydl_opts: Opções do YoutubeDL
|
||||||
operation: Função a ser executada
|
operation: Função a ser executada
|
||||||
retry_per_proxy: Número de tentativas por proxy antes de pular para o próximo
|
retry_per_proxy: Número de tentativas por proxy antes de pular para o próximo
|
||||||
|
max_proxies_to_try: Número máximo de proxies a tentar (None = todos)
|
||||||
"""
|
"""
|
||||||
# Busca TODOS os proxies ativos
|
# Busca TODOS os proxies ativos
|
||||||
all_proxies = get_all_active_proxies()
|
all_proxies = get_all_active_proxies()
|
||||||
total_proxies = len(all_proxies)
|
|
||||||
|
|
||||||
print(f"\n{'='*60}")
|
# Limita número de proxies se especificado
|
||||||
print(f"Proxies disponíveis no banco: {total_proxies}")
|
if max_proxies_to_try is not None and max_proxies_to_try > 0:
|
||||||
print(f"Tentativas por proxy: {retry_per_proxy}")
|
all_proxies = all_proxies[:max_proxies_to_try]
|
||||||
print(f"{'='*60}\n")
|
|
||||||
|
total_proxies = len(all_proxies)
|
||||||
|
|
||||||
last_error = None
|
last_error = None
|
||||||
|
|
||||||
# Tenta cada proxy da lista
|
# Tenta cada proxy da lista
|
||||||
for proxy_index, proxy_data in enumerate(all_proxies, 1):
|
for proxy_index, proxy_data in enumerate(all_proxies, 1):
|
||||||
proxy_url = format_proxy_url(proxy_data)
|
proxy_url = format_proxy_url(proxy_data)
|
||||||
ydl_opts_with_proxy = {**ydl_opts, 'proxy': proxy_url}
|
# Proxy desativado temporariamente - Para reativar, descomente a linha abaixo:
|
||||||
|
# ydl_opts_with_proxy = {**ydl_opts, 'proxy': proxy_url}
|
||||||
|
ydl_opts_with_proxy = {**ydl_opts} # Sem proxy por enquanto
|
||||||
|
|
||||||
print(f"\n[Proxy {proxy_index}/{total_proxies}] {proxy_url} (ID: {proxy_data['id']})")
|
proxy_failed = False
|
||||||
|
|
||||||
# Tenta N vezes com o MESMO proxy
|
# Tenta N vezes com o MESMO proxy
|
||||||
for attempt in range(1, retry_per_proxy + 1):
|
for attempt in range(1, retry_per_proxy + 1):
|
||||||
try:
|
try:
|
||||||
print(f" → Tentativa {attempt}/{retry_per_proxy}...", end=" ")
|
|
||||||
|
|
||||||
with YoutubeDL(ydl_opts_with_proxy) as ydl:
|
with YoutubeDL(ydl_opts_with_proxy) as ydl:
|
||||||
result = operation(ydl)
|
result = operation(ydl)
|
||||||
|
|
||||||
print(f"✓ SUCESSO!")
|
|
||||||
mark_proxy_success(proxy_data['id'])
|
mark_proxy_success(proxy_data['id'])
|
||||||
return result
|
return result
|
||||||
|
|
||||||
@@ -99,32 +105,22 @@ def execute_with_proxy_retry(
|
|||||||
error_msg = str(e)
|
error_msg = str(e)
|
||||||
last_error = e
|
last_error = e
|
||||||
|
|
||||||
print(f"✗ Falhou")
|
|
||||||
print(f" Erro: {error_msg[:80]}...")
|
|
||||||
|
|
||||||
# Se não for erro de proxy, lança imediatamente
|
# Se não for erro de proxy, lança imediatamente
|
||||||
if not is_proxy_error(error_msg):
|
if not is_proxy_error(error_msg):
|
||||||
print(f" ⚠ Erro não relacionado a proxy, abortando")
|
|
||||||
raise e
|
raise e
|
||||||
|
|
||||||
|
proxy_failed = True
|
||||||
|
|
||||||
# Se chegou aqui, falhou todas as tentativas com este proxy
|
# Se chegou aqui, falhou todas as tentativas com este proxy
|
||||||
print(f" ⨯ Proxy falhou {retry_per_proxy} vezes, pulando para o próximo...")
|
if proxy_failed:
|
||||||
|
mark_proxy_failure(proxy_data['id'])
|
||||||
|
|
||||||
# Se chegou aqui, todos os proxies falharam, tenta SEM proxy
|
# Se chegou aqui, todos os proxies falharam, tenta SEM proxy
|
||||||
print(f"\n{'='*60}")
|
|
||||||
print(f"Todos os {total_proxies} proxies falharam")
|
|
||||||
print(f"Tentando SEM proxy como último recurso...")
|
|
||||||
print(f"{'='*60}\n")
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
print(f" → Tentativa sem proxy...", end=" ")
|
|
||||||
with YoutubeDL(ydl_opts) as ydl:
|
with YoutubeDL(ydl_opts) as ydl:
|
||||||
result = operation(ydl)
|
result = operation(ydl)
|
||||||
print(f"✓ SUCESSO!")
|
|
||||||
return result
|
return result
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
print(f"✗ Falhou")
|
|
||||||
print(f" Erro: {str(e)[:80]}...")
|
|
||||||
last_error = e
|
last_error = e
|
||||||
|
|
||||||
raise ProxyError(
|
raise ProxyError(
|
||||||
|
|||||||
Reference in New Issue
Block a user