forked from oldratlee/useful-scripts
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathc
executable file
·153 lines (135 loc) · 3.6 KB
/
c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
#!/bin/bash
# @Function
# Run command and put output to system clipper.
#
# @Usage
# $ c echo 'hello world!'
# $ echo 'hello world!' | c
#
# @online-doc https://github.com/oldratlee/useful-scripts/blob/dev-2.x/docs/shell.md#-c
# @author Jerry Lee (oldratlee at gmail dot com)
#
# NOTE about Bash Traps and Pitfalls:
#
# 1. DO NOT combine var declaration and assignment which value supplied by subshell in ONE line!
# for example: readonly var1=$(echo value1)
# local var2=$(echo value1)
#
# Because the combination make exit code of assignment to be always 0,
# aka. the exit code of command in subshell is discarded.
# tested on bash 3.2.57/4.2.46
#
# solution is separation of var declaration and assignment:
# var1=$(echo value1)
# readonly var1
# local var2
# var2=$(echo value1)
set -eEuo pipefail
# NOTE: DO NOT declare var PROG as readonly in ONE line!
PROG="$(basename "$0")"
readonly PROG
readonly PROG_VERSION='2.5.0-dev'
################################################################################
# util functions
################################################################################
readonly ec=$'\033' # escape char
readonly eend=$'\033[0m' # escape end
readonly nl=$'\n' # new line
redEcho() {
[ -t 1 ] && echo "${ec}[1;31m$*$eend" || echo "$*"
}
usage() {
local -r exit_code="${1:-0}"
(($# > 0)) && shift
# shellcheck disable=SC2015
[ "$exit_code" != 0 ] && local -r out=/dev/stderr || local -r out=/dev/stdout
(($# > 0)) && redEcho "$*$nl" >$out
cat >$out <<EOF
Usage: ${PROG} [OPTION]... [command [command_args ...]]
Run command and put output to system clipper.
If no command is specified, read from stdin(pipe).
Example:
${PROG} echo "hello world!"
${PROG} grep -i 'hello world' menu.h main.c
set | ${PROG}
${PROG} -q < ~/.ssh/id_rsa.pub
Options:
-k, --keep-eol do not trim new line at end of file
-q, --quiet suppress all normal output, default is false
-h, --help display this help and exit
-V, --version display version information and exit
EOF
exit "$exit_code"
}
progVersion() {
echo "$PROG $PROG_VERSION"
exit
}
################################################################################
# parse options
################################################################################
quiet=false
eol=-n
declare -a args=()
while [ $# -gt 0 ]; do
case "$1" in
-k | --keep-eol)
eol=
shift
;;
-q | --quiet)
quiet=true
shift
;;
-h | --help)
usage
;;
-V | --version)
progVersion
;;
--)
shift
args=(${args[@]:+"${args[@]}"} "$@")
break
;;
-*)
usage 2 "${PROG}: unrecognized option '$1'"
;;
*)
# if not option, treat all follow args as command
args=(${args[@]:+"${args[@]}"} "$@")
break
;;
esac
done
readonly eol quiet args
################################################################################
# biz logic
################################################################################
copy() {
case "$(uname)" in
Darwin*)
pbcopy
;;
CYGWIN* | MINGW*)
clip
;;
*)
xsel -b
;;
esac
}
teeAndCopy() {
# shellcheck disable=SC2015
$quiet && local -r out=/dev/null || local -r out=/dev/stdout
tee >(
content="$(cat)"
# shellcheck disable=SC2086
echo $eol "$content" | copy
) >$out
}
if [ ${#args[@]} -eq 0 ]; then
teeAndCopy
else
"${args[@]}" | teeAndCopy
fi