Compare commits
17 Commits
Author | SHA1 | Date | |
---|---|---|---|
b10611f5ae | |||
6afb99d1ce | |||
0deb0b449c | |||
597c801bdf | |||
0347154d6b | |||
2470681dcb | |||
6cb89f0cd6 | |||
958ae3f264 | |||
303bdf5d71 | |||
d8b9b5dc01 | |||
7696ea5ce8 | |||
214ab03647 | |||
97b1be6732 | |||
545447c9b0 | |||
93ec32f793 | |||
02c7e5cb1a | |||
b32491ea9c |
22
README.md
22
README.md
@@ -1,10 +1,22 @@
|
|||||||
sx-open
|
# sx-open
|
||||||
=======
|
|
||||||
|
|
||||||
sx-open is an attemt to implement a saner alternative to xdg-open.
|
sx-open is an attempt to implement a saner alternative to xdg-open.
|
||||||
|
|
||||||
Installation
|
## Installation
|
||||||
------------
|
|
||||||
|
|
||||||
Clone the repo, drop sx-open.cfg into your $HOME/.config.
|
Clone the repo, drop sx-open.cfg into your $HOME/.config.
|
||||||
As this thing is meant to replace xdg-open, you will probably want to link sx-open into xdg-open somewhere in your $PATH as to override the default one.
|
As this thing is meant to replace xdg-open, you will probably want to link sx-open into xdg-open somewhere in your $PATH as to override the default one.
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
sx-open [-dhv] <uri/file>
|
||||||
|
Flags:
|
||||||
|
-d Dry run
|
||||||
|
-v Verbose
|
||||||
|
-h Help
|
||||||
|
|
||||||
|
### Exit codes
|
||||||
|
|
||||||
|
* 1 — action failed
|
||||||
|
* 2 — file not found
|
||||||
|
* 3 — no handlers found
|
||||||
|
132
sx-open
132
sx-open
@@ -1,51 +1,137 @@
|
|||||||
#!/usr/bin/env bash
|
#!/usr/bin/env bash
|
||||||
# This is an attempt to replace xdg-open with something sane.
|
# This is an attempt to replace xdg-open with something sane.
|
||||||
|
|
||||||
# Source the config file.
|
usage() { printf '%s [-dhv] <uri/file>\n' "${0##*/}"; }
|
||||||
cfg_file="$HOME/.config/sx-open.cfg"
|
|
||||||
[[ -f "$cfg_file" ]] && { source "$cfg_file"; }
|
|
||||||
|
|
||||||
usage() { echo "usage function not implemented yet."; }
|
usage() {
|
||||||
|
cat <<- EOF
|
||||||
|
sx-open [-dhv] <uri/file>
|
||||||
|
Flags:
|
||||||
|
-d Dry run
|
||||||
|
-v Verbose
|
||||||
|
-h Help
|
||||||
|
EOF
|
||||||
|
}
|
||||||
|
|
||||||
|
act() {
|
||||||
|
(( verbose )) && printf 'CMD: %s\n' "$*" >&2
|
||||||
|
(( dry_run )) || { "$@"; return $?; }
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
# handle_uri <res> <uri>
|
||||||
|
# 1: cmd failed
|
||||||
|
# 3: no handler
|
||||||
handle_uri() {
|
handle_uri() {
|
||||||
local target="$1"
|
declare -n result=$1
|
||||||
|
declare h cmd regex target=$2
|
||||||
|
|
||||||
|
for h in "${uri_handlers[@]}"; do
|
||||||
|
IFS='=' read cmd regex <<< "$h"
|
||||||
|
|
||||||
|
if [[ "$target" =~ ${regex} ]]; then
|
||||||
|
act ${cmd} "$target"; result=$?
|
||||||
|
(( result )) && return 1
|
||||||
|
|
||||||
for h in "${!uri_handlers[@]}"; do
|
|
||||||
grep -oE "${uri_handlers[${h}]}" &>/dev/null <<< "$target" && {
|
|
||||||
${h} "$target" &
|
|
||||||
return 0
|
return 0
|
||||||
}
|
fi
|
||||||
done
|
done
|
||||||
|
|
||||||
return 1
|
return 3
|
||||||
}
|
}
|
||||||
|
|
||||||
handle_fs_target() {
|
# handle_file <res> <file>
|
||||||
local target="${1##*file://}"
|
# 1: cmd failed
|
||||||
|
# 2: no such file
|
||||||
|
# 3: no handler
|
||||||
|
handle_file() {
|
||||||
|
declare -n result=$1
|
||||||
|
declare m \
|
||||||
|
target_mimetype charset \
|
||||||
|
cmd regex \
|
||||||
|
target=$2
|
||||||
|
|
||||||
target_mimetype=$(file -ib "$target")
|
[[ -e "$target" ]] || return 3
|
||||||
|
|
||||||
|
IFS=';' read target_mimetype charset <<< $( file -ib "$target" )
|
||||||
|
|
||||||
|
for m in "${mime_handlers[@]}"; do
|
||||||
|
IFS='=' read cmd regex <<< "$m"
|
||||||
|
|
||||||
|
if [[ "$target_mimetype" =~ ${regex} ]]; then
|
||||||
|
act ${cmd} "$target"; result=$?
|
||||||
|
(( result )) && return 1
|
||||||
|
|
||||||
for m in "${!mime_handlers[@]}"; do
|
|
||||||
grep -oE "${mime_handlers[${m}]}" &>/dev/null <<< "$target_mimetype" && {
|
|
||||||
${m} "$target" &
|
|
||||||
return 0
|
return 0
|
||||||
}
|
fi
|
||||||
done
|
done
|
||||||
|
|
||||||
return 1
|
return 3
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# DSL
|
||||||
|
uri() { uri_handlers+=( "$1=$2" ); }
|
||||||
|
mime() { mime_handlers+=( "$1=$2" ); }
|
||||||
|
|
||||||
|
is_uri() [[ $1 =~ ^[a-zA-Z][a-zA-Z0-9\+\.\-]+:.+ ]]
|
||||||
|
|
||||||
main() {
|
main() {
|
||||||
target="$1"
|
declare cmd_result target
|
||||||
|
|
||||||
|
# Source the config file.
|
||||||
|
cfg_file="$HOME/.config/sx-open.cfg"
|
||||||
|
[[ -f "$cfg_file" ]] && { source "$cfg_file"; }
|
||||||
|
|
||||||
|
while (( $# )); do
|
||||||
|
case $1 in
|
||||||
|
(-d)
|
||||||
|
printf 'Dry run: not actually running the handler\n' >&2
|
||||||
|
|
||||||
|
dry_run=1
|
||||||
|
verbose=1
|
||||||
|
;;
|
||||||
|
|
||||||
|
(-h) usage; return 0;;
|
||||||
|
|
||||||
|
(-v) verbose=1;;
|
||||||
|
|
||||||
|
(--) shift; break;;
|
||||||
|
(*) break;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
shift
|
||||||
|
done
|
||||||
|
|
||||||
|
target=$1
|
||||||
|
|
||||||
[[ "$target" ]] || { usage; exit; }
|
[[ "$target" ]] || { usage; exit; }
|
||||||
|
|
||||||
if [[ -e "$target" || "$target" == 'file://'* ]]; then
|
# Treat file:// as local paths.
|
||||||
handle_fs_target "$target"
|
[[ "$target" =~ ^file:(//)?(/.+) ]] && target=${BASH_REMATCH[2]}
|
||||||
|
|
||||||
|
if [[ -e "$target" ]]; then
|
||||||
|
[[ "$target" =~ ^/.* ]] || { target="${PWD}/${target}"; } # Turn relative paths to absolute ones.
|
||||||
|
|
||||||
|
handle_file cmd_result "$target"
|
||||||
|
elif is_uri "$target"; then
|
||||||
|
handle_uri cmd_result "$target"
|
||||||
else
|
else
|
||||||
handle_uri "$target"
|
printf 'No such file or directory: %s\n' "$target" >&2
|
||||||
|
return 2
|
||||||
fi
|
fi
|
||||||
|
|
||||||
[[ "$?" -gt 0 ]] && { echo "No handlers found for $target"; }
|
case $? in
|
||||||
|
(1)
|
||||||
|
printf 'Action failed with exit code: %s\n' "$cmd_result" >&2
|
||||||
|
return 4
|
||||||
|
;;
|
||||||
|
|
||||||
|
(3)
|
||||||
|
printf 'No handlers found\n' >&2
|
||||||
|
return 3
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
18
sx-open.cfg
18
sx-open.cfg
@@ -1,16 +1,12 @@
|
|||||||
|
#!syntax bash
|
||||||
# Configuration file for sx-open
|
# Configuration file for sx-open
|
||||||
# Note that as sx-open checks the regexes in order, they should be placed in order from specific to less so.
|
# Note that as sx-open checks the regexes in order, they should be placed in order from specific to less so.
|
||||||
|
# Format: type cmd regex
|
||||||
|
|
||||||
declare -A uri_handlers
|
# Examples:
|
||||||
declare -A mime_handlers
|
|
||||||
|
|
||||||
uri_handlers=(
|
#uri browser '^https?:'
|
||||||
["steam"]='^steam://.+'
|
#uri steam '^steam:'
|
||||||
["browser"]='.+'
|
|
||||||
)
|
|
||||||
|
|
||||||
mime_handlers=(
|
#mime sxiv '^image/'
|
||||||
["sxiv"]='image/.+'
|
#mime libreoffice '^application/msword$'
|
||||||
)
|
|
||||||
|
|
||||||
# vim: syntax=sh
|
|
||||||
|
Reference in New Issue
Block a user