linuxmacosbashsed

sed command with -i option failing on Mac, but works on Linux


I've successfully used the following sed command to search/replace text in Linux:

sed -i 's/old_link/new_link/g' *

However, when I try it on my Mac OS X, I get:

"command c expects \ followed by text"

I thought my Mac runs a normal BASH shell. What's up?

EDIT:

According to @High Performance, this is due to Mac sed being of a different (BSD) flavor, so my question would therefore be how do I replicate this command in BSD sed?

EDIT:

Here is an actual example that causes this:

sed -i 's/hello/gbye/g' *

Solution

  • Portable solution below

    Why you get the error

    The -i option (alternatively, --in-place) means that you want files edited in-place, rather than streaming the change to a new place.

    Modifying a file in-place suggests a need for a backup file - and so a user-specified extension is expected after -i, but the parsing of the extension argument is handled differently under GNU sed & Mac (BSD) sed:

    So GNU & Mac will interpret this differently:

    sed -i 's/hello/bye/g' just_a_file.txt
    
    # This still create a `my_file.txt-e` backup on macOS Sonoma (14.5)
    # and a `my_file.txt''` on Linux
    sed -i'' -e 's/hello/bye/g' my_file.txt
    

    Placing the extension immediately after the -i (eg -i'' or -i'.bak', without a space) is what GNU sed expects, but macOS expect a space after -i (eg -i '' or -i '.bak').

    and is now accepted by Mac (BSD) sed too, though it wasn't tolerated by earlier versions (eg with Mac OS X v10.6, a space was required after -i, eg -i '.bak').

    The -e parameter allows us to be explicit about where we're declaring the edit command.

    Until Mac OS was updated in 2013, there wasn't

    Still there isn't any portable command across GNU and Mac (BSD), as these variants all failed (with an error or unexpected backup files):

    Portable solution

    You have few options to achieve the same result on Linux and macOS, e.g.:

    1. Use Perl: perl -i -pe's/old_link/new_link/g' *.

    2. Use gnu-sed on macOS (Install using Homebrew)

    # Install 'gnu-sed' on macOS using Homebrew
    brew install gnu-sed
    # Use 'gsed' instead of 'sed' on macOS.
    gsed -i'' -e 's/hello/bye/g' my_file.txt
    

    Note: On macOS, you could add the bin path of gnu-sed containing the sed command to the PATH environment variable in your shell configuration file (.zshrc).
    It is best not to do this, since there may be scripts that rely on the macOS built-in version.

    You can add an alias for gsed as sed using alias sed=gsed (replacing macOS sed with GNU sed) in your ~/.zshrc. This should allow you to use sed "linux-stile" in your shell and will have no effects on scripts unless they contain shopt -s expand_aliases.

    If you are using sed in a script, you can try to automate switching to gsed:

    #!/usr/bin/env bash
    set -Eeuo pipefail
    
    if [[ "$OSTYPE" == "darwin"* ]]; then
      # Require gnu-sed.
      if ! [ -x "$(command -v gsed)" ]; then
        echo "Error: 'gsed' is not istalled." >&2
        echo "If you are using Homebrew, install with 'brew install gnu-sed'." >&2
        exit 1
      fi
      SED_CMD=gsed
    else
      SED_CMD=sed
    fi
    
    # Use '${SED_CMD}' instead of 'sed'
    ${SED_CMD} -i'' -e 's/hello/bye/g' my_file.txt
    

    You can temporarily set PATH to use "gnu-sed" sed for a script:

    # run a linux script containing sed without changing it
    PATH="$(brew --prefix)/opt/gnu-sed/libexec/gnubin:$PATH" ./linux_script_using_sed.sh
    

    If you are copy/pasting linux scripts, you can alias gsed to sed in the current shell:

    alias sed=gsed
    sed -i 's/hello/bye/g' just_a_file.txt
    
    1. Use -i '' on macOS and BSD or -i (GNU sed) otherwise
    #!/usr/bin/env bash
    set -Eeuo pipefail
    
    case "$OSTYPE" in
      darwin*|bsd*)
        echo "Using BSD sed style"
        sed_no_backup=( -i '' )
        ;; 
      *)
        echo "Using GNU sed style"
        sed_no_backup=( -i )
        ;;
    esac
    
    sed ${sed_no_backup[@]} -e 's/hello/bye/g' my_file.txt