| 12
 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
 154
 155
 156
 157
 158
 159
 160
 161
 162
 163
 164
 165
 166
 167
 168
 169
 170
 171
 172
 173
 174
 175
 176
 177
 178
 179
 180
 181
 182
 183
 184
 185
 186
 187
 188
 189
 190
 191
 192
 193
 194
 195
 196
 197
 198
 199
 200
 201
 202
 203
 204
 205
 206
 207
 208
 209
 210
 211
 212
 213
 214
 215
 216
 217
 218
 219
 220
 221
 222
 223
 224
 225
 226
 227
 228
 229
 230
 231
 232
 233
 234
 235
 236
 237
 238
 239
 240
 241
 242
 243
 244
 245
 246
 247
 248
 249
 
 | # This tries to find wordcode files and automatically re-compile them if
# at least one of the original files is newer than the wordcode file.
# This will only work if the original files were added with their full
# paths or if the names stored in the wordcode files are relative to the
# directory where the wordcode file is.
#
# Arguments are the names of wordcode files and directories containing
# wordcode files that should be checked. If no arguments are given, the
# directories and wordcode files in $fpath are used.
#
# And then there are two options:
#   -t: Only check if there are wordcode files that have to be 
#       re-compiled. The return status is zero if there are files
#       that need to be re-compiled and non-zero otherwise.
#   -q: Be quiet, i.e.: only set the return status.
#   -p: If this is given, the arguments are interpreted differently:
#       they should form one or more sets of arguments for zcompile,
#       separated by `--'. For example:
#
#         zrecompile -p \
#                    -R ~/.zshrc -- \
#                    -M ~/.zcompdump -- \
#                    ~/zsh/comp.zwc ~/zsh/Completion/*/_* \
#
#       This makes ~/.zshrc be compiled into ~/.zshrc.zwc if that doesn't
#       exist or if it is older than ~/.zshrc. The wordcode file will be
#       marked for reading instead of mapping. The same is done for
#       ~/.zcompdump and ~/.zcompdump.zwc, but the wordcode file is marked
#       for mapping. The last line re-creates the file ~/zsh/comp.zwc if
#       any of the files matching the given pattern is newer than it.
#
# Without the -t option, the return status is zero if all wordcode files
# that needed re-compilation could be compiled and non-zero if compilation
# for at least one of the files failed.
setopt localoptions extendedglob noshwordsplit noksharrays
local opt check quiet zwc files re file pre ret map tmp mesg pats
tmp=()
while getopts ":tqp" opt; do
  case $opt in
  t) check=yes ;;
  q) quiet=yes ;;
  p) pats=yes  ;;
  *)
    if [[ -n $pats ]]; then
      tmp=( $tmp $OPTARG )
    else
      print -u2 zrecompile: bad option: -$OPTARG
      return 1
    fi
  esac
done
shift OPTIND-${#tmp:-1}
if [[ -n $check ]]; then
  ret=1
else
  ret=0
fi
if [[ -n $pats ]]; then
  local end num
  while (( $# )); do
    end=$argv[(i)--]
    if [[ end -le $# ]]; then
      files=( $argv[1,end-1] )
      shift end
    else
      files=( $argv )
      argv=()
    fi
    tmp=()
    map=()
    OPTIND=1
    while getopts :MR opt $files; do
      case $opt in
      [MR]) map=( -$opt ) ;;
      *) tmp=( $tmp $files[OPTIND] );;
      esac
    done
    shift OPTIND-1 files
    (( $#files )) || continue
    files=( $files[1] ${files[2,-1]:#*(.zwc|~)} )
    (( $#files )) || continue
    zwc=${files[1]%.zwc}.zwc
    shift 1 files
    (( $#files )) || files=( ${zwc%.zwc} )
    if [[ -f $zwc ]]; then
      num=$(zcompile -t $zwc | wc -l)
      if [[ num-1 -ne $#files ]]; then
        re=yes
      else
        re=
        for file in $files; do
          if [[ $file -nt $zwc ]]; then
            re=yes
	    break
          fi
        done
      fi
    else
      re=yes
    fi
    if [[ -n $re ]]; then
      if [[ -n $check ]]; then
        # ... say so.
        [[ -z $quiet ]] && print $zwc needs re-compilation
        ret=0
      else
        # ... or do it.
        [[ -z $quiet ]] && print -n "re-compiling ${zwc}: "
        # If the file is mapped, it might be mapped right now, so keep the
	# old file by renaming it.
	if [[ -z "$quiet" ]] &&
           { [[ ! -f $zwc ]] || mv -f $zwc ${zwc}.old } &&
           zcompile $map $tmp $zwc $files; then
          print succeeded
	elif ! { { [[ ! -f $zwc ]] || mv -f $zwc ${zwc}.old } &&
                 zcompile $map $tmp $zwc $files 2> /dev/null } then
          [[ -z $quiet ]] && print "re-compiling ${zwc}: failed"
          ret=1
        fi
      fi
    fi
  done
  return ret
fi
# Get the names of wordcode files.
if (( $# )); then
  argv=( ${^argv}/*.zwc(ND)  ${^argv}.zwc(ND)  ${(M)argv:#*.zwc}  )
else
  argv=( ${^fpath}/*.zwc(ND) ${^fpath}.zwc(ND) ${(M)fpath:#*.zwc} )
fi
# We only handle *.zwc files. zcompile only handles *.zwc files. Everybody
# seems to handle only *.zwc files.
argv=( ${^argv%.zwc}.zwc )
for zwc; do
  # Get the files in the wordcode file.
  files=( ${(f)"$(zcompile -t $zwc)"} )
  # See if the wordcode file will be mapped.
  if [[ $files[1] = *\(mapped\)* ]]; then
    map=-M
    mesg='succeeded (old saved)'
  else
    map=-R
    mesg=succeeded
  fi
  # Get the path prefix of the wordcode file to prepend it to names of
  # original files that are relative pathnames.
  
  if [[ $zwc = */* ]]; then
    pre=${zwc%/*}/
  else
    pre=
  fi
  # Maybe this is even for an older version of the shell?
  if [[ $files[1] != *$ZSH_VERSION ]]; then
    re=yes
  else
    re=
  fi
  files=( ${pre}${^files[2,-1]:#/*} ${(M)files[2,-1]:#/*} )
  # If the version is correct, compare the age of every original file
  # to the age of the wordcode file.
  [[ -z $re ]] &&
    for file in $files; do
      if [[ $file -nt $zwc ]]; then
        re=yes
        break
      fi
    done
  if [[ -n $re ]]; then
    # The wordcode files needs re-compilation...
    if [[ -n $check ]]; then
      # ... say so.
      [[ -z $quiet ]] && print $zwc needs re-compilation
      ret=0
    else
      # ... or do it.
      [[ -z $quiet ]] && print -n "re-compiling ${zwc}: "
      tmp=( ${^files}(N) )
      # Here is the call to zcompile, but if we can't find all the original
      # files, we don't try compilation.
      if [[ $#tmp -ne $#files ]]; then
        [[ -z $quiet ]] && print 'failed (missing files)'
        ret=1
      else
        # If the file is mapped, it might be mapped right now, so keep the
	# old file by renaming it.
	if [[ -z "$quiet" ]] &&
           mv -f $zwc ${zwc}.old &&
           zcompile $map $zwc $files; then
          print $mesg
	elif ! { mv -f $zwc ${zwc}.old &&
                 zcompile $map $zwc $files 2> /dev/null } then
          [[ -z $quiet ]] && print "re-compiling ${zwc}: failed"
          ret=1
        fi
      fi
    fi
  fi
done
return ret
 |