1268 lines
32 KiB
Bash
Executable File
1268 lines
32 KiB
Bash
Executable File
#!/bin/sh
|
|
#---------------------------------------------
|
|
# xdg-open
|
|
#
|
|
# Utility script to open a URL in the registered default application.
|
|
#
|
|
# Refer to the usage() function below for usage.
|
|
#
|
|
# Copyright 2009-2010, Fathi Boudra <fabo@freedesktop.org>
|
|
# Copyright 2009-2016, Rex Dieter <rdieter@fedoraproject.org>
|
|
# Copyright 2006, Kevin Krammer <kevin.krammer@gmx.at>
|
|
# Copyright 2006, Jeremy White <jwhite@codeweavers.com>
|
|
#
|
|
# LICENSE:
|
|
#
|
|
# Permission is hereby granted, free of charge, to any person obtaining a
|
|
# copy of this software and associated documentation files (the "Software"),
|
|
# to deal in the Software without restriction, including without limitation
|
|
# the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
# and/or sell copies of the Software, and to permit persons to whom the
|
|
# Software is furnished to do so, subject to the following conditions:
|
|
#
|
|
# The above copyright notice and this permission notice shall be included
|
|
# in all copies or substantial portions of the Software.
|
|
#
|
|
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
# OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
# THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
|
|
# OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
|
# ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
# OTHER DEALINGS IN THE SOFTWARE.
|
|
#
|
|
#---------------------------------------------
|
|
|
|
manualpage()
|
|
{
|
|
cat << '_MANUALPAGE'
|
|
Name
|
|
|
|
xdg-open -- opens a file or URL in the user's preferred
|
|
application
|
|
|
|
Synopsis
|
|
|
|
xdg-open { file | URL }
|
|
|
|
xdg-open { --help | --manual | --version }
|
|
|
|
Description
|
|
|
|
xdg-open opens a file or URL in the user's preferred
|
|
application. If a URL is provided the URL will be opened in the
|
|
user's preferred web browser. If a file is provided the file
|
|
will be opened in the preferred application for files of that
|
|
type. xdg-open supports file, ftp, http and https URLs.
|
|
|
|
xdg-open is for use inside a desktop session only. It is not
|
|
recommended to use xdg-open as root.
|
|
|
|
As xdg-open can not handle arguments that begin with a "-" it
|
|
is recommended to pass filepaths in one of the following ways:
|
|
* Pass absolute paths, i.e. by using realpath as a
|
|
preprocessor.
|
|
* Prefix known relative filepaths with a "./". For example
|
|
using sed -E 's|^[^/]|./\0|'.
|
|
* Pass a file URL.
|
|
|
|
Options
|
|
|
|
--help
|
|
Show command synopsis.
|
|
|
|
--manual
|
|
Show this manual page.
|
|
|
|
--version
|
|
Show the xdg-utils version information.
|
|
|
|
Exit Codes
|
|
|
|
An exit code of 0 indicates success while a non-zero exit code
|
|
indicates failure. The following failure codes can be returned:
|
|
|
|
1
|
|
Error in command line syntax.
|
|
|
|
2
|
|
One of the files passed on the command line did not
|
|
exist.
|
|
|
|
3
|
|
A required tool could not be found.
|
|
|
|
4
|
|
The action failed.
|
|
|
|
In case of success the process launched from the .desktop file
|
|
will not be forked off and therefore may result in xdg-open
|
|
running for a very long time. This behaviour intentionally
|
|
differs from most desktop specific openers to allow terminal
|
|
based applications to run using the same terminal xdg-open was
|
|
called from.
|
|
|
|
Reporting Issues
|
|
|
|
Please keep in mind xdg-open inherits most of the flaws of its
|
|
configuration and the underlying opener.
|
|
|
|
In case the command xdg-mime query default "$(xdg-mime query
|
|
filetype path/to/troublesome_file)" names the program
|
|
responsible for any unexpected behaviour you can fix that by
|
|
setting a different handler. (If the program is broken let the
|
|
developers know)
|
|
|
|
Also see the security note on xdg-mime(1) for the default
|
|
subcommand.
|
|
|
|
If a flaw is reproducible using the desktop specific opener
|
|
(and isn't a configuration issue): Please report to whoever is
|
|
responsible for that first (reporting to xdg-utils is better
|
|
than not reporting at all, but since the xdg-utils are
|
|
maintained in very little spare time a fix will take much
|
|
longer)
|
|
|
|
In case an issue specific to xdg-open please report it to
|
|
https://gitlab.freedesktop.org/xdg/xdg-utils/-/issues .
|
|
|
|
See Also
|
|
|
|
xdg-mime(1), xdg-settings(1), MIME applications associations
|
|
specification
|
|
|
|
Examples
|
|
|
|
xdg-open 'http://www.freedesktop.org/'
|
|
|
|
Opens the freedesktop.org website in the user's default
|
|
browser.
|
|
|
|
xdg-open /tmp/foobar.png
|
|
|
|
Opens the PNG image file /tmp/foobar.png in the user's default
|
|
image viewing application.
|
|
_MANUALPAGE
|
|
}
|
|
|
|
usage()
|
|
{
|
|
cat << '_USAGE'
|
|
xdg-open -- opens a file or URL in the user's preferred
|
|
application
|
|
|
|
Synopsis
|
|
|
|
xdg-open { file | URL }
|
|
|
|
xdg-open { --help | --manual | --version }
|
|
|
|
_USAGE
|
|
}
|
|
|
|
#@xdg-utils-common@
|
|
#----------------------------------------------------------------------------
|
|
# Common utility functions included in all XDG wrapper scripts
|
|
#----------------------------------------------------------------------------
|
|
|
|
#shellcheck shell=sh
|
|
|
|
DEBUG()
|
|
{
|
|
[ -z "${XDG_UTILS_DEBUG_LEVEL}" ] && return 0;
|
|
[ "${XDG_UTILS_DEBUG_LEVEL}" -lt "$1" ] && return 0;
|
|
shift
|
|
echo "$@" >&2
|
|
}
|
|
|
|
# This handles backslashes but not quote marks.
|
|
first_word()
|
|
{
|
|
# shellcheck disable=SC2162 # No -r is intended here
|
|
read first rest
|
|
echo "$first"
|
|
}
|
|
|
|
#-------------------------------------------------------------
|
|
# map a binary to a .desktop file
|
|
binary_to_desktop_file()
|
|
{
|
|
search="${XDG_DATA_HOME:-$HOME/.local/share}:${XDG_DATA_DIRS:-/usr/local/share:/usr/share}"
|
|
binary="$(command -v "$1")"
|
|
binary="$(xdg_realpath "$binary")"
|
|
base="$(basename "$binary")"
|
|
IFS=:
|
|
for dir in $search; do
|
|
unset IFS
|
|
[ "$dir" ] || continue
|
|
[ -d "$dir/applications" ] || [ -d "$dir/applnk" ] || continue
|
|
for file in "$dir"/applications/*.desktop "$dir"/applications/*/*.desktop "$dir"/applnk/*.desktop "$dir"/applnk/*/*.desktop; do
|
|
[ -r "$file" ] || continue
|
|
# Check to make sure it's worth the processing.
|
|
grep -q "^Exec.*$base" "$file" || continue
|
|
# Make sure it's a visible desktop file (e.g. not "preferred-web-browser.desktop").
|
|
grep -Eq "^(NoDisplay|Hidden)=true" "$file" && continue
|
|
command="$(grep -E "^Exec(\[[^]=]*])?=" "$file" | cut -d= -f 2- | first_word)"
|
|
command="$(command -v "$command")"
|
|
if [ x"$(xdg_realpath "$command")" = x"$binary" ]; then
|
|
# Fix any double slashes that got added path composition
|
|
echo "$file" | tr -s /
|
|
return
|
|
fi
|
|
done
|
|
done
|
|
}
|
|
|
|
#-------------------------------------------------------------
|
|
# map a .desktop file to a binary
|
|
desktop_file_to_binary()
|
|
{
|
|
search="${XDG_DATA_HOME:-$HOME/.local/share}:${XDG_DATA_DIRS:-/usr/local/share:/usr/share}"
|
|
desktop="$(basename "$1")"
|
|
IFS=:
|
|
for dir in $search; do
|
|
unset IFS
|
|
[ "$dir" ] && [ -d "$dir/applications" ] || [ -d "$dir/applnk" ] || continue
|
|
# Check if desktop file contains -
|
|
if [ "${desktop#*-}" != "$desktop" ]; then
|
|
vendor=${desktop%-*}
|
|
app=${desktop#*-}
|
|
if [ -r "$dir/applications/$vendor/$app" ]; then
|
|
file_path="$dir/applications/$vendor/$app"
|
|
elif [ -r "$dir/applnk/$vendor/$app" ]; then
|
|
file_path="$dir/applnk/$vendor/$app"
|
|
fi
|
|
fi
|
|
if test -z "$file_path" ; then
|
|
for indir in "$dir"/applications/ "$dir"/applications/*/ "$dir"/applnk/ "$dir"/applnk/*/; do
|
|
file="$indir/$desktop"
|
|
if [ -r "$file" ]; then
|
|
file_path=$file
|
|
break
|
|
fi
|
|
done
|
|
fi
|
|
if [ -r "$file_path" ]; then
|
|
# Remove any arguments (%F, %f, %U, %u, etc.).
|
|
command="$(grep -E "^Exec(\[[^]=]*])?=" "$file_path" | cut -d= -f 2- | first_word)"
|
|
command="$(command -v "$command")"
|
|
xdg_realpath "$command"
|
|
return
|
|
fi
|
|
done
|
|
}
|
|
|
|
#-------------------------------------------------------------
|
|
# Exit script on successfully completing the desired operation
|
|
|
|
# shellcheck disable=SC2120 # It is okay to call this without arguments
|
|
exit_success()
|
|
{
|
|
if [ $# -gt 0 ]; then
|
|
echo "$*"
|
|
echo
|
|
fi
|
|
|
|
exit 0
|
|
}
|
|
|
|
|
|
#-----------------------------------------
|
|
# Exit script on malformed arguments, not enough arguments
|
|
# or missing required option.
|
|
# prints usage information
|
|
|
|
exit_failure_syntax()
|
|
{
|
|
if [ $# -gt 0 ]; then
|
|
echo "xdg-open: $*" >&2
|
|
echo "Try 'xdg-open --help' for more information." >&2
|
|
else
|
|
usage
|
|
echo "Use 'man xdg-open' or 'xdg-open --manual' for additional info."
|
|
fi
|
|
|
|
exit 1
|
|
}
|
|
|
|
#-------------------------------------------------------------
|
|
# Exit script on missing file specified on command line
|
|
|
|
exit_failure_file_missing()
|
|
{
|
|
if [ $# -gt 0 ]; then
|
|
echo "xdg-open: $*" >&2
|
|
fi
|
|
|
|
exit 2
|
|
}
|
|
|
|
#-------------------------------------------------------------
|
|
# Exit script on failure to locate necessary tool applications
|
|
|
|
exit_failure_operation_impossible()
|
|
{
|
|
if [ $# -gt 0 ]; then
|
|
echo "xdg-open: $*" >&2
|
|
fi
|
|
|
|
exit 3
|
|
}
|
|
|
|
#-------------------------------------------------------------
|
|
# Exit script on failure returned by a tool application
|
|
|
|
exit_failure_operation_failed()
|
|
{
|
|
if [ $# -gt 0 ]; then
|
|
echo "xdg-open: $*" >&2
|
|
fi
|
|
|
|
exit 4
|
|
}
|
|
|
|
#------------------------------------------------------------
|
|
# Exit script on insufficient permission to read a specified file
|
|
|
|
exit_failure_file_permission_read()
|
|
{
|
|
if [ $# -gt 0 ]; then
|
|
echo "xdg-open: $*" >&2
|
|
fi
|
|
|
|
exit 5
|
|
}
|
|
|
|
#------------------------------------------------------------
|
|
# Exit script on insufficient permission to write a specified file
|
|
|
|
exit_failure_file_permission_write()
|
|
{
|
|
if [ $# -gt 0 ]; then
|
|
echo "xdg-open: $*" >&2
|
|
fi
|
|
|
|
exit 6
|
|
}
|
|
|
|
check_input_file()
|
|
{
|
|
if [ ! -e "$1" ]; then
|
|
exit_failure_file_missing "file '$1' does not exist"
|
|
fi
|
|
if [ ! -r "$1" ]; then
|
|
exit_failure_file_permission_read "no permission to read file '$1'"
|
|
fi
|
|
}
|
|
|
|
check_vendor_prefix()
|
|
{
|
|
file_label="$2"
|
|
[ -n "$file_label" ] || file_label="filename"
|
|
file="$(basename "$1")"
|
|
case "$file" in
|
|
[[:alpha:]]*-*)
|
|
return
|
|
;;
|
|
esac
|
|
|
|
echo "xdg-open: $file_label '$file' does not have a proper vendor prefix" >&2
|
|
echo 'A vendor prefix consists of alpha characters ([a-zA-Z]) and is terminated' >&2
|
|
echo 'with a dash ("-"). An example '"$file_label"' is '"'example-$file'" >&2
|
|
echo "Use --novendor to override or 'xdg-open --manual' for additional info." >&2
|
|
exit 1
|
|
}
|
|
|
|
check_output_file()
|
|
{
|
|
# if the file exists, check if it is writeable
|
|
# if it does not exists, check if we are allowed to write on the directory
|
|
if [ -e "$1" ]; then
|
|
if [ ! -w "$1" ]; then
|
|
exit_failure_file_permission_write "no permission to write to file '$1'"
|
|
fi
|
|
else
|
|
DIR="$(dirname "$1")"
|
|
if [ ! -w "$DIR" ] || [ ! -x "$DIR" ]; then
|
|
exit_failure_file_permission_write "no permission to create file '$1'"
|
|
fi
|
|
fi
|
|
}
|
|
|
|
#----------------------------------------
|
|
# Checks for shared commands, e.g. --help
|
|
|
|
check_common_commands()
|
|
{
|
|
while [ $# -gt 0 ] ; do
|
|
parm="$1"
|
|
shift
|
|
|
|
case "$parm" in
|
|
--help)
|
|
usage
|
|
echo "Use 'man xdg-open' or 'xdg-open --manual' for additional info."
|
|
exit_success
|
|
;;
|
|
|
|
--manual)
|
|
manualpage
|
|
exit_success
|
|
;;
|
|
|
|
--version)
|
|
echo "xdg-open 1.2.1"
|
|
exit_success
|
|
;;
|
|
|
|
--)
|
|
[ -z "$XDG_UTILS_ENABLE_DOUBLE_HYPEN" ] || break
|
|
;;
|
|
esac
|
|
done
|
|
}
|
|
|
|
check_common_commands "$@"
|
|
|
|
[ -z "${XDG_UTILS_DEBUG_LEVEL}" ] && unset XDG_UTILS_DEBUG_LEVEL;
|
|
# shellcheck disable=SC2034
|
|
if [ "${XDG_UTILS_DEBUG_LEVEL-0}" -lt 1 ]; then
|
|
# Be silent
|
|
xdg_redirect_output=" > /dev/null 2> /dev/null"
|
|
else
|
|
# All output to stderr
|
|
xdg_redirect_output=" >&2"
|
|
fi
|
|
|
|
#--------------------------------------
|
|
# Checks for known desktop environments
|
|
# set variable DE to the desktop environments name, lowercase
|
|
|
|
detectDE()
|
|
{
|
|
# see https://bugs.freedesktop.org/show_bug.cgi?id=34164
|
|
unset GREP_OPTIONS
|
|
|
|
if [ -n "${XDG_CURRENT_DESKTOP}" ]; then
|
|
case "${XDG_CURRENT_DESKTOP}" in
|
|
# only recently added to menu-spec, pre-spec X- still in use
|
|
Cinnamon|X-Cinnamon)
|
|
DE=cinnamon;
|
|
;;
|
|
ENLIGHTENMENT)
|
|
DE=enlightenment;
|
|
;;
|
|
# GNOME, GNOME-Classic:GNOME, or GNOME-Flashback:GNOME
|
|
GNOME*)
|
|
DE=gnome;
|
|
;;
|
|
KDE)
|
|
DE=kde;
|
|
;;
|
|
DEEPIN|Deepin|deepin)
|
|
DE=deepin;
|
|
;;
|
|
LXDE)
|
|
DE=lxde;
|
|
;;
|
|
LXQt)
|
|
DE=lxqt;
|
|
;;
|
|
MATE)
|
|
DE=mate;
|
|
;;
|
|
XFCE)
|
|
DE=xfce
|
|
;;
|
|
X-Generic)
|
|
DE=generic
|
|
;;
|
|
esac
|
|
fi
|
|
|
|
# shellcheck disable=SC2153
|
|
if [ -z "$DE" ]; then
|
|
# classic fallbacks
|
|
if [ -n "$KDE_FULL_SESSION" ]; then DE=kde;
|
|
elif [ -n "$GNOME_DESKTOP_SESSION_ID" ]; then DE=gnome;
|
|
elif [ -n "$MATE_DESKTOP_SESSION_ID" ]; then DE=mate;
|
|
elif dbus-send --print-reply --dest=org.freedesktop.DBus /org/freedesktop/DBus org.freedesktop.DBus.GetNameOwner string:org.gnome.SessionManager > /dev/null 2>&1 ; then DE=gnome;
|
|
elif xprop -root _DT_SAVE_MODE 2> /dev/null | grep ' = \"xfce4\"$' >/dev/null 2>&1; then DE=xfce;
|
|
elif xprop -root 2> /dev/null | grep -i '^xfce_desktop_window' >/dev/null 2>&1; then DE=xfce
|
|
elif echo "$DESKTOP" | grep -q '^Enlightenment'; then DE=enlightenment;
|
|
elif [ -n "$LXQT_SESSION_CONFIG" ]; then DE=lxqt;
|
|
fi
|
|
fi
|
|
|
|
if [ -z "$DE" ]; then
|
|
# fallback to checking $DESKTOP_SESSION
|
|
case "$DESKTOP_SESSION" in
|
|
gnome)
|
|
DE=gnome;
|
|
;;
|
|
LXDE|Lubuntu)
|
|
DE=lxde;
|
|
;;
|
|
MATE)
|
|
DE=mate;
|
|
;;
|
|
xfce|xfce4|'Xfce Session')
|
|
DE=xfce;
|
|
;;
|
|
esac
|
|
fi
|
|
|
|
if [ -z "$DE" ]; then
|
|
# fallback to uname output for other platforms
|
|
case "$(uname 2>/dev/null)" in
|
|
CYGWIN*)
|
|
DE=cygwin;
|
|
;;
|
|
Darwin)
|
|
DE=darwin;
|
|
;;
|
|
Linux)
|
|
grep -q microsoft /proc/version > /dev/null 2>&1 && \
|
|
command -v explorer.exe > /dev/null 2>&1 && \
|
|
DE=wsl;
|
|
;;
|
|
esac
|
|
fi
|
|
|
|
if [ x"$DE" = x"gnome" ]; then
|
|
# gnome-default-applications-properties is only available in GNOME 2.x
|
|
# but not in GNOME 3.x
|
|
command -v gnome-default-applications-properties > /dev/null || DE="gnome3"
|
|
fi
|
|
|
|
if [ -f "$XDG_RUNTIME_DIR/flatpak-info" ]; then
|
|
DE="flatpak"
|
|
fi
|
|
}
|
|
|
|
#----------------------------------------------------------------------------
|
|
# kfmclient exec/openURL can give bogus exit value in KDE <= 3.5.4
|
|
# It also always returns 1 in KDE 3.4 and earlier
|
|
# Simply return 0 in such case
|
|
|
|
kfmclient_fix_exit_code()
|
|
{
|
|
version="$(LC_ALL=C.UTF-8 kde-config --version 2>/dev/null | grep '^KDE')"
|
|
major="$(echo "$version" | sed 's/KDE.*: \([0-9]\).*/\1/')"
|
|
minor="$(echo "$version" | sed 's/KDE.*: [0-9]*\.\([0-9]\).*/\1/')"
|
|
release="$(echo "$version" | sed 's/KDE.*: [0-9]*\.[0-9]*\.\([0-9]\).*/\1/')"
|
|
test "$major" -gt 3 && return "$1"
|
|
test "$minor" -gt 5 && return "$1"
|
|
test "$release" -gt 4 && return "$1"
|
|
return 0
|
|
}
|
|
|
|
#----------------------------------------------------------------------------
|
|
# Returns true if there is a graphical display attached.
|
|
|
|
has_display()
|
|
{
|
|
if [ -n "$DISPLAY" ] || [ -n "$WAYLAND_DISPLAY" ]; then
|
|
return 0
|
|
else
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
#----------------------------------------------------------------------------
|
|
# Prefixes a path with a "./" if it starts with a "-".
|
|
# This is useful for programs to not confuse paths with options.
|
|
|
|
unoption_path()
|
|
{
|
|
case "$1" in
|
|
-*)
|
|
printf "./%s" "$1" ;;
|
|
*)
|
|
printf "%s" "$1" ;;
|
|
esac
|
|
}
|
|
|
|
#----------------------------------------------------------------------------
|
|
# Performs a symlink and relative path resolving for a single argument.
|
|
# This will always fail if the given file does not exist!
|
|
|
|
xdg_realpath()
|
|
{
|
|
# allow caching and external configuration
|
|
if [ -z "$XDG_UTILS_REALPATH_BACKEND" ] ; then
|
|
if command -v realpath >/dev/null 2>/dev/null ; then
|
|
lines="$(realpath -- / 2>&1)"
|
|
if [ $? = 0 ] && [ "$lines" = "/" ] ; then
|
|
XDG_UTILS_REALPATH_BACKEND="realpath"
|
|
else
|
|
# The realpath took the -- literally, probably the busybox implementation
|
|
XDG_UTILS_REALPATH_BACKEND="busybox-realpath"
|
|
fi
|
|
unset lines
|
|
elif command -v readlink >/dev/null 2>/dev/null ; then
|
|
XDG_UTILS_REALPATH_BACKEND="readlink"
|
|
else
|
|
exit_failure_operation_failed "No usable realpath backend found. Have a realpath binary or a readlink -f that canonicalizes paths."
|
|
fi
|
|
fi
|
|
# Always fail if the file doesn't exist (busybox realpath does that for example)
|
|
[ -e "$1" ] || return 1
|
|
case "$XDG_UTILS_REALPATH_BACKEND" in
|
|
realpath)
|
|
realpath -- "$1"
|
|
;;
|
|
busybox-realpath)
|
|
# busybox style realpath implementations have options too
|
|
realpath "$(unoption_path "$1")"
|
|
;;
|
|
readlink)
|
|
readlink -f "$(unoption_path "$1")"
|
|
;;
|
|
*)
|
|
exit_failure_operation_impossible "Realpath backend '$XDG_UTILS_REALPATH_BACKEND' not recognized."
|
|
;;
|
|
esac
|
|
}
|
|
|
|
# This handles backslashes but not quote marks.
|
|
last_word()
|
|
{
|
|
# Backslash handling is intended, not using `first` too
|
|
# shellcheck disable=SC2162,SC2034
|
|
read first rest
|
|
echo "$rest"
|
|
}
|
|
|
|
# Get the value of a key in a desktop file's Desktop Entry group.
|
|
# Example: Use get_key foo.desktop Exec
|
|
# to get the values of the Exec= key for the Desktop Entry group.
|
|
get_key()
|
|
{
|
|
local file="${1}"
|
|
local key="${2}"
|
|
local desktop_entry=""
|
|
|
|
IFS_="${IFS}"
|
|
IFS=""
|
|
# No backslash handling here, first_word and last_word do that
|
|
while read -r line
|
|
do
|
|
case "$line" in
|
|
"[Desktop Entry]")
|
|
desktop_entry="y"
|
|
;;
|
|
# Reset match flag for other groups
|
|
"["*)
|
|
desktop_entry=""
|
|
;;
|
|
"${key}="*)
|
|
# Only match Desktop Entry group
|
|
if [ -n "${desktop_entry}" ]
|
|
then
|
|
echo "${line}" | cut -d= -f 2-
|
|
fi
|
|
esac
|
|
done < "${file}"
|
|
IFS="${IFS_}"
|
|
}
|
|
|
|
has_url_scheme()
|
|
{
|
|
echo "$1" | LC_ALL=C grep -Eq '^[[:alpha:]][[:alpha:][:digit:]+\.\-]*:'
|
|
}
|
|
|
|
# Returns true if argument is a file:// URL or path
|
|
is_file_url_or_path()
|
|
{
|
|
if echo "$1" | grep -q '^file://' || ! has_url_scheme "$1" ; then
|
|
return 0
|
|
else
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
get_hostname() {
|
|
if [ -z "$HOSTNAME" ]; then
|
|
if command -v hostname > /dev/null; then
|
|
HOSTNAME=$(hostname)
|
|
else
|
|
HOSTNAME=$(uname -n)
|
|
fi
|
|
fi
|
|
}
|
|
|
|
# If argument is a file URL, convert it to a (percent-decoded) path.
|
|
# If not, leave it as it is.
|
|
file_url_to_path()
|
|
{
|
|
local file="$1"
|
|
get_hostname
|
|
if echo "$file" | grep -q '^file://'; then
|
|
file=${file#file://localhost}
|
|
file=${file#file://"$HOSTNAME"}
|
|
file=${file#file://}
|
|
if ! echo "$file" | grep -q '^/'; then
|
|
echo "$file"
|
|
return
|
|
fi
|
|
file=${file%%#*}
|
|
file=${file%%\?*}
|
|
local printf=printf
|
|
if [ -x /usr/bin/printf ]; then
|
|
printf=/usr/bin/printf
|
|
fi
|
|
file=$($printf "$(echo "$file" | sed -e 's@%\([a-f0-9A-F]\{2\}\)@\\x\1@g')")
|
|
fi
|
|
echo "$file"
|
|
}
|
|
|
|
open_cygwin()
|
|
{
|
|
cygstart "$1"
|
|
|
|
if [ $? -eq 0 ]; then
|
|
exit_success
|
|
else
|
|
exit_failure_operation_failed
|
|
fi
|
|
}
|
|
|
|
open_darwin()
|
|
{
|
|
open "$1"
|
|
|
|
if [ $? -eq 0 ]; then
|
|
exit_success
|
|
else
|
|
exit_failure_operation_failed
|
|
fi
|
|
}
|
|
|
|
open_kde()
|
|
{
|
|
if [ -n "${KDE_SESSION_VERSION}" ]; then
|
|
case "${KDE_SESSION_VERSION}" in
|
|
4)
|
|
kde-open "$1"
|
|
;;
|
|
5)
|
|
"kde-open${KDE_SESSION_VERSION}" "$1"
|
|
;;
|
|
6)
|
|
kde-open "$1"
|
|
;;
|
|
esac
|
|
else
|
|
kfmclient exec "$1"
|
|
kfmclient_fix_exit_code $?
|
|
fi
|
|
|
|
if [ $? -eq 0 ]; then
|
|
exit_success
|
|
else
|
|
exit_failure_operation_failed
|
|
fi
|
|
}
|
|
|
|
open_deepin()
|
|
{
|
|
if dde-open -version >/dev/null 2>&1; then
|
|
dde-open "$1"
|
|
else
|
|
open_generic "$1"
|
|
fi
|
|
|
|
if [ $? -eq 0 ]; then
|
|
exit_success
|
|
else
|
|
exit_failure_operation_failed
|
|
fi
|
|
}
|
|
|
|
open_gnome3()
|
|
{
|
|
if gio help open 2>/dev/null 1>&2; then
|
|
gio open "$1"
|
|
elif gvfs-open --help 2>/dev/null 1>&2; then
|
|
gvfs-open "$1"
|
|
else
|
|
open_generic "$1"
|
|
fi
|
|
|
|
if [ $? -eq 0 ]; then
|
|
exit_success
|
|
else
|
|
exit_failure_operation_failed
|
|
fi
|
|
}
|
|
|
|
open_gnome()
|
|
{
|
|
if gio help open 2>/dev/null 1>&2; then
|
|
gio open "$1"
|
|
elif gvfs-open --help 2>/dev/null 1>&2; then
|
|
gvfs-open "$1"
|
|
elif gnome-open --help 2>/dev/null 1>&2; then
|
|
gnome-open "$1"
|
|
else
|
|
open_generic "$1"
|
|
fi
|
|
|
|
if [ $? -eq 0 ]; then
|
|
exit_success
|
|
else
|
|
exit_failure_operation_failed
|
|
fi
|
|
}
|
|
|
|
open_mate()
|
|
{
|
|
if gio help open 2>/dev/null 1>&2; then
|
|
gio open "$1"
|
|
elif gvfs-open --help 2>/dev/null 1>&2; then
|
|
gvfs-open "$1"
|
|
elif mate-open --help 2>/dev/null 1>&2; then
|
|
mate-open "$1"
|
|
else
|
|
open_generic "$1"
|
|
fi
|
|
|
|
if [ $? -eq 0 ]; then
|
|
exit_success
|
|
else
|
|
exit_failure_operation_failed
|
|
fi
|
|
}
|
|
|
|
open_xfce()
|
|
{
|
|
if exo-open --help 2>/dev/null 1>&2; then
|
|
exo-open "$1"
|
|
elif gio help open 2>/dev/null 1>&2; then
|
|
gio open "$1"
|
|
elif gvfs-open --help 2>/dev/null 1>&2; then
|
|
gvfs-open "$1"
|
|
else
|
|
open_generic "$1"
|
|
fi
|
|
|
|
if [ $? -eq 0 ]; then
|
|
exit_success
|
|
else
|
|
exit_failure_operation_failed
|
|
fi
|
|
}
|
|
|
|
open_enlightenment()
|
|
{
|
|
if enlightenment_open --help 2>/dev/null 1>&2; then
|
|
enlightenment_open "$1"
|
|
else
|
|
open_generic "$1"
|
|
fi
|
|
|
|
if [ $? -eq 0 ]; then
|
|
exit_success
|
|
else
|
|
exit_failure_operation_failed
|
|
fi
|
|
}
|
|
|
|
open_flatpak()
|
|
{
|
|
if is_file_url_or_path "$1"; then
|
|
local file
|
|
file="$(file_url_to_path "$1")"
|
|
|
|
check_input_file "$file"
|
|
|
|
gdbus call --session \
|
|
--dest org.freedesktop.portal.Desktop \
|
|
--object-path /org/freedesktop/portal/desktop \
|
|
--method org.freedesktop.portal.OpenURI.OpenFile \
|
|
--timeout 5 \
|
|
"" "3" {} 3< "$file"
|
|
else
|
|
# $1 contains an URI
|
|
|
|
gdbus call --session \
|
|
--dest org.freedesktop.portal.Desktop \
|
|
--object-path /org/freedesktop/portal/desktop \
|
|
--method org.freedesktop.portal.OpenURI.OpenURI \
|
|
--timeout 5 \
|
|
"" "$1" {}
|
|
fi
|
|
|
|
if [ $? -eq 0 ]; then
|
|
exit_success
|
|
else
|
|
exit_failure_operation_failed
|
|
fi
|
|
}
|
|
|
|
#-----------------------------------------
|
|
# Recursively search .desktop file
|
|
|
|
#(application, directory, target file, target_url)
|
|
search_desktop_file()
|
|
{
|
|
local default="$1"
|
|
local dir="$2"
|
|
local target="$3"
|
|
local target_uri="$4"
|
|
|
|
local file=""
|
|
# look for both vendor-app.desktop, vendor/app.desktop
|
|
if [ -r "$dir/$default" ]; then
|
|
file="$dir/$default"
|
|
elif [ -r "$dir/$(echo "$default" | sed -e 's|-|/|')" ]; then
|
|
file="$dir/$(echo "$default" | sed -e 's|-|/|')"
|
|
fi
|
|
|
|
if [ -r "$file" ] ; then
|
|
command="$(get_key "${file}" "Exec" | first_word)"
|
|
if command -v "$command" >/dev/null; then
|
|
icon="$(get_key "${file}" "Icon")"
|
|
# FIXME: Actually LC_MESSAGES should be used as described in
|
|
# http://standards.freedesktop.org/desktop-entry-spec/latest/ar01s04.html
|
|
localised_name="$(get_key "${file}" "Name")"
|
|
#shellcheck disable=SC2046 # Splitting is intentional here
|
|
set -- $(get_key "${file}" "Exec" | last_word)
|
|
# We need to replace any occurrence of "%f", "%F" and
|
|
# the like by the target file. We examine each
|
|
# argument and append the modified argument to the
|
|
# end then shift.
|
|
local args=$#
|
|
local replaced=0
|
|
while [ $args -gt 0 ]; do
|
|
case $1 in
|
|
%[c])
|
|
replaced=1
|
|
arg="${localised_name}"
|
|
shift
|
|
set -- "$@" "$arg"
|
|
;;
|
|
%[fF])
|
|
# if there is only a target_url return,
|
|
# this application can't handle it.
|
|
[ -n "$target" ] || return
|
|
replaced=1
|
|
arg="$target"
|
|
shift
|
|
set -- "$@" "$arg"
|
|
;;
|
|
%[uU])
|
|
replaced=1
|
|
# When an URI is requested use it,
|
|
# otherwise fall back to the filepath.
|
|
arg="${target_uri:-$target}"
|
|
shift
|
|
set -- "$@" "$arg"
|
|
;;
|
|
%[i])
|
|
replaced=1
|
|
shift
|
|
set -- "$@" "--icon" "$icon"
|
|
;;
|
|
*)
|
|
arg="$1"
|
|
shift
|
|
set -- "$@" "$arg"
|
|
;;
|
|
esac
|
|
args=$(( args - 1 ))
|
|
done
|
|
[ $replaced -eq 1 ] || set -- "$@" "${target:-$target_uri}"
|
|
env "$command" "$@"
|
|
exit_success
|
|
fi
|
|
fi
|
|
|
|
for d in "$dir/"*/; do
|
|
[ -d "$d" ] && search_desktop_file "$default" "$d" "$target" "$target_uri"
|
|
done
|
|
}
|
|
|
|
# (file (or empty), mimetype, optional url)
|
|
open_generic_xdg_mime()
|
|
{
|
|
filetype="$2"
|
|
default="$(xdg-mime query default "$filetype")"
|
|
if [ -n "$default" ] ; then
|
|
xdg_user_dir="$XDG_DATA_HOME"
|
|
[ -n "$xdg_user_dir" ] || xdg_user_dir="$HOME/.local/share"
|
|
|
|
xdg_system_dirs="$XDG_DATA_DIRS"
|
|
[ -n "$xdg_system_dirs" ] || xdg_system_dirs=/usr/local/share/:/usr/share/
|
|
|
|
search_dirs="$xdg_user_dir:$xdg_system_dirs"
|
|
DEBUG 3 "$search_dirs"
|
|
old_ifs="$IFS"
|
|
IFS=:
|
|
for x in $search_dirs ; do
|
|
IFS="$old_ifs"
|
|
search_desktop_file "$default" "$x/applications/" "$1" "$3"
|
|
done
|
|
fi
|
|
}
|
|
|
|
open_generic_xdg_x_scheme_handler()
|
|
{
|
|
scheme="$(echo "$1" | LC_ALL=C sed -n 's/\(^[[:alpha:]][[:alnum:]+\.-]*\):.*$/\1/p')"
|
|
if [ -n "$scheme" ]; then
|
|
filetype="x-scheme-handler/$scheme"
|
|
open_generic_xdg_mime "" "$filetype" "$1"
|
|
fi
|
|
}
|
|
|
|
has_single_argument()
|
|
{
|
|
test $# = 1
|
|
}
|
|
|
|
open_envvar()
|
|
{
|
|
local oldifs="$IFS"
|
|
local browser
|
|
|
|
IFS=":"
|
|
for browser in $BROWSER; do
|
|
IFS="$oldifs"
|
|
|
|
if [ -z "$browser" ]; then
|
|
continue
|
|
fi
|
|
|
|
if echo "$browser" | grep -q %s; then
|
|
# Avoid argument injection.
|
|
# See https://bugs.freedesktop.org/show_bug.cgi?id=103807
|
|
# URIs don't have IFS characters spaces anyway.
|
|
# shellcheck disable=SC2086,SC2091,SC2059
|
|
# All the scary things here are intentional
|
|
has_single_argument $1 && $(printf "$browser" "$1")
|
|
else
|
|
$browser "$1"
|
|
fi
|
|
|
|
if [ $? -eq 0 ]; then
|
|
exit_success
|
|
fi
|
|
done
|
|
}
|
|
|
|
open_wsl()
|
|
{
|
|
local win_path
|
|
if is_file_url_or_path "$1" ; then
|
|
win_path="$(file_url_to_path "$1")"
|
|
win_path="$(wslpath -aw "$win_path")"
|
|
[ $? -eq 0 ] || exit_failure_operation_failed
|
|
explorer.exe "${win_path}"
|
|
else
|
|
rundll32.exe url.dll,FileProtocolHandler "$1"
|
|
fi
|
|
|
|
if [ $? -eq 0 ]; then
|
|
exit_success
|
|
else
|
|
exit_failure_operation_failed
|
|
fi
|
|
}
|
|
|
|
open_generic()
|
|
{
|
|
if is_file_url_or_path "$1"; then
|
|
local file
|
|
file="$(file_url_to_path "$1")"
|
|
|
|
check_input_file "$file"
|
|
|
|
if has_display; then
|
|
filetype="$(xdg-mime query filetype "$file" | sed "s/;.*//")"
|
|
# passing a path a url is okay too,
|
|
# see desktop file specification for '%u'
|
|
open_generic_xdg_mime "$file" "$filetype" "$1"
|
|
fi
|
|
|
|
if command -v run-mailcap >/dev/null; then
|
|
run-mailcap --action=view "$file"
|
|
if [ $? -eq 0 ]; then
|
|
exit_success
|
|
fi
|
|
fi
|
|
|
|
if has_display && mimeopen -v 2>/dev/null 1>&2; then
|
|
mimeopen -L -n "$file"
|
|
if [ $? -eq 0 ]; then
|
|
exit_success
|
|
fi
|
|
fi
|
|
fi
|
|
|
|
if has_display; then
|
|
open_generic_xdg_x_scheme_handler "$1"
|
|
fi
|
|
|
|
if [ -n "$BROWSER" ]; then
|
|
open_envvar "$1"
|
|
fi
|
|
|
|
# if BROWSER variable is not set, check some well known browsers instead
|
|
if [ x"$BROWSER" = x"" ]; then
|
|
BROWSER=www-browser:links2:elinks:links:lynx:w3m
|
|
if has_display; then
|
|
BROWSER=x-www-browser:firefox:iceweasel:seamonkey:mozilla:epiphany:konqueror:chromium:chromium-browser:google-chrome:$BROWSER
|
|
fi
|
|
fi
|
|
|
|
open_envvar "$1"
|
|
|
|
exit_failure_operation_impossible "no method available for opening '$1'"
|
|
}
|
|
|
|
open_lxde()
|
|
{
|
|
|
|
# pcmanfm only knows how to handle file:// urls and filepaths, it seems.
|
|
if pcmanfm --help >/dev/null 2>&1 && is_file_url_or_path "$1"; then
|
|
local file
|
|
file="$(file_url_to_path "$1")"
|
|
|
|
# handle relative paths
|
|
if ! echo "$file" | grep -q ^/; then
|
|
file="$(pwd)/$file"
|
|
fi
|
|
|
|
pcmanfm "$file"
|
|
else
|
|
open_generic "$1"
|
|
fi
|
|
|
|
if [ $? -eq 0 ]; then
|
|
exit_success
|
|
else
|
|
exit_failure_operation_failed
|
|
fi
|
|
}
|
|
|
|
open_lxqt()
|
|
{
|
|
if qtxdg-mat open --help 2>/dev/null 1>&2; then
|
|
qtxdg-mat open "$1"
|
|
else
|
|
exit_failure_operation_impossible "no method available for opening '$1'"
|
|
fi
|
|
|
|
if [ $? -eq 0 ]; then
|
|
exit_success
|
|
else
|
|
exit_failure_operation_failed
|
|
fi
|
|
}
|
|
|
|
[ x"$1" != x"" ] || exit_failure_syntax
|
|
|
|
url=
|
|
while [ $# -gt 0 ] ; do
|
|
parm="$1"
|
|
shift
|
|
|
|
case "$parm" in
|
|
-*)
|
|
exit_failure_syntax "unexpected option '$parm'"
|
|
;;
|
|
|
|
*)
|
|
if [ -n "$url" ] ; then
|
|
exit_failure_syntax "unexpected argument '$parm'"
|
|
fi
|
|
url="$parm"
|
|
;;
|
|
esac
|
|
done
|
|
|
|
if [ -z "${url}" ] ; then
|
|
exit_failure_syntax "file or URL argument missing"
|
|
fi
|
|
|
|
detectDE
|
|
|
|
if [ x"$DE" = x"" ]; then
|
|
DE=generic
|
|
fi
|
|
|
|
DEBUG 2 "Selected DE $DE"
|
|
|
|
# sanitize BROWSER (avoid calling ourselves in particular)
|
|
case "${BROWSER}" in
|
|
*:"xdg-open"|"xdg-open":*)
|
|
BROWSER="$(echo "$BROWSER" | sed -e 's|:xdg-open||g' -e 's|xdg-open:||g')"
|
|
;;
|
|
"xdg-open")
|
|
BROWSER=
|
|
;;
|
|
esac
|
|
|
|
case "$DE" in
|
|
kde)
|
|
open_kde "$url"
|
|
;;
|
|
|
|
deepin)
|
|
open_deepin "$url"
|
|
;;
|
|
|
|
gnome3|cinnamon)
|
|
open_gnome3 "$url"
|
|
;;
|
|
|
|
gnome)
|
|
open_gnome "$url"
|
|
;;
|
|
|
|
mate)
|
|
open_mate "$url"
|
|
;;
|
|
|
|
xfce)
|
|
open_xfce "$url"
|
|
;;
|
|
|
|
lxde)
|
|
open_lxde "$url"
|
|
;;
|
|
|
|
lxqt)
|
|
open_lxqt "$url"
|
|
;;
|
|
|
|
enlightenment)
|
|
open_enlightenment "$url"
|
|
;;
|
|
|
|
cygwin)
|
|
open_cygwin "$url"
|
|
;;
|
|
|
|
darwin)
|
|
open_darwin "$url"
|
|
;;
|
|
|
|
flatpak)
|
|
open_flatpak "$url"
|
|
;;
|
|
|
|
wsl)
|
|
open_wsl "$url"
|
|
;;
|
|
|
|
generic)
|
|
open_generic "$url"
|
|
;;
|
|
|
|
*)
|
|
exit_failure_operation_impossible "no method available for opening '$url'"
|
|
;;
|
|
esac
|