podmanRunEasy 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304
  1. #!/usr/bin/env bash
  2. # shellcheck disable=SC1090,SC2004
  3. podmanRunEasy () {
  4. sourcePlugin "podman/podmanRunWrapper"
  5. ########################
  6. ####### DEFAULTS #######
  7. ########################
  8. # Can be overridden using environment variables
  9. [[ -z $_image ]] && _image="fedora:latest"
  10. [[ -z $_mode ]] && _mode="ephemeral"
  11. [[ -z $_workdir ]] && _workdir="$PWD"
  12. _smartDefaults () {
  13. echo ""
  14. ### UNDER CONSTRUCTION ###
  15. # I would like to handle some options automatically to reduce the number of configuration
  16. # options necessary to run podman and also make it easy for users to define defaults based
  17. # on file extension, command, shebang, etc
  18. # check the basename for defaults
  19. #[[ -z $COMMAND_DEF ]] \
  20. # && COMMAND_BASENAME="${COMMAND_ARR[0]##*/}" \
  21. # && _set_defaults "${COMMAND_BASENAME}"
  22. # check the extension for defaults
  23. #[[ -z $COMMAND_DEF ]] \
  24. # && COMMAND_EXT="${COMMAND_BASENAME##*.}" \
  25. # && _set_defaults "${COMMAND_EXT}"
  26. # add defaults to the string
  27. #if [[ -n $COMMAND_DEF ]]; then
  28. # [[ -z $_silent ]] && echo "Loaded command defaults, executing with \"${COMMAND_DEF}\""
  29. # COMMAND_STR="${COMMAND_DEF} ${COMMAND_STR}"
  30. #else
  31. # [[ -z $_silent ]] && echo "Could not find command defaults from name or extension"
  32. # [[ -z $_silent ]] && echo "Executing script directly"
  33. #fi
  34. }
  35. ########################
  36. ###### FUNCTIONS #######
  37. ########################
  38. _printHelpAndExit () {
  39. cat <<-'EOF'
  40. USAGE
  41. podman-run-easy [-m _mode] [-w PATH] [-d PATH] [-i _image] [--systemd] [--mkexec] [--help]
  42. [--silent] [--debug] [COMMANDS [ARGS...]]
  43. COMMANDS
  44. COMMANDS to run in the container (e.g. the current active file, an external
  45. build script, a program residing in the container, etc.)
  46. Can be empty (default entrypoint)
  47. OPTIONS
  48. --mode, -m MODE
  49. MODE can be any of the following:
  50. 1. ephemeral
  51. 2. persistent
  52. 3. recreate-persistent
  53. 4. remove-persistent
  54. --workdir, -w PATH
  55. The directory in the container where we execute the COMMANDS
  56. Default: The present working directory
  57. --maskdir, -d PATH
  58. Hide this directory from the host OS, store contents in the container only
  59. --image, -i IMAGE
  60. IMAGE used to create the container
  61. Default: fedora:latest
  62. --mkexec, -x
  63. Makes the first argument of COMMANDS executable
  64. --name, -n CONTAINER_NAME
  65. This will form the base of the container name and should be unique to each project
  66. Default: Container name will be set based on a concatenation of the image and commands
  67. --systemd
  68. Force container to init with systemd (--systemd=always)
  69. Default: --systemd=true (systemd will only start if CMD is systemd, /usr/sbin/init or
  70. /sbin/init)
  71. --array, -a ARRAY
  72. Read arguments from an existing or new ARRAY (bash >= 4.3)
  73. This is useful to reduce parsing errors and recommended for build-wrapper plugins
  74. --silent, -s
  75. Don't output anything from this program (container output will still be passed to stdout
  76. if -it option is used instead of -d, see `man podman run` for more information)
  77. --help, -h
  78. Print this help message and exit (overrides --silent)
  79. EOF
  80. # Exit using passed exit code
  81. [[ -z $1 ]] && exit 0 || exit "$1"
  82. }
  83. _parseInput () {
  84. [[ -n $_debug ]] && echo "$@"
  85. # Unset array variable when reparsing array
  86. unset _array
  87. # Parse input and set switches using getopt
  88. if _input=$(getopt -o +m:w:d:i:a:n:xsh -l mode:,workdir:,maskdir:,image:,array:,name:,mkexec,systemd,silent,debug,help -- "$@"); then
  89. eval set -- "$_input"
  90. while true; do
  91. case "$1" in
  92. --mode|-m)
  93. shift
  94. _mode="$1"
  95. ;;
  96. --workdir|-w)
  97. shift
  98. _workdir="$1"
  99. ;;
  100. --maskdir|-d)
  101. shift
  102. _maskdir="$1"
  103. ;;
  104. --image|-i)
  105. shift
  106. _image="$1"
  107. ;;
  108. --name|-n)
  109. shift
  110. _name="$1"
  111. ;;
  112. --array|-a)
  113. shift
  114. _array="$1"
  115. break
  116. ;;
  117. --mkexec|-x)
  118. _mkexec="true"
  119. ;;
  120. --systemd|-s)
  121. _systemd="true"
  122. ;;
  123. --silent)
  124. _silent="true"
  125. ;;
  126. --debug)
  127. _debug="true"
  128. echo "Debugging on!"
  129. ;;
  130. --help|-h)
  131. _printHelpAndExit 0
  132. ;;
  133. --)
  134. shift
  135. break
  136. ;;
  137. esac
  138. shift
  139. done
  140. else
  141. echo "Incorrect options provided"
  142. _printHelpAndExit 1
  143. fi
  144. # If array mode, load input array, reparse input, and return
  145. if [[ -n $_array ]]; then
  146. checkBashVersion
  147. local _n_array
  148. declare -n _n_array="$_array"
  149. _parseInput "${_n_array[@]}"
  150. return
  151. fi
  152. # Create _pre_commands_array from remaining arguments
  153. # shift getopt parameters away
  154. shift $((OPTIND - 1))
  155. # create array
  156. declare -ga _pre_commands_array
  157. _pre_commands_array=("$@")
  158. [[ -n $_debug ]] && echo "_pre_commands_array:" "${_pre_commands_array[@]}"
  159. # Sanitize mode numbers
  160. [[ "$_mode" == "1" ]] && _mode="ephemeral"
  161. [[ "$_mode" == "2" ]] && _mode="persistent"
  162. [[ "$_mode" == "3" ]] && _mode="recreate-persistent"
  163. [[ "$_mode" == "4" ]] && _mode="remove-persistent"
  164. # build the podman options array
  165. declare -ga _pre_options_array
  166. _pre_options_array+=("-it")
  167. #[[ "$_mode" == "ephemeral" ]] && _pre_options_array+=("--rm")
  168. _pre_options_array+=("-v" "${_workdir}:${_workdir}")
  169. _pre_options_array+=("-w" "${_workdir}")
  170. [[ -n $_maskdir ]] && _pre_options_array+=("-v" "${_maskdir}")
  171. [[ -n $_systemd ]] && _pre_options_array+=("--systemd=always")
  172. [[ -n $_debug ]] && echo "_pre_options_array:" "${_pre_options_array[@]}"
  173. }
  174. _makeExec () {
  175. # make executable
  176. if [[ -n $_mkexec ]]; then
  177. # assume script/program is the first argument in the command and break on whitespace
  178. _program="${_pre_commands_array[0]}"
  179. # make executable on the host
  180. chmod +x "${_program}"
  181. # make executable in the container
  182. #COMMAND="chmod 755 ${_program} && $COMMAND"
  183. fi
  184. }
  185. _sanityChecks () {
  186. # If missing, create workdir
  187. if ! [[ -d "$_workdir" ]]; then
  188. [[ -z $_silent ]] && echo "--workdir does not exist!"
  189. [[ -z $_silent ]] && echo "Creating ${_workdir} now..."
  190. mkdir -p "${_workdir}"
  191. fi
  192. # Grant container SELinux access to the _workdir
  193. fixPermissions "${_workdir}"
  194. }
  195. _runPodmanRunWrapper() {
  196. _makeExec
  197. local _prw_array
  198. _prw_array=("-m" "${_mode}" "--optionsarray" "_pre_options_array" "-i" "${_image}" "-n" "${_name}" "--commandsarray" "_pre_commands_array")
  199. [[ -n $_debug ]] && _prw_array+=("--debug")
  200. [[ -n $_silent ]] && _prw_array+=("--silent")
  201. if [[ -n $_debug ]]; then
  202. echo -n "PRE->PRW argument array: "
  203. printf '"%s" ' "${_prw_array[@]}"
  204. echo ""
  205. fi
  206. podmanRunWrapper -a _prw_array
  207. }
  208. #########################
  209. ####### EXECUTE #########
  210. #########################
  211. _execute () {
  212. # Get input
  213. _parseInput "$@"
  214. # Call podman-run-wrapper
  215. _runPodmanRunWrapper
  216. }
  217. # Allow this function to be executed directly
  218. _execute "$@"
  219. }
  220. # Allow this file to be executed directly if not being sourced
  221. if [[ "${BASH_SOURCE[0]}" == "${0}" ]]; then
  222. # The following functions are usually handled by build-wrapper
  223. _getBaseDir () {
  224. # Get base directory name of where this script resides
  225. # https://stackoverflow.com/questions/59895/how-to-get-the-source-directory-of-a-bash-script-from-within-the-script-itself#comment54598418_246128
  226. _basedir=$(dirname "$(readlink -f "$0")")
  227. }
  228. _sourceFunctions () {
  229. # Get the location of this file
  230. _getBaseDir
  231. # Go up two directories
  232. ff="${_basedir%/*/*}/functions"
  233. # Source functions file
  234. if [[ -f "$ff" ]]; then
  235. source "$ff"
  236. else
  237. echo "Cannot find functions file: ${ff}"
  238. fi
  239. }
  240. _sourceFunctions
  241. podmanRunEasy "$@"
  242. fi