726 lines
24 KiB
Text
726 lines
24 KiB
Text
|
#!/bin/bash
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# maildir-notmuch-sync
|
||
|
#
|
||
|
# a script to sync up maildir folders (and thus gmail labels)
|
||
|
# and notmuch tags
|
||
|
#
|
||
|
# Ethan Schoonover / es@ethanschoonover / @ethanschoonover
|
||
|
# ----------------------------------------------------------------------
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# Usage:
|
||
|
# ----------------------------------------------------------------------
|
||
|
#
|
||
|
# maildir-notmuch-sync [--dry-run] /path/to/maildir/account/root
|
||
|
#
|
||
|
# Designed to be called from offlineimap's post/pre sync hooks in the
|
||
|
# account configuration section of .offlineimaprc
|
||
|
#
|
||
|
# The argument passed to maildir-notmuch-sync should be the same as the
|
||
|
# localfolders value for the local respository setup.
|
||
|
#
|
||
|
# For example, if your local repository localfolders looks like this
|
||
|
# in your .offlineimaprc:
|
||
|
#
|
||
|
# [Repository personal-local]
|
||
|
#
|
||
|
# localfolders = ~/var/mail/accounts/personal
|
||
|
#
|
||
|
#
|
||
|
# then the presynchook and postsynchook would look like this (assuming
|
||
|
# that maildir-notmuch-sync is in your path; if not, use the full path
|
||
|
# to the script):
|
||
|
#
|
||
|
# [Account personal]
|
||
|
#
|
||
|
# localrepository = personal-local
|
||
|
# remoterepository = personal-remote
|
||
|
# presynchook = maildir-notmuch-sync "~/var/mail/accounts/personal"
|
||
|
# postsynchook = maildir-notmuch-sync "~/var/mail/accounts/personal"
|
||
|
#
|
||
|
# This allows the script to use short tags for multiple accounts without
|
||
|
# getting confused about what tag goes where.
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# Dry-run
|
||
|
# ----------------------------------------------------------------------
|
||
|
#
|
||
|
# Call the script directly from the command line with the initial
|
||
|
# argument "--dry-run" to test the result:
|
||
|
#
|
||
|
# maildir-notmuch-sync "--dry-run" "~/var/mail/accounts/personal"
|
||
|
#
|
||
|
# Dry-run mode will echo a summary of changes and any deletions, copies
|
||
|
# but will NOT make any changes
|
||
|
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# Values - CHANGE THESE
|
||
|
# ----------------------------------------------------------------------
|
||
|
|
||
|
# Maildir Information
|
||
|
# ----------------------------------
|
||
|
|
||
|
INBOX="inbox" # (mutt's spool dir) - both the tag and folder
|
||
|
SENT="sent" # (mutt's record dir) - both the tag and folder
|
||
|
TRASH="trash" # trash maildir, gmail requires for real deletion
|
||
|
|
||
|
# Note that the tag/folder values (INBOX/SENT/etc) must match
|
||
|
# your local maildir names, after any nametrans by offlineimap. For
|
||
|
# example, my own "INBOX" is translated by offlineimap to "inbox"
|
||
|
# (lowercase) and thus my INBOX value is "inbox".
|
||
|
#
|
||
|
# Again, please note that these ARE case sensitive values and must
|
||
|
# match your local maildir as offlineimap creates it.
|
||
|
|
||
|
|
||
|
# Notmuch Tag Information
|
||
|
# ----------------------------------
|
||
|
|
||
|
# if multiple new.tags in notmuch, identify transient tag here
|
||
|
# this tag is only used as a temporary tag during script run
|
||
|
NEW_TAG="new"
|
||
|
|
||
|
# the tag notmuch uses to keep track of unread status
|
||
|
# also used by mutt-kz (be careful! assigning this in mutt-kz
|
||
|
# and then removing an unread tag really does mark mail as read,
|
||
|
# possibly in bulk!)
|
||
|
UNREAD_TAG="unread"
|
||
|
|
||
|
# if true, convert "new" tag to "unread" at end of script run
|
||
|
# otherwise new tag is simple removed
|
||
|
MAKE_NEW_UNREAD=false
|
||
|
|
||
|
# convert infix slashes to dashes in tags,
|
||
|
# e.g. "clients/bob" becomes "clients-bob"
|
||
|
SLASHES_TO_DASHES=false
|
||
|
|
||
|
# shorter, trimmed tags; see description below
|
||
|
TRIM_ACCOUNT_PREFIX_IN_TAGS=true
|
||
|
|
||
|
# TRIM_ACCOUNT_PREFIX_IN_TAGS (above)
|
||
|
#
|
||
|
# Example: I have two gmail accounts I'm syncing with offlineimap:
|
||
|
# work & personal. The local account respository paths are:
|
||
|
#
|
||
|
# /home/me/mail/personal
|
||
|
# /home/me/mail/work
|
||
|
#
|
||
|
# I index them both in a single notmuch db that is rooted on the
|
||
|
# following path:
|
||
|
#
|
||
|
# /home/me/mail
|
||
|
#
|
||
|
# This allows notmuch to index all my mail in one pass.
|
||
|
# Notmuch indexes mail so that I can search for folder:/personal/INBOX
|
||
|
# (the search without leading slash is almost identical folder:personal/INBOX)
|
||
|
#
|
||
|
# You can then choose here to tag your mails in one of two formats:
|
||
|
#
|
||
|
# UNTRIMMED tag: personal/INBOX
|
||
|
# TRIMMED tag: INBOX
|
||
|
#
|
||
|
# set TRIM_ACCOUNT_PREFIX_IN_TAG=true for the trimmed style
|
||
|
#
|
||
|
# Note that this will not result in a namespace collision since the
|
||
|
# script is being called from offlineimap and passed the path to the
|
||
|
# account root.
|
||
|
#
|
||
|
# The difference between the account root and the notmuch root (as
|
||
|
# set in your notmuch config) determines the account prefix value.
|
||
|
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# Script settings
|
||
|
# ----------------------------------------------------------------------
|
||
|
|
||
|
set -o errexit
|
||
|
set -o nounset
|
||
|
|
||
|
# FIXME add in real opt handling. See TAG_SCRIPT FIXME and CLEANUP_SCRIPT FIXME
|
||
|
# below for details
|
||
|
|
||
|
# check for special run mode
|
||
|
# --dry-run (-d)
|
||
|
# --help (-h)
|
||
|
|
||
|
case ${1:-} in
|
||
|
*-h*) echo "usage: $0 [--dry-run] subcmd maildir_account_root_path";
|
||
|
echo "Valid sub commands are \"pre\" and \"post\"";
|
||
|
exit;
|
||
|
;;
|
||
|
*-d*) RUNCMD=echo; DRYRUN=true; DRYRUN_MSG="- DRYRUN (no changes will be made)"; shift; ;;
|
||
|
*) RUNCMD=eval; DRYRUN=false ;;
|
||
|
esac
|
||
|
|
||
|
|
||
|
# subcommands "pre" and "post" are listed after the options above and before
|
||
|
# the $MAILDIR_ACCOUNT_ROOT path
|
||
|
|
||
|
case ${1:-} in
|
||
|
pre) SUBCMD="pre"; shift; ;;
|
||
|
post) SUBCMD="post"; shift; ;;
|
||
|
*) echo "usage: $0 [--dry-run] subcmd maildir_account_root_path";
|
||
|
echo "Valid sub commands are \"pre\" and \"post\"";
|
||
|
exit;
|
||
|
;;
|
||
|
esac
|
||
|
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# Custom tagging script
|
||
|
# ----------------------------------------------------------------------
|
||
|
|
||
|
# eval a custom tag script used in the post-hook, before we remove all
|
||
|
# of the new tags
|
||
|
# ----------------------------------------------------------------------
|
||
|
|
||
|
TAG_SCRIPT=$HOME/.local/bin/tag_mailing_lists
|
||
|
CLEANUP_SCRIPT=$HOME/.local/bin/tag_clean_up
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# Notmuch config checks
|
||
|
# ----------------------------------------------------------------------
|
||
|
|
||
|
# check if there is an existing notmuch configuration and assign
|
||
|
# the value of new.tags to a variable
|
||
|
# ----------------------------------
|
||
|
if ! notmuch config list &>/dev/null; then
|
||
|
cat <<EOF
|
||
|
------------------------------------
|
||
|
Missing notmuch config file.
|
||
|
|
||
|
Please configure notmuch and run 'notmuch new' once prior to running
|
||
|
$(basename $0). See notes in this script for recommendations on notmuch
|
||
|
configuration values.
|
||
|
------------------------------------
|
||
|
EOF
|
||
|
exit 1;
|
||
|
else # get the current notmuch new.tags value
|
||
|
NM_NEW_TAGS="$(notmuch config list | grep 'new.tags' | cut -f2 -d'=')"
|
||
|
fi
|
||
|
|
||
|
|
||
|
# confirm that the "new.tags" value in notmuch is not empty
|
||
|
# ----------------------------------
|
||
|
if [[ -z "$NM_NEW_TAGS" ]]; then
|
||
|
cat <<-EOF
|
||
|
------------------------------------
|
||
|
The "new.tags" value in your notmuch config is set to an empty value.
|
||
|
Please add a new.tags value by running:
|
||
|
|
||
|
notmuch config set new.tags new
|
||
|
|
||
|
or edit the [new] section of your ~/.notmuch-config file and add:
|
||
|
|
||
|
tags=new
|
||
|
|
||
|
or your desired set of new tags.
|
||
|
------------------------------------
|
||
|
EOF
|
||
|
exit 1; fi
|
||
|
|
||
|
|
||
|
# confirm that the "new.tags" value in notmuch is singlular, or request
|
||
|
# that the user set a manual value here in the script
|
||
|
# ----------------------------------
|
||
|
if [[ "$NM_NEW_TAGS" != "${NM_NEW_TAGS#*;}" ]] && \
|
||
|
[[ "$NM_NEW_TAGS" == "${NM_NEW_TAGS#*${NEW_TAG:-}*}" ]];
|
||
|
then
|
||
|
cat <<-EOF
|
||
|
------------------------------------
|
||
|
You have multiple "new.tags" values specified in your notmuch config.
|
||
|
This is fine as long as it's intentional. However this script requires
|
||
|
one "transient" new tag (e.g. "new") that will be used during the script
|
||
|
execution, and removed at the end of the script run in order to identify
|
||
|
truly new messages.
|
||
|
|
||
|
Please identify this tag in the GLOBAL VARIABLES section of this script
|
||
|
($0) by adding/uncommenting a value such as:
|
||
|
|
||
|
NEW_TAG=new
|
||
|
|
||
|
If you did not intend to set multiple tags in your notmuch config, please
|
||
|
correct it by running:
|
||
|
|
||
|
notmuch config set new.tags new
|
||
|
|
||
|
or editing the [new] section of your ~/.notmuch-config file and
|
||
|
adding
|
||
|
|
||
|
tags=new
|
||
|
|
||
|
or your desired set of new tags.
|
||
|
------------------------------------
|
||
|
EOF
|
||
|
exit 1;
|
||
|
elif [[ "$NM_NEW_TAGS" == "${NM_NEW_TAGS#*;}" ]] && \
|
||
|
[[ "$NM_NEW_TAGS" != "${NEW_TAG}" ]]; then
|
||
|
cat <<-EOF
|
||
|
------------------------------------
|
||
|
You have the notmuch config value "new.tags" set to a value that does
|
||
|
not correspond to the value for \$NEW_TAG in this script.
|
||
|
|
||
|
Your notmuch new.tags == $NM_NEW_TAGS
|
||
|
This script's NEW_TAG == $NEW_TAG
|
||
|
|
||
|
This tag will be used as a transient indicator during script processing.
|
||
|
Please set it to a matching value such as "new" for both the notmuch
|
||
|
config and NEW_TAG in this script.
|
||
|
|
||
|
Note that you may use multiple tags in your notmuch new.tags value that
|
||
|
are separated by ; as long as at least one of them matches the value for
|
||
|
NEW_TAG in this script.
|
||
|
------------------------------------
|
||
|
EOF
|
||
|
exit 1; fi
|
||
|
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# Warning:
|
||
|
# ----------------------------------------------------------------------
|
||
|
#
|
||
|
# Requires a specific offlineimap, notmuch, and mutt-kz setup
|
||
|
# anything else will probably not work and will likely mess
|
||
|
# up your email something fierce; you've been warned.
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# notmuch config notes:
|
||
|
# ----------------------------------------------------------------------
|
||
|
#
|
||
|
# Turn off maildir flag syncing for now. the problematic
|
||
|
# issues there are unread tag -> unread status, and
|
||
|
# deletions, which we prefer to happen on a directory-by-
|
||
|
# directory basis for now (e.g. deleting from one maildir
|
||
|
# - the equivalent to removing a gmail label - doesn't
|
||
|
# delete from *all* maildirs).
|
||
|
#
|
||
|
# Also, .notmuch-config's [new] tags settings should be "new"
|
||
|
# (or similar, and make sure it matches $NEW_TAG in this script)
|
||
|
#
|
||
|
# You can do these via the command line:
|
||
|
#
|
||
|
# notmuch config set maildir.synchronize_flags false
|
||
|
# notmuch config set new.tags new
|
||
|
#
|
||
|
# or in ~/.notmuch-config manually
|
||
|
|
||
|
# TODO: fix maildir flag sync so that it works here as well and can
|
||
|
# be used, specifically for unread status, which is useful
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# offlineimap setup:
|
||
|
# ----------------------------------------------------------------------
|
||
|
#
|
||
|
# You'll most likely want to run this via offlineimap's postsynchook.
|
||
|
# Another option would be inotify/systemd monitoring of your maildir
|
||
|
# root directory for changes.
|
||
|
#
|
||
|
# This script is designed to sync pretty much your entire IMAP folder
|
||
|
# structure from gmail, including "All Mail".
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# Use with Mutt-kz
|
||
|
# ----------------------------------------------------------------------
|
||
|
#
|
||
|
# Using mutt-kz virtual folders, you'll want to use a keybinding such
|
||
|
# as this to quickly archive any mail the way Gmail archives (which is
|
||
|
# to say, it removes the Inbox label):
|
||
|
#
|
||
|
# macro e index,pager ""
|
||
|
#
|
||
|
# Normally I work with the virtual folders, almost exclusively, in
|
||
|
# mutt-kz. However, if you wish to also manually delete and move mails
|
||
|
# from the non-virtual (normal mutt) folders, you'll have to create
|
||
|
# macros that add a tag prior to these functions so that this script
|
||
|
# doesn't revert those changes automatically.
|
||
|
#
|
||
|
# macro d index,pager ""
|
||
|
# macro s index,pager ""
|
||
|
|
||
|
# TODO: check to see what mutt menus currently have delete/save support
|
||
|
# and thus existing key bindings
|
||
|
|
||
|
# TODO: mutt-kz trash macro that adds trash (probably removes INBOX) and quasi-deletes
|
||
|
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# Notmuch database path
|
||
|
# ----------------------------------------------------------------------
|
||
|
# the directories contained by this path are scanned for mail, so
|
||
|
# we use this to locate and identify maildir folders
|
||
|
|
||
|
# Source from existing notmuch config
|
||
|
NOTMUCH_ROOT="$(notmuch config list | grep 'database.path' | cut -f2 -d'=')"
|
||
|
|
||
|
# Normalize path by trimming trailing slash, if any
|
||
|
NOTMUCH_ROOT="${NOTMUCH_ROOT%/}"
|
||
|
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# Account maildir root path
|
||
|
# ----------------------------------------------------------------------
|
||
|
# passed as a command line argument
|
||
|
|
||
|
# Normalize path by trimming trailing slash, if any
|
||
|
# (use eval in case user quoted a path with ~ in it, though they shouldn't have)
|
||
|
eval "MAILDIR_ACCOUNT_ROOT=${1%/}"
|
||
|
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# MAILBOXES:
|
||
|
# ----------------------------------------------------------------------
|
||
|
# get list of mailboxes using the mboxes file output by offlineimap
|
||
|
# other strategies to create this list include a directory listing,
|
||
|
# possibly recursive, of the root maildir
|
||
|
|
||
|
#MAILBOXES="$(sed 's/[^"]*"+\([^"]*\)"/\1\n/g' ~/var/mail/mailboxes)"
|
||
|
|
||
|
# example of a recursive find (incomplete):
|
||
|
# TODO: maybe make this the primary method and create pairings?
|
||
|
# find $MAILDIR -name "cur" -type d -exec dirname '{}' \; | sed "s/^$MAILDIR\///" | sort
|
||
|
|
||
|
MAILBOXES_FULL_PATHS="$(echo "$(find $MAILDIR_ACCOUNT_ROOT -name "cur" -type d -exec dirname '{}' \;)" | sort;)"
|
||
|
# | sed "s/^$MAILDIR\///" | sort
|
||
|
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# Helper Functions
|
||
|
# ----------------------------------------------------------------------
|
||
|
|
||
|
Notmuch_Tag_From_Full_Path ()
|
||
|
{
|
||
|
# TODO: update examples here to be full paths
|
||
|
# This take a path such as: /work/INBOX
|
||
|
# and converts it to a tag: work-INBOX or INBOX
|
||
|
#
|
||
|
# A nested maildir such as: /work/clients/bob
|
||
|
# is converted to a tag: work/clients/bob
|
||
|
#
|
||
|
# If the TRIM_ACCOUNT_PREFIX_IN_TAGS variable is set to true, then
|
||
|
# a nested maildir such as: /work/clients/bob
|
||
|
# is converted to a tag: clients/bob
|
||
|
#
|
||
|
# If SLASHES_TO_DASHES is true, infix slashes will be converted to
|
||
|
# dashes, e.g. "client/bob" becomes "client-bob"
|
||
|
|
||
|
case $TRIM_ACCOUNT_PREFIX_IN_TAGS in
|
||
|
true|TRUE|yes|YES|y|Y) local TRIMMER="$MAILDIR_ACCOUNT_ROOT" ;;
|
||
|
*) local TRIMMER="$NOTMUCH_ROOT" ;;
|
||
|
esac
|
||
|
|
||
|
case $SLASHES_TO_DASHES in
|
||
|
true|TRUE|yes|YES|y|Y) echo "${1#$TRIMMER/}" ;;
|
||
|
*) echo "${1#$TRIMMER/}" | sed "s+/+-+g" ;;
|
||
|
esac
|
||
|
|
||
|
}
|
||
|
|
||
|
Notmuch_Folder_From_Full_Path ()
|
||
|
{
|
||
|
# Takes argument:
|
||
|
#
|
||
|
# /home/username/mail/work/INBOX
|
||
|
#
|
||
|
# and uses notmuch root path to trim and return, for example:
|
||
|
#
|
||
|
# /work/INBOX
|
||
|
#
|
||
|
# which is the full form searchable from notmuch using a query such as:
|
||
|
#
|
||
|
# notmuch search folder:/work/INBOX
|
||
|
|
||
|
# XXX MASSIVE PAIN IN THE ASS.
|
||
|
# Took me forever to figure out that I needed to remove the trailing
|
||
|
# slash from the command below. Likely ES is running 0.17 version of
|
||
|
# notmuch, or even older *shudder*
|
||
|
echo "${1#$NOTMUCH_ROOT/}"
|
||
|
|
||
|
}
|
||
|
|
||
|
Maildir_Account_Folder_From_Full_Path ()
|
||
|
{
|
||
|
# Takes argument:
|
||
|
#
|
||
|
# /home/username/mail/work/INBOX
|
||
|
#
|
||
|
# and uses maildir account root path to trim and return, for example:
|
||
|
#
|
||
|
# /INBOX
|
||
|
#
|
||
|
# which is the full form searchable from notmuch using a query such as:
|
||
|
#
|
||
|
# notmuch search folder:/work/INBOX
|
||
|
|
||
|
echo "${1#$MAILDIR_ACCOUNT_ROOT}"
|
||
|
|
||
|
}
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# PRE Notmuch DB Sync Functions
|
||
|
# ----------------------------------------------------------------------
|
||
|
# executed prior to 'notmuch new'
|
||
|
|
||
|
Notmuch_State_To_Maildir__Move_To_Maildir ()
|
||
|
{
|
||
|
# Scenario:
|
||
|
#
|
||
|
# NOTMUCH STATE (per message):
|
||
|
# Number of Notmuch Tags > Number of Notmuch Folders
|
||
|
#
|
||
|
# MAILDIR STATE:
|
||
|
# No change from previous state.
|
||
|
#
|
||
|
# Tags have been added to a message in a virtual folder (in the notmuch db).
|
||
|
# The number of folders associated with a message has not been changed in
|
||
|
# the notmuch db. This indicates that we need to copy the message to a
|
||
|
# new maildir. After the next 'notmuch new' db update, the tags/folders
|
||
|
# should thus be at parity again.
|
||
|
|
||
|
local THIS_MAILDIR_FULL_PATH="$1"
|
||
|
local THIS_NOTMUCH_FOLDER="$(Notmuch_Folder_From_Full_Path $THIS_MAILDIR_FULL_PATH)"
|
||
|
local THIS_NOTMUCH_TAG="$(Notmuch_Tag_From_Full_Path $THIS_MAILDIR_FULL_PATH)"
|
||
|
local THESE_MESSAGE_IDS_TO_COPY="$(\
|
||
|
notmuch search --output=messages\
|
||
|
tag:"$THIS_NOTMUCH_TAG" \
|
||
|
NOT folder:"$THIS_NOTMUCH_FOLDER" \
|
||
|
NOT folder:"$TRASH" \
|
||
|
NOT tag:"$NEW_TAG")"
|
||
|
|
||
|
# We are running this function prior to the remove function below
|
||
|
# but there is still the edge case wherein the user has manually
|
||
|
# deleted a mail message in mutt (better to do all this with tags
|
||
|
# and virtual folders, but let's accommodate).
|
||
|
|
||
|
for THIS_MESSAGE_ID in $THESE_MESSAGE_IDS_TO_COPY; do
|
||
|
|
||
|
local THIS_MESSAGE_ALL_SOURCE_PATHS="$(notmuch search --output=files "$THIS_MESSAGE_ID")"
|
||
|
|
||
|
local FOUND=false
|
||
|
|
||
|
while read line; do
|
||
|
|
||
|
local THIS_MESSAGE_SOURCE_PATH="$line"
|
||
|
|
||
|
if [[ -e "$THIS_MESSAGE_SOURCE_PATH" ]]; then
|
||
|
FOUND=true
|
||
|
break
|
||
|
fi
|
||
|
|
||
|
done <<< "$THIS_MESSAGE_ALL_SOURCE_PATHS"
|
||
|
|
||
|
if $FOUND; then
|
||
|
|
||
|
if $RUNCMD "cp \"$THIS_MESSAGE_SOURCE_PATH\" \"$THIS_MAILDIR_FULL_PATH/cur\""; then
|
||
|
echo -n "Copied message with new tag to"
|
||
|
echo " $(Maildir_Account_Folder_From_Full_Path "$THIS_MAILDIR_FULL_PATH")"
|
||
|
else
|
||
|
echo -e "\nWARNING: Failed to copy mail file (unknown error):"
|
||
|
echo -e "SOURCE: \"$THIS_MESSAGE_SOURCE_PATH\"\nDESTINATION\"$THIS_MAILDIR_FULL_PATH/cur\"\n"
|
||
|
fi
|
||
|
|
||
|
else
|
||
|
|
||
|
echo -e "\nWARNING: Failed to copy mail file (no valid source paths!):"
|
||
|
echo "ID: $THIS_MESSAGE_ID"
|
||
|
echo "NOTMUCH FOLDER: $THIS_NOTMUCH_FOLDER"
|
||
|
echo -e "DESTINATION MAILDIR: $THIS_MAILDIR_FULL_PATH\n"
|
||
|
fi
|
||
|
|
||
|
done
|
||
|
|
||
|
}
|
||
|
|
||
|
Notmuch_State_To_Maildir__Remove_From_Maildir ()
|
||
|
{
|
||
|
# Scenario:
|
||
|
#
|
||
|
# NOTMUCH STATE (per message):
|
||
|
# Number of Notmuch Tags < Number of Notmuch Folders
|
||
|
#
|
||
|
# MAILDIR STATE:
|
||
|
# No change from previous state.
|
||
|
#
|
||
|
# Tags have been removed from a message in a virtual folder (and thus
|
||
|
# in the notmuch db). The number of folders associated with a message
|
||
|
# has of course not yet changed. We need to remove the messages from
|
||
|
# maildir folders from which it has been untagged.
|
||
|
|
||
|
local THIS_MAILDIR_FULL_PATH="$1"
|
||
|
local THIS_NOTMUCH_FOLDER="$(Notmuch_Folder_From_Full_Path $THIS_MAILDIR_FULL_PATH)"
|
||
|
local THIS_NOTMUCH_TAG="$(Notmuch_Tag_From_Full_Path $THIS_MAILDIR_FULL_PATH)"
|
||
|
local THESE_MESSAGE_IDS_TO_REMOVE="$(\
|
||
|
notmuch search --output=messages\
|
||
|
folder:"$THIS_NOTMUCH_FOLDER" \
|
||
|
NOT tag:"$THIS_NOTMUCH_TAG" \
|
||
|
NOT tag:"$NEW_TAG")"
|
||
|
|
||
|
for THIS_MESSAGE_ID in $THESE_MESSAGE_IDS_TO_REMOVE; do
|
||
|
|
||
|
local THIS_MESSAGE_PATH="$(notmuch search --output=files "$THIS_MESSAGE_ID" | \
|
||
|
grep -e "^$THIS_MAILDIR_FULL_PATH")"
|
||
|
|
||
|
if [[ -e "$THIS_MESSAGE_PATH" ]]; then
|
||
|
|
||
|
if $RUNCMD "rm \"$THIS_MESSAGE_PATH\""; then
|
||
|
echo -n "Removed untagged message from"
|
||
|
echo " $(Maildir_Account_Folder_From_Full_Path "$THIS_MAILDIR_FULL_PATH")"
|
||
|
else
|
||
|
echo -e "\nWARNING: Failed to remove mail file (unknown error):"
|
||
|
echo "ID:$THIS_MESSAGE_ID"
|
||
|
echo "FOLDER:$THIS_NOTMUCH_FOLDER"
|
||
|
echo -e "MESSAGE PATH:$THIS_MESSAGE_PATH\n"
|
||
|
fi
|
||
|
|
||
|
else
|
||
|
|
||
|
echo -e "\nWARNING: Unable to remove missing mail file:"
|
||
|
echo "ID:$THIS_MESSAGE_ID"
|
||
|
echo "FOLDER:$THIS_NOTMUCH_FOLDER"
|
||
|
echo -e "MESSAGE PATH:$THIS_MESSAGE_PATH\n"
|
||
|
fi
|
||
|
|
||
|
done
|
||
|
|
||
|
}
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# SYNC Notmuch DB Sync Functions
|
||
|
# ----------------------------------------------------------------------
|
||
|
|
||
|
Notmuch_Update ()
|
||
|
{
|
||
|
$RUNCMD "notmuch new";
|
||
|
# FIXME pass TAG_SCRIPT as an argument
|
||
|
if [[ -e "$TAG_SCRIPT" ]]; then
|
||
|
$RUNCMD $TAG_SCRIPT
|
||
|
fi
|
||
|
}
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# POST Notmuch DB Sync Functions
|
||
|
# ----------------------------------------------------------------------
|
||
|
# executed after 'notmuch new' (otherwise the notmuch state looks the
|
||
|
# same as the states above)
|
||
|
|
||
|
Maildir_State_To_Notmuch__Add_Tags_To_Notmuch ()
|
||
|
{
|
||
|
# Scenario:
|
||
|
#
|
||
|
# NOTMUCH STATE (per message):
|
||
|
# Number of Notmuch Tags < Number of Notmuch Folders
|
||
|
#
|
||
|
# MAILDIR STATE:
|
||
|
# Message in a new folder (either via CLI/mutt copy, move or incoming sync)
|
||
|
#
|
||
|
# A message is in a "physical" maildir directory but does not have a
|
||
|
# corresponding notmuch tag. For example:
|
||
|
#
|
||
|
# ~/mail/INBOX/message123 should have a tag "INBOX"
|
||
|
#
|
||
|
# We process all mails in each maildir directory (mailbox) and add tags
|
||
|
# as required.
|
||
|
|
||
|
local THIS_MAILDIR_FULL_PATH="$1"
|
||
|
local THIS_NOTMUCH_FOLDER="$(Notmuch_Folder_From_Full_Path $THIS_MAILDIR_FULL_PATH)"
|
||
|
local THIS_NOTMUCH_TAG="$(Notmuch_Tag_From_Full_Path $THIS_MAILDIR_FULL_PATH)"
|
||
|
local THIS_NOTMUCH_QUERY="folder:\"$THIS_NOTMUCH_FOLDER\" NOT tag:\"$THIS_NOTMUCH_TAG\""
|
||
|
local THIS_COUNT="$(notmuch count $THIS_NOTMUCH_QUERY)"
|
||
|
|
||
|
$DRYRUN || notmuch tag +"$THIS_NOTMUCH_TAG" -- $THIS_NOTMUCH_QUERY
|
||
|
[[ $THIS_COUNT > 0 ]] && echo "Tagged $THIS_COUNT messages with \"$THIS_NOTMUCH_TAG\"" || true
|
||
|
|
||
|
}
|
||
|
|
||
|
Maildir_State_To_Notmuch__Remove_Tags_From_Notmuch ()
|
||
|
{
|
||
|
# Scenario:
|
||
|
#
|
||
|
# NOTMUCH STATE (per message):
|
||
|
# Number of Notmuch Tags > Number of Notmuch Folders
|
||
|
#
|
||
|
# MAILDIR STATE:
|
||
|
# Message removed from folder, either via rm, mutt delete, or offlineimap sync
|
||
|
#
|
||
|
# A message has been removed from a maildir directory. Notmuch is aware of
|
||
|
# this (this should only be checked/run after a 'notmuch new' update).
|
||
|
# However, we still have the "old" tag on the message.
|
||
|
#
|
||
|
# We skip the trash since we might want to restore those in future?
|
||
|
|
||
|
local THIS_MAILDIR_FULL_PATH="$1"
|
||
|
local THIS_NOTMUCH_FOLDER="$(Notmuch_Folder_From_Full_Path $THIS_MAILDIR_FULL_PATH)"
|
||
|
local THIS_NOTMUCH_TAG="$(Notmuch_Tag_From_Full_Path $THIS_MAILDIR_FULL_PATH)"
|
||
|
local THIS_NOTMUCH_QUERY="tag:\"$THIS_NOTMUCH_TAG\" \
|
||
|
NOT folder:\"$THIS_NOTMUCH_FOLDER\" \
|
||
|
NOT folder:\"$TRASH\""
|
||
|
local THIS_COUNT="$(notmuch count $THIS_NOTMUCH_QUERY)"
|
||
|
|
||
|
$DRYRUN || notmuch tag -"$THIS_NOTMUCH_TAG" -- $THIS_NOTMUCH_QUERY
|
||
|
[[ $THIS_COUNT > 0 ]] && echo "Untagged $THIS_COUNT messages, removed \"$THIS_NOTMUCH_TAG\"" || true
|
||
|
|
||
|
}
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# CLEANUP Functions
|
||
|
# ----------------------------------------------------------------------
|
||
|
|
||
|
Notmuch_Cleanup ()
|
||
|
{
|
||
|
|
||
|
# anything in sent mail should have the unread flag removed
|
||
|
$RUNCMD "notmuch tag -\"$UNREAD_TAG\" -- folder:\"$SENT\""
|
||
|
|
||
|
# FIXME pass CLEANUP_SCRIPT as an argument
|
||
|
if [[ -e "$CLEANUP_SCRIPT" ]]; then
|
||
|
$RUNCMD $CLEANUP_SCRIPT
|
||
|
fi
|
||
|
|
||
|
# remove "$NEW_TAG" tags, optionally converting to "$UNREAD_TAG"
|
||
|
case $MAKE_NEW_UNREAD in
|
||
|
true|TRUE|yes|YES|y|Y)
|
||
|
$RUNCMD "notmuch tag -\"$NEW_TAG\" +\"$UNREAD_TAG\" -- tag:\"$NEW_TAG\"" ;;
|
||
|
*) $RUNCMD "notmuch tag -\"$NEW_TAG\" -- tag:\"$NEW_TAG\"" ;;
|
||
|
esac
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
# ----------------------------------------------------------------------
|
||
|
# ----------------------------------------------------------------------
|
||
|
# MAIN
|
||
|
# ----------------------------------------------------------------------
|
||
|
# ----------------------------------------------------------------------
|
||
|
|
||
|
echo -e "\n----------------------------------------------------------------------"
|
||
|
echo "$(basename $0) ${SUBCMD}-sync hook ${DRYRUN_MSG:-}"
|
||
|
echo "----------------------------------------------------------------------"
|
||
|
echo "NOTMUCH ROOT: $NOTMUCH_ROOT"
|
||
|
echo "ACCOUNT ROOT: $MAILDIR_ACCOUNT_ROOT"
|
||
|
|
||
|
# Review the notmuch database state and sync up any changes first
|
||
|
# (e.g. any retagged messages that need refiling)
|
||
|
if [ "$SUBCMD" == "pre" ]; then
|
||
|
for MAILBOX_FULL_PATH in $MAILBOXES_FULL_PATHS; do
|
||
|
Notmuch_State_To_Maildir__Move_To_Maildir $MAILBOX_FULL_PATH
|
||
|
Notmuch_State_To_Maildir__Remove_From_Maildir $MAILBOX_FULL_PATH
|
||
|
done
|
||
|
fi
|
||
|
|
||
|
# Update the notmuch database to reflect the changes we just made,
|
||
|
# if any (so it can find the new messages)
|
||
|
if [ "$SUBCMD" == "post" ]; then
|
||
|
Notmuch_Update
|
||
|
|
||
|
for MAILBOX_FULL_PATH in $MAILBOXES_FULL_PATHS; do
|
||
|
Maildir_State_To_Notmuch__Add_Tags_To_Notmuch $MAILBOX_FULL_PATH
|
||
|
Maildir_State_To_Notmuch__Remove_Tags_From_Notmuch $MAILBOX_FULL_PATH
|
||
|
done
|
||
|
|
||
|
Notmuch_Cleanup
|
||
|
fi
|
||
|
|
||
|
echo -e "maildir-notmuch-sync complete ----------------------------------------\n"
|