123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328 |
- #!/bin/bash
- # This Source Code Form is subject to the terms of the Mozilla Public
- # License, v. 2.0. If a copy of the MPL was not distributed with this
- # file, You can obtain one at http://mozilla.org/MPL/2.0/.
- #
- # This tool generates incremental update packages for the update system.
- # Author: Darin Fisher
- #
- . $(dirname "$0")/common.sh
- # -----------------------------------------------------------------------------
- print_usage() {
- notice "Usage: $(basename $0) [OPTIONS] ARCHIVE FROMDIR TODIR"
- notice ""
- notice "The differences between FROMDIR and TODIR will be stored in ARCHIVE."
- notice ""
- notice "Options:"
- notice " -h show this help text"
- notice " -f clobber this file in the installation"
- notice " Must be a path to a file to clobber in the partial update."
- notice ""
- }
- check_for_forced_update() {
- force_list="$1"
- forced_file_chk="$2"
- local f
- if [ "$forced_file_chk" = "precomplete" ]; then
- ## "true" *giggle*
- return 0;
- fi
- if [ "$forced_file_chk" = "Contents/Resources/precomplete" ]; then
- ## "true" *giggle*
- return 0;
- fi
- if [ "$forced_file_chk" = "removed-files" ]; then
- ## "true" *giggle*
- return 0;
- fi
- if [ "$forced_file_chk" = "Contents/Resources/removed-files" ]; then
- ## "true" *giggle*
- return 0;
- fi
- if [ "$forced_file_chk" = "chrome.manifest" ]; then
- ## "true" *giggle*
- return 0;
- fi
- if [ "$forced_file_chk" = "Contents/Resources/chrome.manifest" ]; then
- ## "true" *giggle*
- return 0;
- fi
- if [ "${forced_file_chk##*.}" = "chk" ]; then
- ## "true" *giggle*
- return 0;
- fi
- for f in $force_list; do
- #echo comparing $forced_file_chk to $f
- if [ "$forced_file_chk" = "$f" ]; then
- ## "true" *giggle*
- return 0;
- fi
- done
- ## 'false'... because this is bash. Oh yay!
- return 1;
- }
- if [ $# = 0 ]; then
- print_usage
- exit 1
- fi
- requested_forced_updates='Contents/MacOS/firefox'
- while getopts "hf:" flag
- do
- case "$flag" in
- h) print_usage; exit 0
- ;;
- f) requested_forced_updates="$requested_forced_updates $OPTARG"
- ;;
- ?) print_usage; exit 1
- ;;
- esac
- done
- # -----------------------------------------------------------------------------
- let arg_start=$OPTIND-1
- shift $arg_start
- archive="$1"
- olddir="$2"
- newdir="$3"
- # Prevent the workdir from being inside the targetdir so it isn't included in
- # the update mar.
- if [ $(echo "$newdir" | grep -c '\/$') = 1 ]; then
- # Remove the /
- newdir=$(echo "$newdir" | sed -e 's:\/$::')
- fi
- workdir="$newdir.work"
- updatemanifestv2="$workdir/updatev2.manifest"
- updatemanifestv3="$workdir/updatev3.manifest"
- archivefiles="updatev2.manifest updatev3.manifest"
- mkdir -p "$workdir"
- # Generate a list of all files in the target directory.
- pushd "$olddir"
- if test $? -ne 0 ; then
- exit 1
- fi
- list_files oldfiles
- list_dirs olddirs
- popd
- pushd "$newdir"
- if test $? -ne 0 ; then
- exit 1
- fi
- if [ ! -f "precomplete" ]; then
- if [ ! -f "Contents/Resources/precomplete" ]; then
- notice "precomplete file is missing!"
- exit 1
- fi
- fi
- list_dirs newdirs
- list_files newfiles
- popd
- # Add the type of update to the beginning of the update manifests.
- notice ""
- notice "Adding type instruction to update manifests"
- > $updatemanifestv2
- > $updatemanifestv3
- notice " type partial"
- echo "type \"partial\"" >> $updatemanifestv2
- echo "type \"partial\"" >> $updatemanifestv3
- notice ""
- notice "Adding file patch and add instructions to update manifests"
- num_oldfiles=${#oldfiles[*]}
- remove_array=
- num_removes=0
- for ((i=0; $i<$num_oldfiles; i=$i+1)); do
- f="${oldfiles[$i]}"
- # If this file exists in the new directory as well, then check if it differs.
- if [ -f "$newdir/$f" ]; then
- if check_for_add_if_not_update "$f"; then
- # The full workdir may not exist yet, so create it if necessary.
- mkdir -p `dirname "$workdir/$f"`
- $BZIP2 -cz9 "$newdir/$f" > "$workdir/$f"
- copy_perm "$newdir/$f" "$workdir/$f"
- make_add_if_not_instruction "$f" "$updatemanifestv3"
- archivefiles="$archivefiles \"$f\""
- continue 1
- fi
- if check_for_forced_update "$requested_forced_updates" "$f"; then
- # The full workdir may not exist yet, so create it if necessary.
- mkdir -p `dirname "$workdir/$f"`
- $BZIP2 -cz9 "$newdir/$f" > "$workdir/$f"
- copy_perm "$newdir/$f" "$workdir/$f"
- make_add_instruction "$f" "$updatemanifestv2" "$updatemanifestv3" 1
- archivefiles="$archivefiles \"$f\""
- continue 1
- fi
- if ! diff "$olddir/$f" "$newdir/$f" > /dev/null; then
- # Compute both the compressed binary diff and the compressed file, and
- # compare the sizes. Then choose the smaller of the two to package.
- dir=$(dirname "$workdir/$f")
- mkdir -p "$dir"
- notice "diffing \"$f\""
- # MBSDIFF_HOOK represents the communication interface with funsize and,
- # if enabled, caches the intermediate patches for future use and
- # compute avoidance
- #
- # An example of MBSDIFF_HOOK env variable could look like this:
- # export MBSDIFF_HOOK="myscript.sh -A https://funsize/api -c /home/user"
- # where myscript.sh has the following usage:
- # myscript.sh -A SERVER-URL [-c LOCAL-CACHE-DIR-PATH] [-g] [-u] \
- # PATH-FROM-URL PATH-TO-URL PATH-PATCH SERVER-URL
- #
- # Note: patches are bzipped stashed in funsize to gain more speed
- # if service is not enabled then default to old behavior
- if [ -z "$MBSDIFF_HOOK" ]; then
- $MBSDIFF "$olddir/$f" "$newdir/$f" "$workdir/$f.patch"
- $BZIP2 -z9 "$workdir/$f.patch"
- else
- # if service enabled then check patch existence for retrieval
- if $MBSDIFF_HOOK -g "$olddir/$f" "$newdir/$f" "$workdir/$f.patch.bz2"; then
- notice "file \"$f\" found in funsize, diffing skipped"
- else
- # if not found already - compute it and cache it for future use
- $MBSDIFF "$olddir/$f" "$newdir/$f" "$workdir/$f.patch"
- $BZIP2 -z9 "$workdir/$f.patch"
- $MBSDIFF_HOOK -u "$olddir/$f" "$newdir/$f" "$workdir/$f.patch.bz2"
- fi
- fi
- $BZIP2 -cz9 "$newdir/$f" > "$workdir/$f"
- copy_perm "$newdir/$f" "$workdir/$f"
- patchfile="$workdir/$f.patch.bz2"
- patchsize=$(get_file_size "$patchfile")
- fullsize=$(get_file_size "$workdir/$f")
- if [ $patchsize -lt $fullsize ]; then
- make_patch_instruction "$f" "$updatemanifestv2" "$updatemanifestv3"
- mv -f "$patchfile" "$workdir/$f.patch"
- rm -f "$workdir/$f"
- archivefiles="$archivefiles \"$f.patch\""
- else
- make_add_instruction "$f" "$updatemanifestv2" "$updatemanifestv3"
- rm -f "$patchfile"
- archivefiles="$archivefiles \"$f\""
- fi
- fi
- else
- # remove instructions are added after add / patch instructions for
- # consistency with make_incremental_updates.py
- remove_array[$num_removes]=$f
- (( num_removes++ ))
- fi
- done
- # Newly added files
- notice ""
- notice "Adding file add instructions to update manifests"
- num_newfiles=${#newfiles[*]}
- for ((i=0; $i<$num_newfiles; i=$i+1)); do
- f="${newfiles[$i]}"
- # If we've already tested this file, then skip it
- for ((j=0; $j<$num_oldfiles; j=$j+1)); do
- if [ "$f" = "${oldfiles[j]}" ]; then
- continue 2
- fi
- done
- dir=$(dirname "$workdir/$f")
- mkdir -p "$dir"
- $BZIP2 -cz9 "$newdir/$f" > "$workdir/$f"
- copy_perm "$newdir/$f" "$workdir/$f"
- if check_for_add_if_not_update "$f"; then
- make_add_if_not_instruction "$f" "$updatemanifestv3"
- else
- make_add_instruction "$f" "$updatemanifestv2" "$updatemanifestv3"
- fi
- archivefiles="$archivefiles \"$f\""
- done
- notice ""
- notice "Adding file remove instructions to update manifests"
- for ((i=0; $i<$num_removes; i=$i+1)); do
- f="${remove_array[$i]}"
- notice " remove \"$f\""
- echo "remove \"$f\"" >> $updatemanifestv2
- echo "remove \"$f\"" >> $updatemanifestv3
- done
- # Add remove instructions for any dead files.
- notice ""
- notice "Adding file and directory remove instructions from file 'removed-files'"
- append_remove_instructions "$newdir" "$updatemanifestv2" "$updatemanifestv3"
- notice ""
- notice "Adding directory remove instructions for directories that no longer exist"
- num_olddirs=${#olddirs[*]}
- for ((i=0; $i<$num_olddirs; i=$i+1)); do
- f="${olddirs[$i]}"
- # If this dir doesn't exist in the new directory remove it.
- if [ ! -d "$newdir/$f" ]; then
- notice " rmdir $f/"
- echo "rmdir \"$f/\"" >> $updatemanifestv2
- echo "rmdir \"$f/\"" >> $updatemanifestv3
- fi
- done
- $BZIP2 -z9 "$updatemanifestv2" && mv -f "$updatemanifestv2.bz2" "$updatemanifestv2"
- $BZIP2 -z9 "$updatemanifestv3" && mv -f "$updatemanifestv3.bz2" "$updatemanifestv3"
- mar_command="$MAR"
- if [[ -n $MOZ_PRODUCT_VERSION ]]
- then
- mar_command="$mar_command -V $MOZ_PRODUCT_VERSION"
- fi
- if [[ -n $MOZ_CHANNEL_ID ]]
- then
- mar_command="$mar_command -H $MOZ_CHANNEL_ID"
- fi
- mar_command="$mar_command -C \"$workdir\" -c output.mar"
- eval "$mar_command $archivefiles"
- mv -f "$workdir/output.mar" "$archive"
- # cleanup
- rm -fr "$workdir"
- notice ""
- notice "Finished"
- notice ""
|