Compare commits
43 Commits
890a0dd5c9
..
v1.0.1
| Author | SHA1 | Date | |
|---|---|---|---|
|
5748834af4
|
|||
|
06d2a5ef01
|
|||
|
bdcd77eff1
|
|||
|
4541559f47
|
|||
|
906ef98bd5
|
|||
|
640c952b20
|
|||
|
f30686fd7c
|
|||
|
eceb3b3d76
|
|||
|
23a27217df
|
|||
|
82a6421c88
|
|||
|
c72bc42496
|
|||
|
991a5f376f
|
|||
|
13bd3b82db
|
|||
|
2c99fbf517
|
|||
| 81968a6811 | |||
|
7d83fd8518
|
|||
| 7b34919e72 | |||
|
98fec74ac0
|
|||
| 2269104805 | |||
| 4bea5e020f | |||
| 60803961b3 | |||
|
197e35314a
|
|||
|
db0c0a3893
|
|||
|
46fec9b85a
|
|||
|
204404b761
|
|||
| 6e7fc73420 | |||
|
67c85aebf9
|
|||
| b9088d932c | |||
|
8c1d044f79
|
|||
| f688ee035f | |||
|
7fd5d93b6d
|
|||
|
d7ad90a1d5
|
|||
| ac5abffd74 | |||
|
1c43c62523
|
|||
| bef0a4d593 | |||
|
bf7739228f
|
|||
| 270934613f | |||
|
5cea64626c
|
|||
| 70d72759ce | |||
|
c537874adb
|
|||
|
b496d14cf7
|
|||
|
c2de1abfd2
|
|||
|
635d5d5113
|
@@ -23,14 +23,14 @@ jobs:
|
|||||||
|
|
||||||
# Set up Docker Buildx for building the image
|
# Set up Docker Buildx for building the image
|
||||||
- name: Set up Docker Buildx
|
- name: Set up Docker Buildx
|
||||||
uses: docker/setup-buildx-action@v3
|
uses: docker/setup-buildx-action@v4
|
||||||
with:
|
with:
|
||||||
driver: remote
|
driver: remote
|
||||||
endpoint: 'tcp://buildkitd:1234'
|
endpoint: 'tcp://buildkitd:1234'
|
||||||
|
|
||||||
# Log in to the Gitea container registry
|
# Log in to the Gitea container registry
|
||||||
- name: Log in to Gitea Container Registry
|
- name: Log in to Gitea Container Registry
|
||||||
uses: docker/login-action@v3
|
uses: docker/login-action@v4
|
||||||
with:
|
with:
|
||||||
registry: git.dubyatp.xyz
|
registry: git.dubyatp.xyz
|
||||||
username: williamp
|
username: williamp
|
||||||
@@ -38,7 +38,7 @@ jobs:
|
|||||||
|
|
||||||
# Build and push the Docker image
|
# Build and push the Docker image
|
||||||
- name: Build and Push Docker Image
|
- name: Build and Push Docker Image
|
||||||
uses: docker/build-push-action@v6
|
uses: docker/build-push-action@v7
|
||||||
with:
|
with:
|
||||||
context: . # Build context (current directory)
|
context: . # Build context (current directory)
|
||||||
file: ./Dockerfile # Path to Dockerfile
|
file: ./Dockerfile # Path to Dockerfile
|
||||||
@@ -75,8 +75,13 @@ jobs:
|
|||||||
git config --local user.signingkey ~/.ssh/id_ed25519
|
git config --local user.signingkey ~/.ssh/id_ed25519
|
||||||
git config --local gpg.format ssh
|
git config --local gpg.format ssh
|
||||||
git config --local commit.gpgsign true
|
git config --local commit.gpgsign true
|
||||||
git commit -a -m "yt-dlp-bot: deploy update to ${{ needs.build-and-push.outputs.sha_short }}"
|
|
||||||
|
if [ -n "$(git status --porcelain)" ]; then
|
||||||
|
git commit -a -m "yt-dlp-bot: deploy update to ${{ needs.build-and-push.outputs.sha_short }}"
|
||||||
|
else
|
||||||
|
echo "No changes to commit, skipping..."
|
||||||
|
fi
|
||||||
|
exit 0
|
||||||
- name: Push changes
|
- name: Push changes
|
||||||
uses: ad-m/github-push-action@v1.0.0
|
uses: ad-m/github-push-action@v1.0.0
|
||||||
with:
|
with:
|
||||||
|
|||||||
@@ -0,0 +1,35 @@
|
|||||||
|
name: Build only (for PRs)
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build-only:
|
||||||
|
runs-on: ubuntu-latest # Use a runner with Docker support
|
||||||
|
container: ghcr.io/catthehacker/ubuntu:act-latest # Image with Docker pre-installed
|
||||||
|
outputs:
|
||||||
|
sha_short: ${{ steps.vars.outputs.sha_short }}
|
||||||
|
steps:
|
||||||
|
# Checkout the repository code
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v6
|
||||||
|
|
||||||
|
- name: Set outputs
|
||||||
|
id: vars
|
||||||
|
run: |
|
||||||
|
echo "sha_short=$(git rev-parse --short HEAD)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
# Set up Docker Buildx for building the image
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v3
|
||||||
|
with:
|
||||||
|
driver: remote
|
||||||
|
endpoint: 'tcp://buildkitd:1234'
|
||||||
|
|
||||||
|
# Build the Docker image
|
||||||
|
- name: Build Docker Image
|
||||||
|
uses: docker/build-push-action@v6
|
||||||
|
with:
|
||||||
|
context: . # Build context (current directory)
|
||||||
|
file: ./Dockerfile # Path to Dockerfile
|
||||||
|
platforms: "linux/amd64,linux/arm64"
|
||||||
+91
-6
@@ -1,6 +1,91 @@
|
|||||||
FROM python:3.14.2-alpine3.22
|
# Portions of this Dockerfile are sourced from GPLv3 licensed `yt-dlp slim` by Henrique Almeida (https://github.com/h3nc4/yt-dlp-slim)
|
||||||
COPY ./app /app
|
# Derivations to this Dockerfile in this repository following March 3, 2026 should be considered licensed under this project's MIT license (see ../LICENSE) unless otherwise stated
|
||||||
WORKDIR /app
|
|
||||||
RUN apk add ffmpeg deno
|
########################################
|
||||||
RUN pip install -r requirements.txt
|
# Versions
|
||||||
CMD ["python", "/app/main.py"]
|
ARG YT_DLP_VERSION="2026.03.03"
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# Deno builder stage
|
||||||
|
FROM denoland/deno:bin-2.6.6@sha256:9f18d20207f2699595ea26d14e0b7e123cd0cd01100a577bc11f8ca5906c2d81 AS deno-builder
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# YT-DLP builder stage
|
||||||
|
FROM alpine:3.23@sha256:25109184c71bdad752c8312a8623239686a9a2071e8825f20acb8f2198c3f659 AS yt-dlp-builder
|
||||||
|
ARG YT_DLP_VERSION
|
||||||
|
ARG TARGETARCH
|
||||||
|
|
||||||
|
RUN mkdir -p /rootfs/target /rootfs/tmp /rootfs/bin
|
||||||
|
|
||||||
|
ADD "https://github.com/yt-dlp/yt-dlp/releases/download/${YT_DLP_VERSION}/SHA2-256SUMS" /SHA2-256SUMS
|
||||||
|
ADD "https://github.com/yt-dlp/yt-dlp/releases/download/${YT_DLP_VERSION}/SHA2-256SUMS.sig" /SHA2-256SUMS.sig
|
||||||
|
ADD "https://keyserver.ubuntu.com/pks/lookup?op=get&search=0xAC0CBBE6848D6A873464AF4E57CF65933B5A7581" "/yt-dlp_pubkey.asc"
|
||||||
|
|
||||||
|
RUN apk add --no-cache gnupg && \
|
||||||
|
gpg --import /yt-dlp_pubkey.asc && \
|
||||||
|
gpg --verify /SHA2-256SUMS.sig /SHA2-256SUMS
|
||||||
|
|
||||||
|
RUN case "$TARGETARCH" in \
|
||||||
|
amd64) YT_DLP_FILE="yt-dlp_linux" ;; \
|
||||||
|
arm64) YT_DLP_FILE="yt-dlp_linux_aarch64" ;; \
|
||||||
|
*) echo "Unsupported architecture: $TARGETARCH" && exit 1 ;; \
|
||||||
|
esac && \
|
||||||
|
wget -qO "/${YT_DLP_FILE}" "https://github.com/yt-dlp/yt-dlp/releases/download/${YT_DLP_VERSION}/${YT_DLP_FILE}" && \
|
||||||
|
grep " ${YT_DLP_FILE}$" /SHA2-256SUMS | sha256sum -c - && \
|
||||||
|
mv "/${YT_DLP_FILE}" /rootfs/bin/yt-dlp && \
|
||||||
|
chmod 755 /rootfs/bin/yt-dlp && \
|
||||||
|
chmod 1777 /rootfs/tmp
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# FFmpeg builder stage
|
||||||
|
FROM debian:13-slim@sha256:1d3c811171a08a5adaa4a163fbafd96b61b87aa871bbc7aa15431ac275d3d430 AS ffmpeg-builder
|
||||||
|
RUN apt-get update && \
|
||||||
|
apt-get install -y --no-install-recommends ffmpeg
|
||||||
|
COPY --from=yt-dlp-builder /rootfs/bin/yt-dlp /yt-dlp
|
||||||
|
RUN mkdir -p /rootfs/bin && \
|
||||||
|
cp /usr/bin/ffmpeg /usr/bin/ffprobe /rootfs/bin/ && \
|
||||||
|
{ ldd /usr/bin/ffmpeg; ldd /yt-dlp; } 2>/dev/null | \
|
||||||
|
grep -o '/[^ ]*' | sort -u | \
|
||||||
|
xargs -I '{}' cp --parents '{}' /rootfs && \
|
||||||
|
LIBDIR=$(dirname "$(find /rootfs -name 'libc.so.6' | head -1)") && \
|
||||||
|
for stub in libutil.so.1 libdl.so.2 libpthread.so.0 librt.so.1; do \
|
||||||
|
[ -f "${LIBDIR}/${stub}" ] || ln -sf libc.so.6 "${LIBDIR}/${stub}"; \
|
||||||
|
done
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# App builder stage
|
||||||
|
FROM golang:1.25.8-trixie@sha256:bc16125656839ffe56154c675f7a9662bec2ef7d4060177239914e7c6d2fd8a8 AS app-builder
|
||||||
|
|
||||||
|
COPY app/ /opt/app
|
||||||
|
WORKDIR /opt/app
|
||||||
|
|
||||||
|
RUN go get && go build -o out/yt-dlp-bot
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# Final squashed image
|
||||||
|
FROM scratch AS final
|
||||||
|
|
||||||
|
# Copy deno, yt-dlp, and ffmpeg binaries
|
||||||
|
COPY --from=deno-builder /deno /bin/deno
|
||||||
|
COPY --from=yt-dlp-builder /rootfs /
|
||||||
|
COPY --from=ffmpeg-builder /rootfs/ /
|
||||||
|
|
||||||
|
# Copy yt-dlp-bot app binary
|
||||||
|
COPY --from=app-builder /opt/app/out/yt-dlp-bot /bin/
|
||||||
|
|
||||||
|
# Copy SSL CA's (needed for Discord)
|
||||||
|
COPY --from=app-builder /etc/ssl/certs /etc/ssl/certs
|
||||||
|
|
||||||
|
WORKDIR /target
|
||||||
|
ENV XDG_CACHE_HOME=/tmp/.cache
|
||||||
|
|
||||||
|
ENV YTDLP_BIN=/bin/yt-dlp
|
||||||
|
|
||||||
|
ENTRYPOINT ["/bin/yt-dlp-bot"]
|
||||||
|
|
||||||
|
LABEL org.opencontainers.image.title="yt-dlp bot" \
|
||||||
|
org.opencontainers.image.description="A totally overengineered Discord bot to locally download YouTube videos for private use" \
|
||||||
|
org.opencontainers.image.authors="William Peebles <me@williamtpeebles.com>" \
|
||||||
|
org.opencontainers.image.vendor="William Peebles" \
|
||||||
|
org.opencontainers.image.licenses="MIT" \
|
||||||
|
org.opencontainers.image.source="https://git.dubyatp.xyz/williamp/yt-dlp-bot"
|
||||||
+7
-7
@@ -20,7 +20,7 @@ func startAsyncDownload(s *discordgo.Session, i *discordgo.InteractionCreate, re
|
|||||||
// Handle panic from DownloadVideo
|
// Handle panic from DownloadVideo
|
||||||
resultChan <- DownloadResult{
|
resultChan <- DownloadResult{
|
||||||
Success: false,
|
Success: false,
|
||||||
Message: fmt.Sprintf("Download failed: %v", r),
|
Message: fmt.Sprintf("❌ **Download Failed**: %v", r),
|
||||||
URL: url,
|
URL: url,
|
||||||
Format: fmt.Sprintf("video: %s, audio: %s", videoFormatID, audioFormatID),
|
Format: fmt.Sprintf("video: %s, audio: %s", videoFormatID, audioFormatID),
|
||||||
Error: fmt.Errorf("%v", r),
|
Error: fmt.Errorf("%v", r),
|
||||||
@@ -39,7 +39,7 @@ func startAsyncDownload(s *discordgo.Session, i *discordgo.InteractionCreate, re
|
|||||||
// If we reach here, download was successful
|
// If we reach here, download was successful
|
||||||
resultChan <- DownloadResult{
|
resultChan <- DownloadResult{
|
||||||
Success: true,
|
Success: true,
|
||||||
Message: "Video Downloaded Successfully!",
|
Message: "✅ **Successfully Downloaded**",
|
||||||
URL: url,
|
URL: url,
|
||||||
Format: fmt.Sprintf("video: %s, audio: %s", videoFormatID, audioFormatID),
|
Format: fmt.Sprintf("video: %s, audio: %s", videoFormatID, audioFormatID),
|
||||||
Error: nil,
|
Error: nil,
|
||||||
@@ -49,7 +49,7 @@ func startAsyncDownload(s *discordgo.Session, i *discordgo.InteractionCreate, re
|
|||||||
// Handle progress and results asynchronously
|
// Handle progress and results asynchronously
|
||||||
go func() {
|
go func() {
|
||||||
_, err := s.InteractionResponseEdit(i.Interaction, &discordgo.WebhookEdit{
|
_, err := s.InteractionResponseEdit(i.Interaction, &discordgo.WebhookEdit{
|
||||||
Content: ptr("<a:loading:1479131733910618153> downloading: starting..."),
|
Content: ptr(fmt.Sprintf("%s **Starting Download**", loading_emoji)),
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("Error updating interaction: %v", err)
|
log.Printf("Error updating interaction: %v", err)
|
||||||
@@ -65,7 +65,7 @@ func startAsyncDownload(s *discordgo.Session, i *discordgo.InteractionCreate, re
|
|||||||
|
|
||||||
var content string
|
var content string
|
||||||
if prog.Phase == "post-processing" {
|
if prog.Phase == "post-processing" {
|
||||||
content = "<a:loading:1479131733910618153> post-processing"
|
content = fmt.Sprintf("%s **Post Processing**", loading_emoji)
|
||||||
} else {
|
} else {
|
||||||
var progressStr string
|
var progressStr string
|
||||||
if prog.DownloadedBytes > 0 {
|
if prog.DownloadedBytes > 0 {
|
||||||
@@ -73,7 +73,7 @@ func startAsyncDownload(s *discordgo.Session, i *discordgo.InteractionCreate, re
|
|||||||
} else {
|
} else {
|
||||||
progressStr = "starting..."
|
progressStr = "starting..."
|
||||||
}
|
}
|
||||||
content = fmt.Sprintf("<a:loading:1479131733910618153> Downloading Video: %s", progressStr)
|
content = fmt.Sprintf("%s **Downloading Video**: %s", loading_emoji, progressStr)
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err := s.InteractionResponseEdit(i.Interaction, &discordgo.WebhookEdit{
|
_, err := s.InteractionResponseEdit(i.Interaction, &discordgo.WebhookEdit{
|
||||||
@@ -86,7 +86,7 @@ func startAsyncDownload(s *discordgo.Session, i *discordgo.InteractionCreate, re
|
|||||||
case result := <-resultChan:
|
case result := <-resultChan:
|
||||||
if result.Success {
|
if result.Success {
|
||||||
_, err = s.InteractionResponseEdit(i.Interaction, &discordgo.WebhookEdit{
|
_, err = s.InteractionResponseEdit(i.Interaction, &discordgo.WebhookEdit{
|
||||||
Content: ptr("✅ Success"),
|
Content: ptr("✅ **Successfully Downloaded**"),
|
||||||
})
|
})
|
||||||
_, err = s.FollowupMessageCreate(i.Interaction, false, &discordgo.WebhookParams{
|
_, err = s.FollowupMessageCreate(i.Interaction, false, &discordgo.WebhookParams{
|
||||||
Content: fmt.Sprintf("## Video Downloaded \n**URL**: %s \n**Quality**: %s + %s \n**Requested By**: <@%s> \n", result.URL, videoFormatName, audioFormatName, requester),
|
Content: fmt.Sprintf("## Video Downloaded \n**URL**: %s \n**Quality**: %s + %s \n**Requested By**: <@%s> \n", result.URL, videoFormatName, audioFormatName, requester),
|
||||||
@@ -96,7 +96,7 @@ func startAsyncDownload(s *discordgo.Session, i *discordgo.InteractionCreate, re
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
_, err = s.InteractionResponseEdit(i.Interaction, &discordgo.WebhookEdit{
|
_, err = s.InteractionResponseEdit(i.Interaction, &discordgo.WebhookEdit{
|
||||||
Content: ptr("❌ Download failed: " + result.Message),
|
Content: ptr("❌ **Download Failed**: " + result.Message),
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("Error updating interaction: %v", err)
|
log.Printf("Error updating interaction: %v", err)
|
||||||
|
|||||||
+2
-4
@@ -13,7 +13,6 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
|
|
||||||
out_dir := os.Getenv("OUT_PATH")
|
out_dir := os.Getenv("OUT_PATH")
|
||||||
temp_dir := os.Getenv("TEMP_PATH")
|
temp_dir := os.Getenv("TEMP_PATH")
|
||||||
bot_token := os.Getenv("DISCORD_TOKEN")
|
bot_token := os.Getenv("DISCORD_TOKEN")
|
||||||
@@ -273,8 +272,7 @@ func main() {
|
|||||||
response := ""
|
response := ""
|
||||||
if state.URL != "" {
|
if state.URL != "" {
|
||||||
// Respond immediately to prevent timeout
|
// Respond immediately to prevent timeout
|
||||||
response = fmt.Sprintf("🚀 Starting download...\nURL: %s\nVideo: %s\nAudio: %s\n\nYou'll receive an update when the download completes!",
|
response = fmt.Sprintf("%s **Starting download**", loading_emoji)
|
||||||
state.URL, state.VideoFormatID, state.AudioFormatID)
|
|
||||||
|
|
||||||
// Start async download after responding
|
// Start async download after responding
|
||||||
go func() {
|
go func() {
|
||||||
@@ -327,7 +325,7 @@ func main() {
|
|||||||
err := s.InteractionRespond(i.Interaction, &discordgo.InteractionResponse{
|
err := s.InteractionRespond(i.Interaction, &discordgo.InteractionResponse{
|
||||||
Type: discordgo.InteractionResponseChannelMessageWithSource,
|
Type: discordgo.InteractionResponseChannelMessageWithSource,
|
||||||
Data: &discordgo.InteractionResponseData{
|
Data: &discordgo.InteractionResponseData{
|
||||||
Content: "🔍 Fetching available formats...",
|
Content: fmt.Sprintf("%s Fetching available formats...", loading_emoji),
|
||||||
Flags: discordgo.MessageFlagsEphemeral,
|
Flags: discordgo.MessageFlagsEphemeral,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|||||||
+6
-1
@@ -1,6 +1,11 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import "fmt"
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
)
|
||||||
|
|
||||||
|
var loading_emoji = os.Getenv("LOADING_EMOJI")
|
||||||
|
|
||||||
// Helper function to create string pointer
|
// Helper function to create string pointer
|
||||||
func ptr(s string) *string {
|
func ptr(s string) *string {
|
||||||
|
|||||||
@@ -154,7 +154,6 @@ func DownloadVideo(out_dir, temp_dir, url string, opts DownloadOptions, progress
|
|||||||
SetExecutable(ytdlpBinary).
|
SetExecutable(ytdlpBinary).
|
||||||
Paths(homePath).
|
Paths(homePath).
|
||||||
Paths(tempPath).
|
Paths(tempPath).
|
||||||
RecodeVideo("mp4").
|
|
||||||
ProgressFunc(100*time.Millisecond, func(prog ytdlp.ProgressUpdate) {
|
ProgressFunc(100*time.Millisecond, func(prog ytdlp.ProgressUpdate) {
|
||||||
if prog.Status == ytdlp.ProgressStatusFinished ||
|
if prog.Status == ytdlp.ProgressStatusFinished ||
|
||||||
prog.Status == ytdlp.ProgressStatusStarting ||
|
prog.Status == ytdlp.ProgressStatusStarting ||
|
||||||
|
|||||||
@@ -46,8 +46,11 @@
|
|||||||
.venv/bin/pip install -r ./app/requirements.txt
|
.venv/bin/pip install -r ./app/requirements.txt
|
||||||
source .venv/bin/activate
|
source .venv/bin/activate
|
||||||
fi
|
fi
|
||||||
|
<<<<<<< HEAD
|
||||||
|
=======
|
||||||
|
|
||||||
export YTDLP_BIN=${pkgs.lib.getExe pkgs.yt-dlp}
|
export YTDLP_BIN=${pkgs.lib.getExe pkgs.yt-dlp}
|
||||||
|
>>>>>>> v1-refactor
|
||||||
'';
|
'';
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|||||||
+17
-2
@@ -1,3 +1,18 @@
|
|||||||
{
|
{
|
||||||
"$schema": "https://docs.renovatebot.com/renovate-schema.json"
|
"$schema": "https://docs.renovatebot.com/renovate-schema.json",
|
||||||
}
|
"customManagers": [
|
||||||
|
{
|
||||||
|
"description": "Update yt-dlp version in Dockerfiles",
|
||||||
|
"customType": "regex",
|
||||||
|
"managerFilePatterns": [
|
||||||
|
"/^Dockerfile$/"
|
||||||
|
],
|
||||||
|
"matchStrings": [
|
||||||
|
"ARG YT_DLP_VERSION=\"(?<currentValue>.*?)\""
|
||||||
|
],
|
||||||
|
"datasourceTemplate": "github-tags",
|
||||||
|
"depNameTemplate": "yt-dlp/yt-dlp",
|
||||||
|
"versioningTemplate": "regex:^(?<major>\\d{4})\\.(?<minor>\\d{2})\\.(?<patch>\\d{2})$"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user