ttar 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. #!/usr/bin/env bash
  2. # Purpose: plain text tar format
  3. # Limitations: - only suitable for text files, directories, and symlinks
  4. # - stores only filename, content, and mode
  5. # - not designed for untrusted input
  6. # Note: must work with bash version 3.2 (macOS)
  7. set -o errexit -o nounset
  8. # Sanitize environment (for instance, standard sorting of glob matches)
  9. export LC_ALL=C
  10. path=""
  11. CMD=""
  12. function usage {
  13. bname=$(basename "$0")
  14. cat << USAGE
  15. Usage: $bname [-C <DIR>] -c -f <ARCHIVE> <FILE...> (create archive)
  16. $bname -t -f <ARCHIVE> (list archive contents)
  17. $bname [-C <DIR>] -x -f <ARCHIVE> (extract archive)
  18. Options:
  19. -C <DIR> (change directory)
  20. Example: Change to sysfs directory, create ttar file from fixtures directory
  21. $bname -C sysfs -c -f sysfs/fixtures.ttar fixtures/
  22. USAGE
  23. exit "$1"
  24. }
  25. function vecho {
  26. if [ "${VERBOSE:-}" == "yes" ]; then
  27. echo >&7 "$@"
  28. fi
  29. }
  30. function set_cmd {
  31. if [ -n "$CMD" ]; then
  32. echo "ERROR: more than one command given"
  33. echo
  34. usage 2
  35. fi
  36. CMD=$1
  37. }
  38. while getopts :cf:htxvC: opt; do
  39. case $opt in
  40. c)
  41. set_cmd "create"
  42. ;;
  43. f)
  44. ARCHIVE=$OPTARG
  45. ;;
  46. h)
  47. usage 0
  48. ;;
  49. t)
  50. set_cmd "list"
  51. ;;
  52. x)
  53. set_cmd "extract"
  54. ;;
  55. v)
  56. VERBOSE=yes
  57. exec 7>&1
  58. ;;
  59. C)
  60. CDIR=$OPTARG
  61. ;;
  62. *)
  63. echo >&2 "ERROR: invalid option -$OPTARG"
  64. echo
  65. usage 1
  66. ;;
  67. esac
  68. done
  69. # Remove processed options from arguments
  70. shift $(( OPTIND - 1 ));
  71. if [ "${CMD:-}" == "" ]; then
  72. echo >&2 "ERROR: no command given"
  73. echo
  74. usage 1
  75. elif [ "${ARCHIVE:-}" == "" ]; then
  76. echo >&2 "ERROR: no archive name given"
  77. echo
  78. usage 1
  79. fi
  80. function list {
  81. local path=""
  82. local size=0
  83. local line_no=0
  84. local ttar_file=$1
  85. if [ -n "${2:-}" ]; then
  86. echo >&2 "ERROR: too many arguments."
  87. echo
  88. usage 1
  89. fi
  90. if [ ! -e "$ttar_file" ]; then
  91. echo >&2 "ERROR: file not found ($ttar_file)"
  92. echo
  93. usage 1
  94. fi
  95. while read -r line; do
  96. line_no=$(( line_no + 1 ))
  97. if [ $size -gt 0 ]; then
  98. size=$(( size - 1 ))
  99. continue
  100. fi
  101. if [[ $line =~ ^Path:\ (.*)$ ]]; then
  102. path=${BASH_REMATCH[1]}
  103. elif [[ $line =~ ^Lines:\ (.*)$ ]]; then
  104. size=${BASH_REMATCH[1]}
  105. echo "$path"
  106. elif [[ $line =~ ^Directory:\ (.*)$ ]]; then
  107. path=${BASH_REMATCH[1]}
  108. echo "$path/"
  109. elif [[ $line =~ ^SymlinkTo:\ (.*)$ ]]; then
  110. echo "$path -> ${BASH_REMATCH[1]}"
  111. fi
  112. done < "$ttar_file"
  113. }
  114. function extract {
  115. local path=""
  116. local size=0
  117. local line_no=0
  118. local ttar_file=$1
  119. if [ -n "${2:-}" ]; then
  120. echo >&2 "ERROR: too many arguments."
  121. echo
  122. usage 1
  123. fi
  124. if [ ! -e "$ttar_file" ]; then
  125. echo >&2 "ERROR: file not found ($ttar_file)"
  126. echo
  127. usage 1
  128. fi
  129. while IFS= read -r line; do
  130. line_no=$(( line_no + 1 ))
  131. if [ "$size" -gt 0 ]; then
  132. echo "$line" >> "$path"
  133. size=$(( size - 1 ))
  134. continue
  135. fi
  136. if [[ $line =~ ^Path:\ (.*)$ ]]; then
  137. path=${BASH_REMATCH[1]}
  138. if [ -e "$path" ] || [ -L "$path" ]; then
  139. rm "$path"
  140. fi
  141. elif [[ $line =~ ^Lines:\ (.*)$ ]]; then
  142. size=${BASH_REMATCH[1]}
  143. # Create file even if it is zero-length.
  144. touch "$path"
  145. vecho " $path"
  146. elif [[ $line =~ ^Mode:\ (.*)$ ]]; then
  147. mode=${BASH_REMATCH[1]}
  148. chmod "$mode" "$path"
  149. vecho "$mode"
  150. elif [[ $line =~ ^Directory:\ (.*)$ ]]; then
  151. path=${BASH_REMATCH[1]}
  152. mkdir -p "$path"
  153. vecho " $path/"
  154. elif [[ $line =~ ^SymlinkTo:\ (.*)$ ]]; then
  155. ln -s "${BASH_REMATCH[1]}" "$path"
  156. vecho " $path -> ${BASH_REMATCH[1]}"
  157. elif [[ $line =~ ^# ]]; then
  158. # Ignore comments between files
  159. continue
  160. else
  161. echo >&2 "ERROR: Unknown keyword on line $line_no: $line"
  162. exit 1
  163. fi
  164. done < "$ttar_file"
  165. }
  166. function div {
  167. echo "# ttar - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -" \
  168. "- - - - - -"
  169. }
  170. function get_mode {
  171. local mfile=$1
  172. if [ -z "${STAT_OPTION:-}" ]; then
  173. if stat -c '%a' "$mfile" >/dev/null 2>&1; then
  174. STAT_OPTION='-c'
  175. STAT_FORMAT='%a'
  176. else
  177. STAT_OPTION='-f'
  178. STAT_FORMAT='%A'
  179. fi
  180. fi
  181. stat "${STAT_OPTION}" "${STAT_FORMAT}" "$mfile"
  182. }
  183. function _create {
  184. shopt -s nullglob
  185. local mode
  186. while (( "$#" )); do
  187. file=$1
  188. if [ -L "$file" ]; then
  189. echo "Path: $file"
  190. symlinkTo=$(readlink "$file")
  191. echo "SymlinkTo: $symlinkTo"
  192. vecho " $file -> $symlinkTo"
  193. div
  194. elif [ -d "$file" ]; then
  195. # Strip trailing slash (if there is one)
  196. file=${file%/}
  197. echo "Directory: $file"
  198. mode=$(get_mode "$file")
  199. echo "Mode: $mode"
  200. vecho "$mode $file/"
  201. div
  202. # Find all files and dirs, including hidden/dot files
  203. for x in "$file/"{*,.[^.]*}; do
  204. _create "$x"
  205. done
  206. elif [ -f "$file" ]; then
  207. echo "Path: $file"
  208. lines=$(wc -l "$file"|awk '{print $1}')
  209. echo "Lines: $lines"
  210. cat "$file"
  211. mode=$(get_mode "$file")
  212. echo "Mode: $mode"
  213. vecho "$mode $file"
  214. div
  215. else
  216. echo >&2 "ERROR: file not found ($file in $(pwd))"
  217. exit 2
  218. fi
  219. shift
  220. done
  221. }
  222. function create {
  223. ttar_file=$1
  224. shift
  225. if [ -z "${1:-}" ]; then
  226. echo >&2 "ERROR: missing arguments."
  227. echo
  228. usage 1
  229. fi
  230. if [ -e "$ttar_file" ]; then
  231. rm "$ttar_file"
  232. fi
  233. exec > "$ttar_file"
  234. _create "$@"
  235. }
  236. if [ -n "${CDIR:-}" ]; then
  237. if [[ "$ARCHIVE" != /* ]]; then
  238. # Relative path: preserve the archive's location before changing
  239. # directory
  240. ARCHIVE="$(pwd)/$ARCHIVE"
  241. fi
  242. cd "$CDIR"
  243. fi
  244. "$CMD" "$ARCHIVE" "$@"