Não consegui encontrar uma solução simples. A maneira mais fácil que encontrei para resolvê-lo foi usar uma lista "keep" em vez de uma lista "remove". Então eu poderia adicionar os últimos N backups à lista, e o mais novo de cada semana, por M semanas. (No meu próprio código, escrito há algum tempo, estendi-me a meses, trimestres e anos.) Obviamente, haverá alguma sobreposição, mas porque é uma lista de "manter" que não importa. Em seguida, você exclui tudo o que não está na lista. Primeiro, verifique se a lista não está vazia.
#!/bin/bash
#
dirs=(
backup-2018-01-12
backup-2018-01-13
backup-2018-01-14
backup-2018-01-15
backup-2018-01-16
backup-2018-01-17
backup-2018-01-18
backup-2018-01-19
backup-2018-01-20
backup-2018-01-21
backup-2018-01-22
backup-2018-01-23
backup-2018-01-24
backup-2018-01-25
backup-2018-01-26
backup-2018-01-27
backup-2018-01-28
backup-2018-01-29
backup-2018-01-30
backup-2018-01-31
backup-2018-02-01
backup-2018-02-02
backup-2018-02-03
backup-2018-02-04
backup-2018-02-05
backup-2018-02-06
backup-2018-02-07
backup-2018-02-08
backup-2018-02-09
backup-2018-02-10
backup-2018-02-11
backup-2018-02-12
backup-2018-02-13
backup-2018-02-14
backup-2018-02-15
)
declare -A list
declare -A keep
####################################################################################
# Go
#
prefix="backup-"
today=$(date +%Y-%m-%d)
# Populate list from set of directories
#
for item in "${dirs[@]}"
do
list[$item]="$item"
done
# Build keep list from business criteria
#
for dayback in {0..6}
do
dateback=$(date --date "$today - $dayback day" +%Y-%m-%d)
keep[$prefix$dateback]=1
done
for weekback in {0..3}
do
dateback1=$(date --date "$today - $weekback week" +%Y-%m-%d)
dateback2=$(date --date "$today - $((weekback + 1)) week" +%Y-%m-%d)
# Look for newest in range
#
newest=
dateback="$dateback1"
while [[ $dateback != $dateback2 ]]
do
[[ -n ${list[$prefix$dateback]} ]] && newest=$dateback && break
dateback=$(date --date "$dateback - 1 day" +%Y-%m-%d)
done
[[ -n $newest ]] && keep[$prefix$newest]=1
done
# We now have a keep list, so delete anything not in that list
#
for item in "${list[@]}"
do
[[ -n ${keep[$item]} ]] && echo "KEEP $item" || echo "DELETE $item"
done
# All done
#
exit 0
Quando estiver satisfeito, altere ou amplie as instruções de eco no final para realizar a exclusão real.