Recentemente criei um script, usando a ferramenta yt-dlp –
para Linux –, para baixar o áudio de vídeos do YouTube e
organizá-los em pastas específicas no meu celular/computador.
Apesar do projeto ser simples (é só um arquivo de umas 30 linhas),
aprendi algumas coisas interessantes e tive ideias que não acho
que seria legal não deixar elas registradas em algum canto.
Vou me atentar em registrar o que eu fiz, como eu fiz e porquê eu fiz. Além disso, tem algumas coisas que preciso estudar mais pra frente e coisas que não tem muito haver com código em si que quero abordar.
Porquê eu não uso um aplicativo aleatório de conversão
Sites de conversão não me permitem ter a liberdade que eu teria
se usasse algum aplicativo para celular que fizesse o mesmo
serviço. Normalmente o navegador salva esses arquivos na pasta
/emulated/0/Downloads/, o que deixa o processo de
baixar o áudio dos vídeos bem mais trabalhoso, visto que eu teria
que mover manualmente os arquivos para as pastas onde guardo
minhas músicas, o trabalho seria muito maior se eu quisesse baixar
uma playlist inteira, e maior ainda para organizar.
Conheço poucos, mas com certeza há aplicativos que consigam
fazer tudo o que eu preciso, mas ainda acho que não seja uma boa
solução para o meu contexto. Não me entenda mal, eu entendo que
não há necessidade de reinventar a roda, mas achei que seria uma
boa oportunidade de aprender algo novo nessa questão. Outro motivo
de eu não usar usar aplicativos para Andorid: Eles não seguem a
Unix philosophyA Unix philosophy é uma filosofia para
desenvolver software que muitas aplicações para ambientes
Unix-like (tipo Linux, MacOS,
BSD, etc). Ela pode ser resumida em poucas palavras:
“Faça só uma coisa, e faça bem feita”.
, o que não é algo ruim (longe disso), mas só quer
dizer que les fazem o que eu preciso e mais um pouco, e esse “mais
um pouco” acaba ocupando um espaço desnecessário no meu
armazenamento.
Vantagens de se saber programar
Outra razão de eu ter feito essa ferramenta: eu sei.
Lógicamente, eu não fiz o yt-dlp, apenas o usei para
baixar as músicas e extrair para MP3 e escrevi o script para
salvar onde eu quero. Estive estudando sobre como rodar comandos
em Shell por programas em Python e uso Linux – significa que sei
Shell porque sempre estou usando o terminal –, acabei conhecendo o
yt-dlp e aprendi a extrair o áudio em MP3 com ele.
Juntando uma linguagem simples como Python e essa ferramenta que é
executada por linha de comando, criei esse facilitador para mim
que gosta de escutar música offline.
O interessante foi que essa ideia surgiu quando pensei: “Putz, acho que dava pra automatizar esse processo”. Programar é uma skill que me ajuda muito a ser “preguiçoso”, no sentido de gastar tempo e esforço para criar alguma coisa que repita um processo várias vezes para mim.
Como eu criei essa ferramenta
Eu escolhi o Python pela facilidade, realmente foi um programa que comecei a escrever segundos depois que pensei na solução para o meu problema. Não só isso, como os sitemas Android e Arch Linux (distribuição Linux que estou usando) são Unix-like, o Python já vem pré-instalado – e se não vier, um siples comando resolve em 5 segundos. Escolhi Python também por me permitir usar essa função de executar comandos do sistema nativamente, o que me permitiu escrever tudo em um único arquivo, aí eu só precisei me preocupar com a lógica e como eu iria estruturar tudo.
Como você deve imaginar, esse aplicativo não têm interface gráfica, você executa ele por linha de comando.
Organização desses arquivos e como eu rodo o código
Uma vez com o script criado, precisei passar ele para o meu celular e organizar tudo. Eu estou usando o Termux para emular o terminal no Android, é com esse aplicativo que configuro o meu ambiente pessoal de estudos (ou de desenvolvimento numa emergência) e anotações.
A pasta ~/scripts/bin/ está no $PATH
do meu android, é lá onde guardo arquivos com permissão de
execução. Faço isso por que não preciso me preocupar com a
extensão do arquivo ou algo assim e fica mais fácil quando escrevo
algo realmente útil para o meu celular. Salvei o arquivo como
yt, ou seja, ~/scripts/bin/yt, então só
preciso o dar o comando yt e ele começa a rodar.
No entanto, eu programei para a pasta de origem ser o diretório
atual, mas dá para editar isso passando o novo local do
output nos argumentos do comando – acredito que faça
sentido, já que a URL também vai ser passada como um argumento, e
será mais fácil de mudar o comportamento disso se necessário. Eu
sempre guardo minhas músicas em /emulated/0/Musics/,
então configurei meu ~/.bashrc do seguinte modo:
alias yt="yt --dir /emulated/0/Musics"
alias ytP="yt --playlist --dir /emulated/0/Musics"O comando para baixar todos os vídeos de uma playlist salva os vídeos em uma pasta separada. Com isso tudo, fica bem mais fácil o trabalho de criar meus álbuns e organizar o que quero ouvir sem depender do 4G que falha toda hora.
E como esse programa foi feita por mim, para resolver um
problema meu, não têm necessidade de ele ser incrível, afinal é só
um projeto pessoal afinal de contas. Um único arquivinho já basta,
não têm porquê colocar ele em algum repositório de pacotes grande
– tipo o do pip ou da AUR – e usar o
yt-dlp como dependência principal, o script só têm 70
linhas.
Contras da minha solução…
É muito lento, apesar de não ser um problema pra mim, ainda é
meio chato. O yt-dlp precisa baixar o vídeo em
WEBM pra depois converter para MP3. Além disso, a
tela do meu celular precisa estar ligada para fazer o download – o
restante ele roda em background, e consegue converter para MP3 com
a tela desligada sem problemas –, e também acredito que o tamanho
dos arquivos sejam muito grandes. Não entendo muito como essas
coisas são formatadas, mas alguns áudios conseguem serm só alguns
MB mais leve, o que me incomoda.
Apesar disso tudo, estou satisfeito com o resultado. O tempo entre eu copiar um link de um vídeo/playlist e eu ter as músicas na pasta que eu quero pronto para o VLC (midia player que uso no meu Android) tocar, ficou mais curto, mesmo com a demora no download e na conversão. Sem falar que consegui, ao menos:
- Clean code com Python
- Exercitei a minha skill pra solucionar problemas
- Me divertir por um tempo
- Me atualizar com Python, que já fazia um tempo que sequer encostava
Vou continuar usando isso e desse jeito até eu achar a solução para esses problemas.
Ainda preciso resolver depois
- Encontrar um formato de compressão de áudio mais leve que MP3
-
Ler melhor a documentação do
yt-dlp - Escrever um script para limpar o nome dos arquivos2
- Treinar com esse projeto usando outras linguagens
Python code documentation
Tentei organizar o código seguindo os paterns de clean code que os PyThOnIsTaS costumam usar, então acredito que esteja self documentaded o sulficiente. Essa é a segunda versão do código, a primeira eu tentave renomear o nome dos arquivos salvos e usava OOP, o que não dá muito certo com Python num projeto pequeno, nessa versão eu foquei em deixar o código legível e menos bloat.
E só por que sim, resolvi chamar o arquivo de
better_yt-dlp.py. Se for usar esse pequeno programa,
salve-o com um nome menos complicado, e numa pasta que esteja no
path do seu sistema operacional. A extensão do arquivo
não é necessária.
#!/usr/bin/env python3
from os import mkdir, system
from sys import argv
# DESC: Função para escrever mensagens em AZUL
def __blue(msg: str) -> str:
return f'\033[31m{msg}\033[m'
# DESC: Função para escrever mensagens em VERMELHO
def __red(msg: str) -> str:
return f'\033[34m{msg}\033[m'
# INFO: Interrompe o programa e devolve a mensagem de erro
def error_callback(error: Exception()) -> None:
print(__red('😓 Argumentos inválidos... :/'))
raise error
# INFO: Seleciona as informações relevantes dos argumentos e joga num dicionário
def process_arguments() -> dict():
default_opts: dict() = { 'playlist': '', 'dir': '.', 'video': None }
args: list(str) = argv[1:]
for key, value in enumerate(args):
if value in ['--playlist', '-p']:
default_opts['playlist'] = '--no-flat-playlist'
elif value in ['--dir', '-d']:
try:
default_opts['dir'] = args[key + 1]
except Exception as error:
error_callback(error)
elif 'http://' in value or 'https://' in value:
default_opts['video'] = value
if not default_opts['video']:
error_callback(Exception(__blue('O vídeo não foi especificado nos argumentos')))
return default_opts
# DESC: Bloco principal do script
def main() -> None:
opts: dict() = process_arguments()
if opts['playlist']:
playlist_name: str = input(__blue('Qual é o nome da playlist? 😎 '))
opts['dir'] = f'{opts["dir"]}/{playlist_name}'
try:
mkdir(opts['dir'])
except Exception as error:
error_callback(error)
system(f'cd {opts["dir"]}; yt-dlp {opts["playlist"]} -x --audio-format mp3 {opts["video"]}')
if __name__ == '__main__':
main()Usage
O ideal é você definir algum tipo de atalho para esses comandos, para evitar de precisar escrever isso tudo sempre que for baixar alguma música – o objetivo é ser rápido nesse sentido, afinal. E acredito que você sempre vá querer baixar as músicas sempre na mesma pasta, para manter organizado.
desc: Programinha simples para baixar música do YouTube.
command: ./better_yt-dlp.py [--dir OUTPUT_PATH] [--playlist] VIDEO_URL
arguments:
--playlist|-p: Salva todos os vídeos de uma playlist, mas pergunta o nome de uma nova pasta para salvá-las
--dir|-d OUTPUT_PATH: Pasta para salvar as músicas/playlists, o valor padrão é o diretório atual
VIDEO_URL: URL do(s) vídeo(s), é melhor passar entre aspas para evitar erros com o shell...
examples:
- ./better_yt-dlp.py 'https://youtube.com/SomeURL/'
- ./better_yt-dlp.py --playlist 'https://youtube.com/SomePlaylistURL/'
- ./better_yt-dlp.py -p --dir ~/Musics 'https://youtube.com/SomePlaylistURL/'