forked from ungleich-public/cdist
		
	Merge branch 'fix/type/__package_opkg/lock' into 'master'
__package_opkg: Add locking See merge request ungleich-public/cdist!896
This commit is contained in:
		
				commit
				
					
						684043bf37
					
				
			
		
					 2 changed files with 109 additions and 41 deletions
				
			
		|  | @ -1,7 +1,8 @@ | ||||||
| #!/bin/sh | #!/bin/sh -e | ||||||
| # | # | ||||||
| # 2011 Nico Schottelius (nico-cdist at schottelius.org) | # 2011 Nico Schottelius (nico-cdist at schottelius.org) | ||||||
| # 2012 Giel van Schijndel (giel plus cdist at mortis dot eu) | # 2012 Giel van Schijndel (giel plus cdist at mortis dot eu) | ||||||
|  | # 2020 Dennis Camera (dennis.camera at ssrq-sds-fds.ch) | ||||||
| # | # | ||||||
| # This file is part of cdist. | # This file is part of cdist. | ||||||
| # | # | ||||||
|  | @ -19,21 +20,78 @@ | ||||||
| # along with cdist. If not, see <http://www.gnu.org/licenses/>. | # along with cdist. If not, see <http://www.gnu.org/licenses/>. | ||||||
| # | # | ||||||
| # | # | ||||||
| # Retrieve the status of a package - parsed opkg output | # Retrieve the status of a package - parses opkg output | ||||||
| # | # | ||||||
| 
 | 
 | ||||||
| if [ -f "$__object/parameter/name" ]; then | readonly __type_path=${__object%%${__object_id}*} | ||||||
|    name="$(cat "$__object/parameter/name")" | test -d "${__type_path}" || { echo 'Cannot determine __type_path' >&2; exit 1; } | ||||||
|  | readonly LOCKFILE="${__type_path:?}/.cdist_opkg.lock" | ||||||
|  | 
 | ||||||
|  | if command -v flock >/dev/null 2>&1 | ||||||
|  | then | ||||||
|  | 	# use flock (if available) on FD 9 | ||||||
|  | 	_lock() { | ||||||
|  | 		exec 9<>"${LOCKFILE:?}" | ||||||
|  | 		flock -x 9 | ||||||
|  | 		echo $$>&9 | ||||||
|  | 	} | ||||||
|  | 	_unlock() { | ||||||
|  | 		:>"${LOCKFILE:?}" | ||||||
|  | 		flock -u 9 | ||||||
|  | 		exec 9<&- | ||||||
|  | 	} | ||||||
| else | else | ||||||
|    name="$__object_id" | 	# fallback to mkdir if flock is missing | ||||||
|  | 	_lock() { | ||||||
|  | 		until mkdir "${LOCKFILE:?}.dir" 2>/dev/null | ||||||
|  | 		do | ||||||
|  | 			while test -d "${LOCKFILE}.dir" | ||||||
|  | 			do | ||||||
|  | 				# DEBUG: | ||||||
|  | 				# printf 'Locked by PID: %u\n' "$(cat "${LOCKFILE}.dir/pid")" | ||||||
|  | 				sleep 1 | ||||||
|  | 			done | ||||||
|  | 		done | ||||||
|  | 		echo $$ >"${LOCKFILE:?}.dir/pid" | ||||||
|  | 	} | ||||||
|  | 	_unlock() { | ||||||
|  | 		test -d "${LOCKFILE}.dir" || return 0 | ||||||
|  | 		if test -s "${LOCKFILE}.dir/pid" | ||||||
|  | 		then | ||||||
|  | 			test "$(cat "${LOCKFILE}.dir/pid")" = $$ || return 1 | ||||||
|  | 			rm "${LOCKFILE:?}.dir/pid" | ||||||
|  | 		fi | ||||||
|  | 		rmdir "${LOCKFILE:?}.dir" | ||||||
|  | 	} | ||||||
| fi | fi | ||||||
| 
 | 
 | ||||||
| # Except dpkg failing, if package is not known / installed | 
 | ||||||
| if opkg status "$name" 2>/dev/null | grep -q "^Status: install user installed$"; then | if test -f "${__object}/parameter/name" | ||||||
|    echo "present" | then | ||||||
|    exit 0 | 	pkg_name=$(cat "${__object}/parameter/name") | ||||||
| elif [ "$(opkg info "$name" 2> /dev/null | wc -l)" -eq 0 ]; then | else | ||||||
|    echo "absent notpresent" | 	pkg_name=$__object_id | ||||||
|    exit 0 | fi | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # NOTE: We need to lock parallel execution of type explorers and code-remote | ||||||
|  | # because opkg will try to acquire the OPKG lock (usually /var/lock/opkg.lock) | ||||||
|  | # using lockf(2) for every operation. | ||||||
|  | # It will not wait for the lock but terminate with an error. | ||||||
|  | # This leads to incorrect 'absent notpresent' statuses when parallel execution | ||||||
|  | # is enabled. | ||||||
|  | trap _unlock EXIT | ||||||
|  | _lock | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # Except opkg failing, if package is not known / installed | ||||||
|  | if opkg status "${pkg_name}" 2>/dev/null \ | ||||||
|  | 	| grep -q -e '^Status: [^ ][^ ]* [^ ][^ ]* installed$' | ||||||
|  | then | ||||||
|  | 	echo 'present' | ||||||
|  | elif opkg info "${pkg_name}" 2>/dev/null | grep -q . | ||||||
|  | then | ||||||
|  | 	echo 'absent notpresent' | ||||||
|  | else | ||||||
|  | 	echo 'absent' | ||||||
| fi | fi | ||||||
| echo "absent" |  | ||||||
|  |  | ||||||
|  | @ -2,6 +2,7 @@ | ||||||
| # | # | ||||||
| # 2011,2013 Nico Schottelius (nico-cdist at schottelius.org) | # 2011,2013 Nico Schottelius (nico-cdist at schottelius.org) | ||||||
| # 2012 Giel van Schijndel (giel plus cdist at mortis dot eu) | # 2012 Giel van Schijndel (giel plus cdist at mortis dot eu) | ||||||
|  | # 2020 Dennis Camera (dennis.camera at ssrq-sds-fds.ch) | ||||||
| # | # | ||||||
| # This file is part of cdist. | # This file is part of cdist. | ||||||
| # | # | ||||||
|  | @ -19,41 +20,50 @@ | ||||||
| # along with cdist. If not, see <http://www.gnu.org/licenses/>. | # along with cdist. If not, see <http://www.gnu.org/licenses/>. | ||||||
| # | # | ||||||
| # | # | ||||||
| # Manage packages on OpenWRT and co. | # Manage packages on OpenWrt, optware, and co. | ||||||
| # | # | ||||||
| 
 | 
 | ||||||
| if [ -f "$__object/parameter/name" ]; then | if test -f "${__object}/parameter/name" | ||||||
|    name="$(cat "$__object/parameter/name")" | then | ||||||
|  | 	name=$(cat "${__object}/parameter/name") | ||||||
| else | else | ||||||
|    name="$__object_id" | 	name=$__object_id | ||||||
| fi | fi | ||||||
| 
 | 
 | ||||||
| state_should="$(cat "$__object/parameter/state")" | state_should=$(cat "${__object}/parameter/state") | ||||||
|  | state_is=$(cat "${__object}/explorer/pkg_status") | ||||||
| 
 | 
 | ||||||
| state_is="$(cat "$__object/explorer/pkg_status")" | case $state_is | ||||||
| case "$state_is" in | in | ||||||
|     absent*) | 	(absent*) | ||||||
|        present="$(echo "$state_is" | cut -d ' ' -f 2)" | 		presence=$(echo "${state_is}" | cut -d ' ' -f 2) | ||||||
|        state_is="absent" | 		state_is='absent' | ||||||
| 		;; | 		;; | ||||||
| esac | esac | ||||||
| 
 | 
 | ||||||
| [ "$state_is" = "$state_should" ] && exit 0 | if test "${state_is}" = "${state_should}" | ||||||
|  | then | ||||||
|  | 	exit 0 | ||||||
|  | fi | ||||||
| 
 | 
 | ||||||
| case "$state_should" in | 
 | ||||||
|     present) | case $state_should | ||||||
|         if [ "$present" = "notpresent" ]; then | in | ||||||
|             echo "opkg --verbosity=0 update" | 	(present) | ||||||
|  | 		if test "${presence}" = 'notpresent' | ||||||
|  | 		then | ||||||
|  | 			echo 'opkg --verbosity=0 update' | ||||||
| 		fi | 		fi | ||||||
|         echo "opkg --verbosity=0 install '$name'" | 
 | ||||||
|         echo "installed" >> "$__messages_out" | 		printf "opkg --verbosity=0 install '%s'\n" "${name}" | ||||||
|  | 		echo 'installed' >>"${__messages_out}" | ||||||
| 		;; | 		;; | ||||||
|     absent) | 	(absent) | ||||||
|         echo "opkg --verbosity=0 remove '$name'" | 		printf "opkg --verbosity=0 remove '%s'" "${name}" | ||||||
|         echo "removed" >> "$__messages_out" | 		echo 'removed' >>"${__messages_out}" | ||||||
| 		;; | 		;; | ||||||
|     *) | 	(*) | ||||||
|         echo "Unknown state: ${state_should}" >&2 | 		printf 'Unknown state: %s\n' "${state_should}" >&2 | ||||||
| 		exit 1 | 		exit 1 | ||||||
| 		;; | 		;; | ||||||
| esac | esac | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue