Skip Navigation

Share a script/alias you use a lot

A while ago I made a tiny function in my ~/.zshrc to download a video from the link in my clipboard. I use this nearly every day to share videos with people without forcing them to watch it on whatever site I found it. What's a script/alias that you use a lot?

 
    
# Download clipboard to tmp with yt-dlp
tmpv() {
  cd /tmp/ && yt-dlp "$(wl-paste)"
}

  
150 comments
  • One of favorites cds to the root of a project directory from a subdirectory,

     bash
        
    # Changes to top-level directory of git repository.
    alias gtop="cd \$(git rev-parse --show-toplevel)"
    
      
  •  
        
    $ which diffuc
    diffuc: aliased to diff -uw --color=always
    
    
      
     
        
    $ which grepnir
    grepnir: aliased to grep -niIr
    
    
      
     
        
    $ cat `which ts`
    #!/bin/bash
    
    if [ "$#" -lt 1 ]; then
                    tmux list-sessions
                    exit
    fi
    
    if ! tmux attach -t "$1"
    then
                    tmux new-session -s "$1"
    fi
    
      
  • on most of my systems I get tired of constantly lsing after a cd so I combine them:

     
        
    cd(){
        cd $1 && ls
    }
    
    
      

    (excuse if this doesn't work, I am writing this from memory)

    I also wrote a function to access docker commands quicker on my Truenas system. If passed nothing, it enters the docker jailmaker system, else it passes the command to docker running inside the system.

     
        
    docker () {
            if [[ "$1" == "" ]]; then
                    jlmkr shell docker
                    return
            else
                    sudo systemd-run --pipe --machine docker docker "$@"
                    return
            fi
    }
    
    
      

    I have a few similar shortcuts for programs inside jailmaker and long directories that I got sick of typing out.

  • alias qr='qrencode -t ansiutf8'

    This makes qr codes in the terminal.

    needs the qrencode package

    Example usage and output:

     
        
    felix@buttsexmachine:~$ qr lemmy.fish
    █████████████████████████████
    █████████████████████████████
    ████ ▄▄▄▄▄ █▄ ██ █ ▄▄▄▄▄ ████
    ████ █   █ █ █▄▀▄█ █   █ ████
    ████ █▄▄▄█ █▄▄▄███ █▄▄▄█ ████
    ████▄▄▄▄▄▄▄█▄▀ █▄█▄▄▄▄▄▄▄████
    ████▄▄▄ █▀▄▀▄▀ █▀▄▀▀   █ ████
    ████▄ ▀▄▀▄▄ ▀▄▄█ ▄▄▄█▀█ ▄████
    ██████▄███▄█▀█ ▄█▄ █▀█▀▄▄████
    ████ ▄▄▄▄▄ ██ ▀▀▀▀▄   ▀█▀████
    ████ █   █ █▀ ▀▄█▀▀▄▄  ▀█████
    ████ █▄▄▄█ █ ▀█ ▀█▀ █▄▄█▀████
    ████▄▄▄▄▄▄▄█▄▄█▄▄▄███▄▄██████
    █████████████████████████████
    █████████████████████████████
    *___*
      
  • it's somewhat vibe coded but the one i probably use the most is this one to swap between speakers and headset. the device name to look for is just put directly in there, it'd take some adjustment to run it on different machines. this is in my .bashrc:

     
        
    # switch sinks
    toggle_audio() {
      # Find headset sink ID dynamically
      headset_id=$(pactl list sinks short | grep "Plantronics" | awk '{print $1}')
      
      # Find speakers sink ID dynamically
      speakers_id=$(pactl list sinks short | grep "pci-0000_05_00.6" | awk '{print $1}')
      
      # Get current default sink
      current_sink=$(pactl get-default-sink)
      
      # Get current sink ID
      current_id=$(pactl list sinks short | grep "$current_sink" | awk '{print $1}')
      
      # Toggle between the two
      if [ "$current_id" = "$headset_id" ]; then
        pactl set-default-sink "$speakers_id"
        echo "Switched to speakers (Sink $speakers_id)"
      else
        pactl set-default-sink "$headset_id"
        echo "Switched to headset (Sink $headset_id)"
      fi
    }
    
      

    generally i try not to use too many custom things because for work i regularly work on all kinds of different servers and i've just been too lazy to set up some solution to keep it all in sync. someday....

  • I made this one to find binaries in NixOs and other systems

     bash
        
    get_bin_path() {
            paths=${2:-$PATH}
            for dr in $(echo $paths | tr ':' '\n') ; do
                    if [ -f "$dr/$1" ] ; then
                            echo "$dr/$1"
                            return 0
                    fi
            done
            return 1
    }
    
      

    Then I made this one to, if I have a shell o opened inside neovim it will tell the neovim process running the shell to open a file on it, instead of starting a new process

     bash
        
    _nvim_con() {
            abs_path=$(readlink --canonicalize "$@" | sed s'| |\\ |'g)
            $(get_bin_path nvim) --server $NVIM --remote-send "<ESC>:edit $abs_path<CR>"
            exit
    }
    
    # start host and open file
    _nvim_srv() {
            $(get_bin_path nvim) --listen $HOME/.cache/nvim/$$-server.pipe $@
    }
    
    if [ -n "$NVIM" ] ; then
            export EDITOR="_nvim_con"
    else
            export EDITOR="_nvim_srv"
    fi
    
      

    Lastly this bit: which if it detects a file and a line number split by a : it will open the file and jump to the line

     bash
        
    _open() {
            path_parts=$(readlink --canonicalize "$@" | sed s'| |\\ |'g | sed 's/:/\t/' )
            file=$(echo "$path_parts" | awk ' { print $1 }' )
            line=$(echo "$path_parts" | awk ' { print $2 }' )
    
            if [ -n "$line" ] ; then
                    # has line number
                    if [ -n "$NVIM" ] ; then
                            $(get_bin_path nvim) --server $NVIM --remote-send "<ESC>:edit $file<CR>:+$line<CR>"
                            exit
                    else
                            $(get_bin_path nvim) --listen $HOME/.cache/nvim/$$-server.pipe $file "+:$line"
                    fi
            else
                    $EDITOR $file
            fi
    }
    
    alias nvim="_open"
    
      

    all of my bash config is here

  • I've only used aliases twice so far. The first was to replace yt-dlp with a newer version because the version that comes pre-installed in Linux Mint is too outdated to download videos from YouTube. The second was because I needed something called "Nuget". I don't remember exactly what Nuget is but I think it was a dependency for some application I tried several months ago.

     
        
    alias yt-dlp='/home/j/yt-dlp/yt-dlp'
    alias nuget="mono /usr/local/bin/nuget.exe"
    
      
  • Well, my full functions.sh won't fit in a comment, so here's 2 of my more unique functions that makes life a little easier when contributing to busy OSS projects:

     
        
    # Git fork sync functions
    # Assumes standard convention: origin = your fork, upstream = original repo
    ## Sync fork with upstream before starting work
    gss() {
            # Safety checks
            if ! git rev-parse --git-dir >/dev/null 2>&1; then
                    echo "❌ Not in a git repository"
                    return 1
            fi
    
            # Check if we're in a git operation state
            local git_dir=$(git rev-parse --git-dir)
            if [[ -f "$git_dir/rebase-merge/interactive" ]] || [[ -d "$git_dir/rebase-apply" ]] || [[ -f "$git_dir/MERGE_HEAD" ]]; then
                    echo "❌ Git operation in progress. Complete or abort current rebase/merge first:"
                    echo "   git rebase --continue  (after resolving conflicts)"
                    echo "   git rebase --abort     (to cancel rebase)"
                    echo "   git merge --abort      (to cancel merge)"
                    return 1
            fi
    
            # Check for uncommitted changes
            if ! git diff-index --quiet HEAD -- 2>/dev/null; then
                    echo "❌ You have uncommitted changes. Commit or stash them first:"
                    git status --porcelain
                    echo ""
                    echo "💡 Quick fix: git add . && git commit -m 'WIP' or git stash"
                    return 1
            fi
    
            # Check for required remotes
            if ! git remote get-url upstream >/dev/null 2>&1; then
                    echo "❌ No 'upstream' remote found. Add it first:"
                    echo "   git remote add upstream <upstream-repo-url>"
                    return 1
            fi
    
            if ! git remote get-url origin >/dev/null 2>&1; then
                    echo "❌ No 'origin' remote found. Add it first:"
                    echo "   git remote add origin <your-fork-url>"
                    return 1
            fi
    
            local current_branch=$(git branch --show-current)
    
            # Ensure we have a main branch locally
            if ! git show-ref --verify --quiet refs/heads/main; then
                    echo "❌ No local 'main' branch found. Create it first:"
                    echo "   git checkout -b main upstream/main"
                    return 1
            fi
    
            echo "🔄 Syncing fork with upstream..."
            echo "   Current branch: $current_branch"
    
            # Fetch with error handling
            if ! git fetch upstream; then
                    echo "❌ Failed to fetch from upstream. Check network connection and remote URL."
                    return 1
            fi
    
            echo "📌 Updating local main..."
            if ! git checkout main; then
                    echo "❌ Failed to checkout main branch"
                    return 1
            fi
    
            if ! git reset --hard upstream/main; then
                    echo "❌ Failed to reset main to upstream/main"
                    return 1
            fi
    
            echo "⬆️  Pushing updated main to fork..."
            if ! git push origin main; then
                    echo "❌ Failed to push main to origin. Check push permissions."
                    return 1
            fi
    
            echo "🔀 Rebasing feature branch on updated main..."
            if ! git checkout "$current_branch"; then
                    echo "❌ Failed to checkout $current_branch"
                    return 1
            fi
    
            if ! git rebase main; then
                    echo "❌ Rebase failed due to conflicts. Resolve them and continue:"
                    echo "   1. Edit conflicted files"
                    echo "   2. git add <resolved-files>"
                    echo "   3. git rebase --continue"
                    echo "   Or: git rebase --abort to cancel"
                    return 1
            fi
    
            echo "✅ Ready to work on branch: $current_branch"
    }
    
    ## Sync fork and push feature branch
    gsp() {
            # Safety checks
            if ! git rev-parse --git-dir >/dev/null 2>&1; then
                    echo "❌ Not in a git repository"
                    return 1
            fi
    
            local git_dir=$(git rev-parse --git-dir)
            if [[ -f "$git_dir/rebase-merge/interactive" ]] || [[ -d "$git_dir/rebase-apply" ]] || [[ -f "$git_dir/MERGE_HEAD" ]]; then
                    echo "❌ Git operation in progress. Complete or abort first."
                    return 1
            fi
    
            if ! git diff-index --quiet HEAD -- 2>/dev/null; then
                    echo "❌ You have uncommitted changes. Commit or stash them first:"
                    git status --porcelain
                    return 1
            fi
    
            if ! git remote get-url upstream >/dev/null 2>&1; then
                    echo "❌ No 'upstream' remote found"
                    return 1
            fi
    
            if ! git remote get-url origin >/dev/null 2>&1; then
                    echo "❌ No 'origin' remote found"
                    return 1
            fi
    
            local current_branch=$(git branch --show-current)
    
            # Prevent pushing from main
            if [[ "$current_branch" == "main" ]]; then
                    echo "❌ Cannot push from main branch. Switch to your feature branch first:"
                    echo "   git checkout <your-feature-branch>"
                    return 1
            fi
    
            # Show what we're about to do
            echo "⚠️  About to sync and push branch: $current_branch"
            echo "   This will:"
            echo "   • Fetch latest changes from upstream"
            echo "   • Rebase your branch on updated main"
            echo "   • Force-push to your fork (updates PR)"
            echo ""
    
            read -p "Continue? [y/N]: " -n 1 -r
            echo
    
            if [[ ! $REPLY =~ ^[Yy]$ ]]; then
                    echo "❌ Operation cancelled"
                    return 0
            fi
    
            echo "🔄 Final sync with upstream..."
            if ! git fetch upstream; then
                    echo "❌ Failed to fetch from upstream"
                    return 1
            fi
    
            echo "📌 Updating local main..."
            if ! git checkout main; then
                    echo "❌ Failed to checkout main"
                    return 1
            fi
    
            if ! git reset --hard upstream/main; then
                    echo "❌ Failed to reset main"
                    return 1
            fi
    
            if ! git push origin main; then
                    echo "❌ Failed to push main to origin"
                    return 1
            fi
    
            echo "🔀 Rebasing feature branch..."
            if ! git checkout "$current_branch"; then
                    echo "❌ Failed to checkout $current_branch"
                    return 1
            fi
    
            if ! git rebase main; then
                    echo "❌ Rebase failed. Resolve conflicts and try again:"
                    echo "   git add <resolved-files> && git rebase --continue"
                    echo "   Then run 'gsp' again"
                    return 1
            fi
    
            echo "🚀 Pushing feature branch to fork..."
            if ! git push origin "$current_branch" --force-with-lease; then
                    echo "❌ Failed to push to origin. The branch may have been updated."
                    echo "   Run 'git pull origin $current_branch' and try again"
                    return 1
            fi
    
            echo "✅ Feature branch $current_branch successfully pushed to fork"
    }
    
      
  • jmpd(jump directory): fuzzy finds and opens directory with fzf

     bash
        
    # fish shell
    function jmpd
        set _selection $(fzf --walker=dir);
        if test -n "$_selection"
            cd "$_selection";
        end
    end
    
      
  • here we go:

    dedup:

     
            #!/usr/bin/awk -f
        !x[$0]++
    
    
      

    this removes duplicate lines, preserving line order

    iter:

     
            #!/usr/bin/bash
        if [[ "${@}" =~ /$ ]]; then
            xargs -rd '\n' -I {} "${@}"{}
        else
            xargs -rd '\n' -I {} "${@}" {}
        fi
    
    
      

    This executes a command for each line. It can also be used to compare two directories, ie:

     
            du -sh * > sizes; ls | iter du -sh ../kittens/ > sizes2
    
    
      

    fadeout:

     
            #!/bin/bash
        # I use this to fade out layered brown noise that I play at a volume of 130%
        # This takes about 2 minutes to run, and the volume is at zero several seconds before it's done.
        # ################
        # DBUS_SESSION_BUS_ADDRESS is needed so that playerctl can find the dbus to use MPRIS so it can control mpv
        export DBUS_SESSION_BUS_ADDRESS="unix:path=/run/user/1000/bus"
        # ################
        for i in {130..0}
        do
            volume=$(echo "scale=3;$i/100" | bc)
            sleep 2.3
            playerctl --player=mpv volume $volume
        done
    
    
      

    lbn:

     
            #!/bin/bash
        #lbn_pid=$(cat ~/.local/state/lbn.pid)
        if pgrep -fl layered_brown
        then
            pkill -f layered_brown
        else
            export DBUS_SESSION_BUS_ADDRESS="unix:path=/run/user/1000/bus"
            mpv -ao pulse ~/layered_brown_noise.mp3 >>lbn.log 2>&1 &
            sleep 3
            playerctl -p mpv volume 1.3 >>lbn.log 2>&1 &
        fi
    
    
      

    This plays "layered brown noise" by crysknife. It's a great sleep aid.

    here are some aliases:

     
            alias m='mpc random off; mpc clear'
        alias mpcc='ncmpcpp'
        alias thesaurus='dict -d moby-thesaurus'
        alias wtf='dict -d vera'
        alias tvplayer='mpv -fs --geometry=768x1366+1366+0'
      
  • Here is on that I actually don't use, but want to use it in scripts. It is meant to be used by piping it. It's simple branch with user interaction. I don't even know if there is a standard program doing exactly that already.

     bash
        
    # usage: yesno [prompt]
    # example:
    #   yesno && echo yes
    #   yesno Continue? && echo yes || echo no
    yesno() {
        local prompt
        local answer
        if [[ "${#}" -gt 0 ]]; then
            prompt="${*} "
        fi
        read -rp "${prompt}[y/n]: " answer
        case "${answer}" in
        [Yy0]*) return 0 ;;
        [Nn1]*) return 1 ;;
        *) return 2 ;;
        esac
    }
    
      
  • Ooooou I got a couple :3

    This one is just a basic mirror fixing thing cuz sometimes I go a while without updating pacman:

     bash
        
    alias fixpkg='rate-mirrors --protocol https arch | sudo tee /etc/pacman.d/mirrorlist && sudo pacman -Syy'
    
      

    This function I made to create virtual audio sinks so I can route audios via qpw and play earrape into discord calls if I want XD

     bash
        
    create_vsink() {
        local sink_name=${1:-vsink}  # Default sink name is 'vsink' if no input is provided
        local description=${2:-"Virtual Sink"}  # Default description
        pactl load-module module-null-sink sink_name="$sink_name" sink_properties=device.des>
        echo "Virtual sink '$sink_name' created with description '$description'."
    }
    
      

    Simple parser function I made that makes a whole repo using my git key so it's not just locally created I kinda forgot why I made it tbh:

     bash
        
    git_clone() {
        local url="${1#https://}"  # Remove "https://" if present
        git clone "https://$git_key@$url"
    }
     
      

    Awesome mpv function I made that allows for real time pitch+speed shifting via hotkeys and is flexible with extra parameters and shit:

     bash
        
    mpv_pitch() {
        if [[ -z "$1" ]]; then
            echo "Usage: mpv_pitch <file> [mpv-options]"
            return 1
        fi
        local file="$1"
        shift
        mpv --input-conf=/dev/stdin "$file" "$@" <<EOF
    SHIFT+RIGHT add audio-pitch-correction 0; add pitch 0.01; add speed 0.01  # Decrease pit>
    SHIFT+LEFT add audio-pitch-correction 0; add pitch -0.01; add speed -0.01 # Increase pit>
    EOF
    }
    
      

    Automatic audio router for firefox audio streams that uses the aforementioned create_sink function to make a specific sink that I can use carla on to mix and make cool shit out of haha

     bash
        
    firefox_crush() {
        create_vsink CrunchSink "CrunchSink" 
        firefox --name firefox-vc &
    
        (while true; do
            SINK_INPUT_ID=$(pactl list sink-inputs short | grep "firefox" | awk '{print $1}')
            if [[ -n "$SINK_INPUT_ID" ]]; then
                pactl move-sink-input "$SINK_INPUT_ID" CrunchSink
                break
            fi
            sleep 0.25
        done) &
    }
    
      
  • I use Clevis to auto-unlock my encrypted root partition with my TPM; this means when my boot partition is updated (E.G a kernel update), I have to update the PCR register values in my TPM. I do it with my little script /usr/bin/update_pcr:

     
        
    #!/bin/bash
    clevis luks regen -d /dev/nvme1n1p3 -s 1 tpm2
    
    
      

    I run it with sudo and this handles it for me. The only issue is I can't regenerate the binding immediately after the update; I have to reboot, manually enter my password to decrypt the drive, and then do it.

    Now, if I were really fancy and could get it to correctly update the TPM binding immediately after the update, I would have something like an apt package shim with a hook that does it seamlessly. Honestly, I'm surprised that distributions haven't developed robust support for this; the technology is clearly available (I'm using it), but no one seems to have made a user-friendly way for the common user to have TPM encryption in the installer.

  • I wrote this suite of scripts a few years ago and still use them to:

    1. Boot into Ventoy and select a Debian Live environment
    2. Optional: connect a storage device (local partition, USB drive, etc) for persistent storage
    3. Modify cfg/cfg.sh if it's the first time using the tool
    4. Run setup.sh to configure the environment into a familiar/productive state

    The tools are flexible on hardware (more directed toward x64 systems at this time), and I (almost) never have to worry about OS upgrades. Just boot into a newer live OS image once it's ready. They are still a work-in-progress and still have a few customizations that I should abstract for more general use, but it's FOSS in case anyone has merge requests, issues, suggestions, etc.

  •  
        
    git() {
      if [ "$1" = "cd" ]; then
        shift
        cd "./$(command git rev-parse --show-cdup)$*"
      else
        command git "$@"
      fi
    }
    
      

    This lets you run git cd to go to the root of your repo, or git cd foo/bar to go to a path relative to that root. You can't do it as an alias because it's conditional, and you can't do it as a git-cd command because that wouldn't affect the current shell.

  •  
        
    #Create predefined session with multiple tabs/panes (rss, bluetooth, docker...)
    tmux-start 
    
    #Create predefined tmux session with ncmpcpp and ueberzug cover
    music 
    
    #Comfort
    ls = "ls --color=auto"
    please = "sudo !!"
    
    #Quick weather check
    weatherH='curl -s "wttr.in/HomeCity?2QF"' 
    
    #Download Youtube playlist videos in separate directory indexed by video order in playlist -> lectures, etc
    ytPlaylist='yt-dlp -o "%(playlist)s/%(playlist_index)s - %(title)s.%(ext)s"'
    
    #Download whole album  -> podcasts primarily 
    ytAlbum='yt-dlp -x --audio-format mp3 --split-chapters --embed-thumbnail -o "chapter:%(section_title)s.%(ext)s"'
    
    # download video -> extract audio -> show notification
    ytm()
    {
        tsp yt-dlp -x --audio-format mp3 --no-playlist -P "~/Music/downloaded" $1 \
            --exec "dunstify -i folder-download -t 3000 -r 2598 -u normal  %(filepath)q"
    
    }
    
    # Provide list of optional packages which can be manually selected
    pacmanOpts()
    {
    typeset -a os
    for o in `expac -S '%o\n' $1`
    do
      read -p "Install ${o}? " r
      [[ ${r,,} =~ ^y(|e|es)$ ]] && os+=( $o )
    done
    
    sudo pacman -S $1 ${os[@]}
    }
    
    # fkill - kill process
    fkill() {
      pid=$(ps -ef | sed 1d | fzf -m --ansi --color fg:-1,bg:-1,hl:46,fg+:40,bg+:233,hl+:46 --color prompt:166,border:46 --height 40%  --border=sharp --prompt="➤  " --pointer="➤ " --marker="➤ " | awk '{print $2}')
    
      if [ "x$pid" != "x" ]
      then
        kill -${1:-9} $pid
      fi
    }
    
      
  • I replaced rm with trash-put, just in case I realize I need some files that I removed down the line.

     
        
    alias rm='trash-put'
    
      

    Official author don't recommend it due to different semantics. But honestly for my own personal use case its fine for me.


    Also I like to alias xclip:

     
        
    alias clippy='xclip -selection clipboard'
    
    # cat things.txt | clippy
    
      
    • Official author don’t recommend it due to different semantics. But honestly for my own personal use case its fine for me.

      I don't recommend that either. If you get used to that 'rm' doesn't actually remove files and then your alias is missing for whatever reason it'll bite you in the rear at some point. And obviously the same hazard goes with a ton of other commands too.

      • Agree, comes down to risk acceptance honestly.

        I accepted the risk that comes with it. Same with some other aliases on equally hazardous commands.

    • Little tip: In case you need to use rm directly, even with the alias in effect, you can put a backslah in front of the command to use its original meaning: \rm filename

150 comments