add ffmpeg to bright release stage

This commit is contained in:
CJ_Clippy 2025-03-05 03:03:52 -08:00
parent 6c1869bd72
commit 1d87177a00
10 changed files with 129 additions and 159 deletions
.gitea/workflows
README.md
apps/bright
Dockerfile
config
lib
bright
bright_web/live/profile
config
devbox.jsondocker-compose.yml
playbooks/opentofu

@ -26,59 +26,59 @@ jobs:
username: ${{ secrets.DOCKER_USERNAME }} username: ${{ secrets.DOCKER_USERNAME }}
password: ${{ secrets.DOCKER_PASSWORD }} password: ${{ secrets.DOCKER_PASSWORD }}
- name: Build futureporn/aquatic # - name: Build futureporn/aquatic
uses: docker/build-push-action@v6
with:
context: ./apps/aquatic
file: ./apps/aquatic/docker/aquatic_udp_futureporn.Dockerfile
push: true
tags: gitea.futureporn.net/futureporn/aquatic:latest
- name: Build futureporn/tracker-helper
uses: docker/build-push-action@v6
with:
context: ./services/tracker-helper
push: true
tags: gitea.futureporn.net/futureporn/tracker-helper:latest
labels: |
org.opencontainers.image.description=Custom Aquatic helper service, adding info_hash accesslist operations via HTTP
org.opencontainers.image.title=tracker-helper
org.opencontainers.image.licenses=unlicense
org.opencontainers.image.source=https://gitea.futureporn.net/futureporn/fp
org.opencontainers.image.url=https://gitea.futureporn.net/futureporn/-/packages/container/tracker-helper
secrets: |
TRACKER_HELPER_USERNAME=${{ secrets.TRACKER_HELPER_USERNAME }}
TRACKER_HELPER_PASSWORD=${{ secrets.TRACKER_HELPER_PASSWORD }}
- name: Build futureporn/tracker
uses: docker/build-push-action@v6
with:
context: ./apps/tracker
push: true
tags: gitea.futureporn.net/futureporn/tracker:latest
labels: |
org.opencontainers.image.description=Aquatic tracker with custom helper service, adding info_hash accesslist operations via HTTP
org.opencontainers.image.title=tracker
org.opencontainers.image.licenses=unlicense
org.opencontainers.image.source=https://gitea.futureporn.net/futureporn/fp
org.opencontainers.image.url=https://gitea.futureporn.net/futureporn/-/packages/container/tracker
secrets: |
TRACKER_HELPER_USERNAME=${{ secrets.TRACKER_HELPER_USERNAME }}
TRACKER_HELPER_PASSWORD=${{ secrets.TRACKER_HELPER_PASSWORD }}
# - name: Build futureporn/bright
# uses: docker/build-push-action@v6 # uses: docker/build-push-action@v6
# with: # with:
# context: ./apps/bright # context: ./apps/aquatic
# file: ./apps/aquatic/docker/aquatic_udp_futureporn.Dockerfile
# push: true # push: true
# tags: gitea.futureporn.net/futureporn/bright:latest # tags: gitea.futureporn.net/futureporn/aquatic:latest
# build-args: |
# MIX_ENV=prod # - name: Build futureporn/tracker-helper
# uses: docker/build-push-action@v6
# with:
# context: ./services/tracker-helper
# push: true
# tags: gitea.futureporn.net/futureporn/tracker-helper:latest
# labels: | # labels: |
# org.opencontainers.image.description=The Galaxy's Best VTuber hentai site # org.opencontainers.image.description=Custom Aquatic helper service, adding info_hash accesslist operations via HTTP
# org.opencontainers.image.title=bright # org.opencontainers.image.title=tracker-helper
# org.opencontainers.image.created={{commit_date 'YYYY-MM-DDTHH:mm:ss.SSS[Z]'}}
# org.opencontainers.image.version={{version}}
# org.opencontainers.image.licenses=unlicense # org.opencontainers.image.licenses=unlicense
# org.opencontainers.image.source=https://gitea.futureporn.net/futureporn/fp # org.opencontainers.image.source=https://gitea.futureporn.net/futureporn/fp
# org.opencontainers.image.url=https://gitea.futureporn.net/futureporn/-/packages/container/bright # org.opencontainers.image.url=https://gitea.futureporn.net/futureporn/-/packages/container/tracker-helper
# secrets: |
# TRACKER_HELPER_USERNAME=${{ secrets.TRACKER_HELPER_USERNAME }}
# TRACKER_HELPER_PASSWORD=${{ secrets.TRACKER_HELPER_PASSWORD }}
# - name: Build futureporn/tracker
# uses: docker/build-push-action@v6
# with:
# context: ./apps/tracker
# push: true
# tags: gitea.futureporn.net/futureporn/tracker:latest
# labels: |
# org.opencontainers.image.description=Aquatic tracker with custom helper service, adding info_hash accesslist operations via HTTP
# org.opencontainers.image.title=tracker
# org.opencontainers.image.licenses=unlicense
# org.opencontainers.image.source=https://gitea.futureporn.net/futureporn/fp
# org.opencontainers.image.url=https://gitea.futureporn.net/futureporn/-/packages/container/tracker
# secrets: |
# TRACKER_HELPER_USERNAME=${{ secrets.TRACKER_HELPER_USERNAME }}
# TRACKER_HELPER_PASSWORD=${{ secrets.TRACKER_HELPER_PASSWORD }}
- name: Build futureporn/bright
uses: docker/build-push-action@v6
with:
context: ./apps/bright
push: true
tags: gitea.futureporn.net/futureporn/bright:latest
build-args: |
MIX_ENV=prod
labels: |
org.opencontainers.image.description=The Galaxy's Best VTuber hentai site
org.opencontainers.image.title=bright
org.opencontainers.image.created={{commit_date 'YYYY-MM-DDTHH:mm:ss.SSS[Z]'}}
org.opencontainers.image.version={{version}}
org.opencontainers.image.licenses=unlicense
org.opencontainers.image.source=https://gitea.futureporn.net/futureporn/fp
org.opencontainers.image.url=https://gitea.futureporn.net/futureporn/-/packages/container/bright

@ -21,7 +21,8 @@ The main gist is as follows.
devbox install devbox install
3. Run `docker compose up --watch` 3. Run database and other accessories with `docker compose up --watch`
4. In another terminal, run the phoenix "bright" app with `devbox run bright:dev`
4. Visit http://localhost:4000 4. Visit http://localhost:4000
If all went well, editing source code will automatically affect the website running in your browser. If all went well, editing source code will automatically affect the website running in your browser.

@ -27,6 +27,7 @@ RUN apt-get update -y && apt-get install -y build-essential git inotify-tools ff
&& apt-get clean && rm -f /var/lib/apt/lists/*_* && apt-get clean && rm -f /var/lib/apt/lists/*_*
# prepare build dir # prepare build dir
WORKDIR /app WORKDIR /app
@ -93,11 +94,17 @@ CMD [ "mix", "phx.server" ]
# start a new build stage so that the final image will only contain # start a new build stage so that the final image will only contain
# the compiled release and other runtime necessities # the compiled release and other runtime necessities
FROM ${RUNNER_IMAGE} AS prod FROM ${RUNNER_IMAGE} AS prod
RUN mkdir -p ~/.config/futureporn RUN mkdir -p /mnt/vfs/futureporn
RUN apt-get update -y && \ RUN apt-get update -y \
apt-get install -y libstdc++6 openssl libncurses5 locales ca-certificates \ && apt-get install -y libstdc++6 openssl libncurses5 locales inotify-tools ffmpeg python3 python3-pip ca-certificates \
&& apt-get clean && rm -f /var/lib/apt/lists/*_* && pip install torrentfile \
&& apt-get clean && rm -f /var/lib/apt/lists/*_* \
&& groupadd bright \
&& useradd bright \
--gid bright \
--shell /bin/sh \
--create-home
# Set the locale # Set the locale
RUN sed -i '/en_US.UTF-8/s/^# //g' /etc/locale.gen && locale-gen RUN sed -i '/en_US.UTF-8/s/^# //g' /etc/locale.gen && locale-gen
@ -105,7 +112,7 @@ RUN sed -i '/en_US.UTF-8/s/^# //g' /etc/locale.gen && locale-gen
ENV LANG=en_US.UTF-8 LANGUAGE=en_US:en LC_ALL=en_US.UTF-8 ENV LANG=en_US.UTF-8 LANGUAGE=en_US:en LC_ALL=en_US.UTF-8
WORKDIR "/app" WORKDIR "/app"
RUN chown nobody /app RUN chown bright /app
# set runner ENV # set runner ENV
ARG MIX_ENV=prod ARG MIX_ENV=prod
@ -113,13 +120,12 @@ ENV MIX_ENV=$MIX_ENV
RUN echo MIX_ENV=$MIX_ENV RUN echo MIX_ENV=$MIX_ENV
# Only copy the final release from the build stage # Only copy the final release from the build stage
COPY --from=builder --chown=nobody:root /app/_build/${MIX_ENV}/rel/bright ./ COPY --from=builder --chown=bright:root /app/_build/${MIX_ENV}/rel/bright ./
USER nobody USER bright
# If using an environment that doesn't automatically reap zombie processes, it is # If using an environment that doesn't automatically reap zombie processes, it is
# advised to add an init process such as tini via `apt-get install` # advised to add an init process such as tini via `apt-get install`
# above and adding an entrypoint. See https://github.com/krallin/tini for details # above and adding an entrypoint. See https://github.com/krallin/tini for details
# ENTRYPOINT ["/tini", "--"] # ENTRYPOINT ["/tini", "--"]
CMD ["/app/bin/server"] CMD ["/app/bin/server"]

@ -94,12 +94,19 @@ if config_env() == :prod do
# We need to stop the program from running if OAuth client IDs and client secrets are not present in env. # We need to stop the program from running if OAuth client IDs and client secrets are not present in env.
# We also do this in config.exs, but we wait to raise until here otherwise mix wouldn't be able to run ecto migrations # We also do this in config.exs, but we wait to raise until here otherwise mix wouldn't be able to run ecto migrations
# System.get_env("GITHUB_CLIENT_ID") || raise("environment variable GITHUB_CLIENT_ID is missing.") System.get_env("PATREON_CLIENT_ID") ||
# System.get_env("GITHUB_CLIENT_SECRET") || raise("environment variable GITHUB_CLIENT_SECRET is missing.") raise("environment variable PATREON_CLIENT_ID is missing.")
# config :ueberauth, Ueberauth.Strategy.Patreon.OAuth, System.get_env("PATREON_CLIENT_SECRET") ||
# client_id: {:system, "PATREON_CLIENT_ID"}, raise("environment variable PATREON_CLIENT_SECRET is missing.")
# client_secret: {:system, "PATREON_CLIENT_SECRET"}
System.get_env("PATREON_REDIRECT_URI") ||
raise("environment variable PATREON_REDIRECT_URI is missing.")
config :ueberauth, Ueberauth.Strategy.Patreon.OAuth,
client_id: System.get_env("PATREON_CLIENT_ID"),
client_secret: System.get_env("PATREON_CLIENT_SECRET"),
redirect_uri: System.get_env("PATREON_REDIRECT_URI")
# config :ueberauth, Ueberauth.Strategy.Github.OAuth, # config :ueberauth, Ueberauth.Strategy.Github.OAuth,
# client_id: {:system, "GITHUB_CLIENT_ID"}, # client_id: {:system, "GITHUB_CLIENT_ID"},

@ -3,7 +3,8 @@ defmodule Bright.Cache do
A simple caching module that saves files to the `/tmp` directory. A simple caching module that saves files to the `/tmp` directory.
""" """
@cache_dir Path.join(System.user_home!(), ".cache/futureporn") # we use Vultr File System to share cache among all Phoenix instances
@cache_dir "/mnt/vfs/futureporn"
require Logger require Logger

@ -61,7 +61,7 @@ defmodule BrightWeb.ProfileLive do
<section class="hero"> <section class="hero">
<div class="hero-body"> <div class="hero-body">
<p class="title">Care to upgrade your Patron Tier?</p> <p class="title">Patron Tiers & Perks</p>
</div> </div>
</section> </section>
@ -90,13 +90,13 @@ defmodule BrightWeb.ProfileLive do
</tr> </tr>
<tr> <tr>
<td>API</td> <td>API</td>
<td></td> <td></td>
<td></td> <td></td>
<td></td> <td></td>
</tr> </tr>
<tr> <tr>
<td>Patron List</td> <td>Patron List</td>
<td></td> <td></td>
<td></td> <td></td>
<td></td> <td></td>
</tr> </tr>

@ -7,20 +7,13 @@ image: futureporn/bright
# Deploy to these servers. # Deploy to these servers.
servers: servers:
web: web:
- 45.76.57.101 - 66.42.125.79
# Enable SSL auto certification via Let's Encrypt and allow for multiple apps on a single web server. # Enable SSL auto certification via Let's Encrypt and allow for multiple apps on a single web server.
# Remove this section when using multiple web servers and ensure you terminate SSL at your load balancer. # Remove this section when using multiple web servers and ensure you terminate SSL at your load balancer.
# #
# Note: If using Cloudflare, set encryption mode in SSL/TLS setting to "Full" to enable CF-to-app encryption. # Note: If using Cloudflare, set encryption mode in SSL/TLS setting to "Full" to enable CF-to-app encryption.
proxy: proxy: false
ssl: true
app_port: 4000
host: bright.futureporn.net
healthcheck:
path: /api/health
interval: 2
timeout: 2
# how long to wait for new containers to boot # how long to wait for new containers to boot
deploy_timeout: 30 deploy_timeout: 30
@ -41,7 +34,9 @@ builder:
args: args:
MIX_ENV: prod MIX_ENV: prod
arch: amd64 arch: amd64
dockerfile: ./dockerfiles/bright.dockerfile dockerfile: ./apps/bright/Dockerfile
context: ./apps/bright
# Pass in additional build args needed for your Dockerfile. # Pass in additional build args needed for your Dockerfile.
# args: # args:
# RUBY_VERSION: <%= File.read('.ruby-version').strip %> # RUBY_VERSION: <%= File.read('.ruby-version').strip %>
@ -49,16 +44,22 @@ builder:
env: env:
clear: clear:
PORT: 4000 PORT: 4000
DATABASE_HOST: futureporn-db DATABASE_HOST: 10.2.128.4
MIX_ENV: prod MIX_ENV: dev
SUPERSTREAMER_URL: http://superstreamer-api
PUBLIC_S3_ENDPOINT: https://futureporn-b2.b-cdn.net PUBLIC_S3_ENDPOINT: https://futureporn-b2.b-cdn.net
PATREON_REDIRECT_URI: https://bright.futureporn.net/auth/patreon/callback
SITE_URL: https://bright.futureporn.net
PHX_HOST: bright.futureporn.net
AWS_BUCKET: futureporn
AWS_REGION: us-west-000
AWS_HOST: s3.us-west-000.backblazeb2.com
secret: secret:
- DATABASE_URL - DATABASE_URL
- SECRET_KEY_BASE - SECRET_KEY_BASE
- SUPERSTREAMER_AUTH_TOKEN - PATREON_CLIENT_SECRET
- GITHUB_CLIENT_SECRET - PATREON_CLIENT_ID
- GITHUB_CLIENT_ID - AWS_ACCESS_KEY_ID
- AWS_SECRET_ACCESS_KEY
# Aliases are triggered with "bin/kamal <alias>". You can overwrite arguments on invocation: # Aliases are triggered with "bin/kamal <alias>". You can overwrite arguments on invocation:
# "bin/kamal logs -r job" will tail logs from the first server in the job section. # "bin/kamal logs -r job" will tail logs from the first server in the job section.
@ -68,11 +69,12 @@ env:
ssh: ssh:
keys: ["~/.ssh/futureporn"] keys: ["~/.ssh/futureporn"]
keys_only: true
# Use a persistent storage volume. # Use a persistent storage volume.
# #
# volumes: volumes:
# - "app_storage:/app/storage" - "app_cache:/mnt/vfs/futureporn"
# Bridge fingerprinted assets, like JS and CSS, between versions to avoid # Bridge fingerprinted assets, like JS and CSS, between versions to avoid
# hitting 404 on in-flight requests. Combines all files from new and old # hitting 404 on in-flight requests. Combines all files from new and old
@ -91,7 +93,7 @@ ssh:
accessories: accessories:
tracker: tracker:
image: gitea.futureporn.net/futureporn/tracker:latest image: gitea.futureporn.net/futureporn/tracker:latest
host: 45.76.57.101 host: 66.42.125.79
port: "0.0.0.0:5063:5063" port: "0.0.0.0:5063:5063"
env: env:
clear: clear:
@ -102,37 +104,15 @@ accessories:
proxy: proxy:
ssl: true ssl: true
forward_headers: false forward_headers: false
# note: tracker also uses port 6969/udp and 9000/tcp, but the api at 5063/tcp is what we specify here. # @todo @blocking https://github.com/basecamp/kamal-proxy/issues/48 # note: tracker also uses port 6969/udp and 9000/tcp, but the api at 5063/tcp is what we specify here. # @todo @blocking https://github.com/basecamp/kamal-proxy/issues/48 app_port: 5063
app_port: 5063
host: tracker.futureporn.net host: tracker.futureporn.net
healthcheck: healthcheck:
path: /health path: /health
qbittorrent:
image: lscr.io/linuxserver/qbittorrent:latest
host: 45.76.57.101
port: "127.0.0.1:8080:8080"
env:
clear:
PUID: "1000"
PGID: "1000"
TZ: "Etc/UTC"
WEBUI_PORT: "8080"
TORRENTING_PORT: "6881"
proxy:
ssl: true
forward_headers: true
app_port: 8080
host: qbittorrent.futureporn.net
healthcheck:
path: /
volumes:
- /root/.cache/futureporn:/root/.cache/futureporn
db: db:
image: postgres:15 image: postgres:15
host: 45.76.57.101 host: 45.76.228.113
port: "127.0.0.1:5432:5432" port: "0.0.0.0:5432:5432"
env: env:
clear: clear:
POSTGRES_USER: postgres POSTGRES_USER: postgres
@ -141,28 +121,3 @@ accessories:
- POSTGRES_PASSWORD - POSTGRES_PASSWORD
directories: directories:
- pg_data:/var/lib/postgresql/data - pg_data:/var/lib/postgresql/data
redis:
image: valkey/valkey:8
host: 45.76.57.101
port: 6379
directories:
- data:/data
pgadmin:
image: dpage/pgadmin4
host: 45.76.57.101
port: "127.0.0.1:5050:5050"
env:
clear:
PGADMIN_LISTEN_PORT: "5050"
secret:
- PGADMIN_DEFAULT_EMAIL
- PGADMIN_DEFAULT_PASSWORD
proxy:
ssl: true
forward_headers: true
app_port: 5050
host: pgadmin.futureporn.net
healthcheck:
path: /login

@ -20,9 +20,9 @@
], ],
"env": { "env": {
"DEVBOX_COREPACK_ENABLED": "true", "DEVBOX_COREPACK_ENABLED": "true",
"ENV": "development", "ENV": "development",
"KUBECONFIG": "$HOME/.kube/futureporn.yaml", "KUBECONFIG": "$HOME/.kube/futureporn.yaml",
"VENV_DIR": ".venv" "VENV_DIR": ".venv"
}, },
"shell": { "shell": {
"init_hook": [ "init_hook": [
@ -31,19 +31,19 @@
"pip install -r requirements.txt" "pip install -r requirements.txt"
], ],
"scripts": { "scripts": {
"tunnel": "dotenvx run -f ./.kamal/secrets.development -- chisel client bright.fp.sbtp.xyz:9090 R:4000", "tunnel": "dotenvx run -f ./.kamal/secrets.development -- chisel client bright.fp.sbtp.xyz:9090 R:4000",
"backup": "docker exec -t postgres_db pg_dumpall -c -U postgres > ./backups/dev_`date +%Y-%m-%d_%H_%M_%S`.sql", "backup": "docker exec -t postgres_db pg_dumpall -c -U postgres > ./backups/dev_`date +%Y-%m-%d_%H_%M_%S`.sql",
"act": "dotenvx run -f ./.kamal/secrets.testing -- act -W ./.gitea/workflows --secret-file .kamal/secrets.development", "act": "dotenvx run -f ./.kamal/secrets.testing -- act -W ./.gitea/workflows --secret-file .kamal/secrets.development",
"act:builder": "dotenvx run -f ./.kamal/secrets.testing -- act -W ./.gitea/workflows/builder.yaml --secret-file .kamal/secrets.testing --var-file .kamal/secrets.testing --insecure-secrets", "act:builder": "dotenvx run -f ./.kamal/secrets.testing -- act -W ./.gitea/workflows/builder.yaml --secret-file .kamal/secrets.testing --var-file .kamal/secrets.testing --insecure-secrets",
"act:tests": "dotenvx run -f ./.kamal/secrets.testing -- act -W ./.gitea/workflows/tests.yaml --secret-file .kamal/secrets.testing --var-file .kamal/secrets.testing --insecure-secrets", "act:tests": "dotenvx run -f ./.kamal/secrets.testing -- act -W ./.gitea/workflows/tests.yaml --secret-file .kamal/secrets.testing --var-file .kamal/secrets.testing --insecure-secrets",
"bright:compile:watch": "cd ./apps/bright && find . -type f -name \"*.ex\" -o -name \"*.exs\" | entr -r mix compile --warnings-as-errors", "bright:compile:watch": "cd ./apps/bright && find . -type f -name \"*.ex\" -o -name \"*.exs\" | entr -r mix compile --warnings-as-errors",
"bright:compile:watch2": "cd ./apps/bright && pnpx chokidar-cli \"**/*\" -i \"deps/**\" -i \"_build/**\" -c \"mix compile --warnings-as-errors\"", "bright:compile:watch2": "cd ./apps/bright && pnpx chokidar-cli \"**/*\" -i \"deps/**\" -i \"_build/**\" -c \"mix compile --warnings-as-errors\"",
"bright:dev": "cd ./apps/bright && dotenvx run -f ../../.kamal/secrets.development -e MIX_ENV=dev -- mix phx.server", "bright:dev": "cd ./apps/bright && dotenvx run -f ../../.kamal/secrets.development -e MIX_ENV=dev -- mix phx.server",
"bright:test:unit:watch": "cd ./apps/bright && pnpx chokidar-cli '**/*' -i \"deps/**\" -i '_build/**' -c 'mix test --only=unit'", "bright:test:unit:watch": "cd ./apps/bright && pnpx chokidar-cli '**/*' -i \"deps/**\" -i '_build/**' -c 'mix test --only=unit'",
"bright:act": "cd ./apps/bright && act --env MIX_ENV=test -W ./.gitea/workflows/tests.yaml --secret-file .kamal/secrets.development", "bright:act": "cd ./apps/bright && act --env MIX_ENV=test -W ./.gitea/workflows/tests.yaml --secret-file .kamal/secrets.development",
"test": "act -W ./.gitea/workflows/tests.yaml --secret-file .kamal/secrets.testing --var-file .kamal/secrets.testing && devbox run beep || devbox run boop", "test": "act -W ./.gitea/workflows/tests.yaml --secret-file .kamal/secrets.testing --var-file .kamal/secrets.testing && devbox run beep || devbox run boop",
"beep": "ffplay -nodisp -loglevel quiet -autoexit ./apps/beep/beep2.wav", "beep": "ffplay -nodisp -loglevel quiet -autoexit ./apps/beep/beep2.wav",
"boop": "ffplay -nodisp -loglevel quiet -autoexit ./apps/beep/beep1.wav" "boop": "ffplay -nodisp -loglevel quiet -autoexit ./apps/beep/beep1.wav"
} }
} }
} }

@ -3,11 +3,10 @@ services:
aquatic: aquatic:
build: build:
context: ./apps/aquatic context: ./apps/aquatic
dockerfile: docker/aquatic_udp_futureporn.Dockerfile
ports: ports:
- "3003:3003/udp" - "6969:6969/udp"
- "9000:9000/tcp" - "9000:9000/tcp"
volumes:
- aquatic
# bright: # bright:
@ -87,12 +86,12 @@ services:
- db - db
environment: environment:
PGADMIN_LISTEN_PORT: "5050" PGADMIN_LISTEN_PORT: "5050"
PGADMIN_DISABLE_POSTFIX: "1"
PGADMIN_DEFAULT_EMAIL: cj@futureporn.net
PGADMIN_DEFAULT_PASSWORD: "password"
env_file: env_file:
- .kamal/secrets.development - .kamal/secrets.development
volumes: volumes:
pg_data: pg_data:
redis_data:
cache: cache:
opentracker-etc:
aquatic:

@ -61,6 +61,7 @@ resource "vultr_instance" "capture_vps" {
- chmod +x ~/.local/bin/thumbnail-generator.sh - chmod +x ~/.local/bin/thumbnail-generator.sh
- curl -fsSL https://github.com/Backblaze/B2_Command_Line_Tool/releases/download/v4.3.1/b2-linux > ~/.local/bin/b2 - curl -fsSL https://github.com/Backblaze/B2_Command_Line_Tool/releases/download/v4.3.1/b2-linux > ~/.local/bin/b2
- chmod +x ~/.local/bin/b2 - chmod +x ~/.local/bin/b2
- export DIR=/usr/local/bin; curl https://raw.githubusercontent.com/jesseduffield/lazydocker/master/scripts/install_update_linux.sh | bash
- curl -fsSL https://dist.ipfs.tech/kubo/v0.33.2/kubo_v0.33.2_linux-amd64.tar.gz > ~/kubo_v0.33.2_linux-amd64.tar.gz - curl -fsSL https://dist.ipfs.tech/kubo/v0.33.2/kubo_v0.33.2_linux-amd64.tar.gz > ~/kubo_v0.33.2_linux-amd64.tar.gz
- tar xvzf ~/kubo_v0.33.2_linux-amd64.tar.gz - tar xvzf ~/kubo_v0.33.2_linux-amd64.tar.gz
- ~/kubo/install.sh - ~/kubo/install.sh