#!/bin/zsh
-# Copyright 2014 - 2016 Paul Hänsch
+# Copyright 2014 - 2017 Paul Hänsch
#
# This file is part of Confetti.
#
# You should have received a copy of the GNU Affero General Public License
# along with Confetti. If not, see <http://www.gnu.org/licenses/>.
-[ -z "${_GET[order]}" ] && _GET[order]="firstname"
+BR='
+'
+
+force_items(){
+ for each in "$@"; do
+ [ -z "${values[$each]+x}" ] && values[${each}]=''
+ done
+}
+
+case $PROFILE in
+medical)
+ SUP_FIELDS=(N NICKNAME GENDER BDAY ADR TEL EMAIL X-HEALTH-INSURANCE X-HEALTH-INSURANCE-NOCONTRIB IMPP URL NOTE X-CLIENT-REFERRAL)
+ FORCE_ITEMS=(ADR TEL EMAIL NOTE X-CLIENT-REFERRAL)
+ _GET[order]="${_GET[order]:-lastname}"
+ _GET[filtertype]="${_GET[filtertype]:-name}"
+ profile_medical=x
+;;
+circus)
+ SUP_FIELDS=(N NICKNAME GENDER BDAY X-ZACK-JOINDATE X-ZACK-LEAVEDATE EMAIL TEL IMPP ADR URL NOTE)
+ FORCE_ITEMS=(BDAY X-ZACK-JOINDATE TEL EMAIL ADR NOTE)
+ _GET[order]="${_GET[order]:-firstname}"
+ _GET[filtertype]="${_GET[filtertype]:-any}"
+ profile_circus=x
+;;
+esac
+
+edit="${_GET[edit]}"
+[ \! -f "vcard/$edit" -a \! -f "temp/$edit" ] && edit=''
listcourses() {
ls -1 ${_DATA}/ical/*ics |while read file; do
done |sort |sed -r 's:^.*\t(.*/)([^/]+)$:\2:'
}
+list_hi_companies(){
+ sed -rn 's;^X-HEALTH-INSURANCE:([^\;]+)\;.*$;\1;p' ${_DATA}/vcard/*vcf \
+ | sort -u
+}
+
+list_categories() {
+ catfile="${_DATA}/mappings/categories"
+ sort -u "$catfile" \
+ | sed -r '/^[\t ]*$/d'
+}
+
listcards() {
- case "${_GET[filtertype]}" in
- any)
- grep -il "${_GET[filter]}" ${_DATA}/vcard/*vcf
- ;;
- name)
- egrep -xil "(FN|NICKNAME|N)(;.+)*:.*${_GET[filter]}.*" ${_DATA}/vcard/*vcf
- ;;
- street)
- egrep -xil "(ADR)(;.+)*:([^;]*;){2}${_GET[filter]}.*" ${_DATA}/vcard/*vcf
- ;;
- zip)
- egrep -xil "(ADR)(;.+)*:([^;]*;){5}${_GET[filter]}.*" ${_DATA}/vcard/*vcf
- ;;
- telephone)
- egrep -xil "(TEL)(;.+)*:.*${_GET[filter]}.*" ${_DATA}/vcard/*vcf
- ;;
- birth)
- egrep -xil "(BDAY)(;.+)*:${_GET[filter]}.*" ${_DATA}/vcard/*vcf
- ;;
- course)
- ;;
- *) ls -1 ${_DATA}/vcard/*vcf 2>/dev/null
- ;;
- esac |case "${_GET[order]}" in
- firstname)
- while read file; do
- fn=$(sed -rn 's:^N(;.+)*\:([^;]*;){1} *([^;]*).*$:\3:p' "$file")
- echo "$fn\t$file"
- done
- ;;
- lastname)
- while read file; do
- ln=$(sed -rn 's:^N(;.+)*\:([^;]*;){0} *([^;]*).*$:\3:p' "$file")
- echo "$ln\t$file"
- done
- ;;
- bdate)
- while read file; do
- bd=$(sed -rn 's:^BDAY(;.+)*\:(.*)$:\2:p' "$file")
- echo "$bd\t$file"
- done
- ;;
- *)
- sed -r 's:^.*$:x\t&:'
- ;;
- esac |sort |sed -r 's:^.*\t(.*/)([^/]+)$:\2:'
+ filterex='s;^([^\n]+)\n.*$;\1;p'
+ printf '%s\n' "${_GET[filter]}" |tr '^' '\n' \
+ | sed -r 's;[]\/\(\)\\\^\$\?\.\+\*\;\[\{\}];\\\\&;g' \
+ | while read each; do
+ case $each in
+ name:*) expr='(FN|NICKNAME|N)(\;[^\n]+)*:[^\n]*'"(${each#*:})";;
+ street:*) expr='ADR(\;[^\n]+)*:([^\;]*;){2}[^\;\n]*'"(${each#*:})";;
+ zip:*) expr='ADR(\;[^\n]+)*:([^\;]*;){5}[^\;\n]*'"(${each#*:})";;
+ any:*|:*) expr="[^\n]*"'(\;[^\n]+)*:[^\n]*'"(${each#*:})";;
+ *:*) expr="${each%%:*}"'(\;[^\n]+)*:[^\n]*'"(${each#*:})";;
+ *) expr="(${each})";;
+ esac
+ filterex='/(^|\n)'"${expr}"'/I{'"${filterex}"'}'
+ done
+
+ for file in "${_DATA}/vcard/"*.vcf; do
+ case "${_GET[order]}" in
+ firstname)
+ printf '%s\t%s\n' "$(sed -rn 's:^N(;.+)*\:([^;]*;){1} *([^;]*).*$:\3:p' "$file")" "$file"
+ ;;
+ lastname)
+ printf '%s\t%s\n' "$(sed -rn 's:^N(;.+)*\:([^;]*;){0} *([^;]*).*$:\3:p' "$file")" "$file"
+ ;;
+ bdate)
+ printf '%s\t%s\n' "$(sed -rn 's:^BDAY(;.+)*\:(.*)$:\2:p' "$file")" "$file"
+ ;;
+ *) printf 'x\t%s\n' "$file"
+ ;;
+ esac
+ done \
+ | sort -u |sed -r 's;^.*\t;;' \
+ | while read n; do
+ { printf '%s\n' "$n"; cat "$n"; } \
+ | sed -rn ':X;N;$!bX; {'"$filterex"'}'
+ done \
+ | sed -r 's;^(.*/)*;;;'
}
vcf_parse() {
- tr -d '\n' <"$1" |sed -r 's:\r ::g;s:\r:\n:g' \
+ unset key
+ sed -r ':X;N;$!bX; s;\r\n[ \t];;g; s;\r\n;\n;g;' "$1" \
| sed -rn '
- s:^PHOTO:001 PHOTO:p;
- s:^X-MS-CARDPICTURE:001 PHOTO:p;
- s:^LOGO:002 LOGO:p;
- s:^FN:003 FN:p;
- s:^N[\:;]:004 &:p
- s:^NICKNAME:005 NICKNAME:p;
- s:^SOUND:006 SOUND:p;
- s:^GENDER:007 GENDER:p;
- s:^X-GENDER:007 GENDER:p;
- s:^KIND:008 KIND:p;
- s:^TITLE:009 TITLE:p;
- s:^ROLE:010 ROLE:p;
- s:^ORG:011 ORG:p;
- s:^MEMBER:012 MEMBER:p;
- s:^CATEGORIES:013 CATEGORIES:p;
- s:^ANNIVERSARY:014 ANNIVERSARY:p;
- s:^X-ANNIVERSARY:014 ANNIVERSARY:p;
- s:^X-EVOLUTION-ANNIVERSARY:014 ANNIVERSARY:p;
- s:^X-KADDRESSBOOK-X-Anniversary:014 ANNIVERSARY:p;
- s:^BDAY:015 BDAY:p;
- s:^EMAIL:016 EMAIL:p;
- s:^TEL:017 TEL:p;
- s:^IMPP:018 IMPP:p;
- s:^X-AIM(;[^"\:]+|;"[^"]+")*\:(.*)$:018 IMPP\1\:aim\:\2:p;
- s:^X-ICQ(;[^"\:]+|;"[^"]+")*\:(.*)$:018 IMPP\1\:aim\:\2:p;
- s:^X-GOOGLE-TALK(;[^"\:]+|;"[^"]+")*\:(.*)$:018 IMPP\1\:xmpp\:\2:p;
- s:^X-JABBER(;[^"\:]+|;"[^"]+")*\:(.*)$:018 IMPP\1\:xmpp\:\2:p;
- s:^X-MSN(;[^"\:]+|;"[^"]+")*\:(.*)$:018 IMPP\1\:msn\:\2:p;
- s:^X-YAHOO(;[^"\:]+|;"[^"]+")*\:(.*)$:018 IMPP\1\:ymsgr\:\2:p;
- s:^X-SIP(;[^"\:]+|;"[^"]+")*\:(sip\:)?(.*)$:018 IMPP\1\:sip\:\3:p;
- s:^ADR:019 ADR:p;
- s:^LABEL(;[^"\:]+|;"[^"]+")*\:(.*)$:019 ADR;LABEL="\2"\1\::p;
- s:^URL:021 URL:p;
- s:^X-EVOLUTION-BLOG-URL:021 URL:p;
- s:^LANG:022 LANG:p;
- s:^NOTE:023 NOTE:p;
- s:^UID:026 UID:p;
+ # === turn property names to upper case, strip group names ===
+ h; s;^([^;:]+);;;
+ x; s;^([^;:\.]+\.)?([^;:]+).*$;\2;;
+ y;abcdefghijklmnopqrstuvwxyz;ABCDEFGHIJKLMNOPQRSTUVWXYZ;
+ G; s;\n;;;
+
+ # === strip trailing CR (but keep CRs in property value) ===
+ s;\r$;;;
+
+ # === Normalise various known vendor properties ===
+ s;^X-MS-CARDPICTURE(\;|:);PHOTO\1;;
+ s;^X-GENDER(\;|:);GENDER\1;;
+ s;^X-ANNIVERSARY(\;|:);ANNIVERSARY\1;;
+ s;^X-EVOLUTION-ANNIVERSARY(\;|:);ANNIVERSARY\1;;
+ s;^X-KADDRESSBOOK-X-ANNIVERSARY(\;|:);ANNIVERSARY\1;;
+ s;^X-EVOLUTION-BLOG-URL(\;|:);URL\1;;
+ s;^AGENT(\;|:);RELATED\;VALUE=text\;TYPE=agent\1;;
+ s;^X-ASSISTANT(\;|:);RELATED\;VALUE=text\;TYPE=assistant\1;;
+ s;^X-EVOLUTION-ASSISTANT(\;|:);RELATED\;VALUE=text\;TYPE=assistant\1;;
+ s;^X-KADDRESSBOOK-X-ASSISTANTSNAME(\;|:);RELATED\;VALUE=text\;TYPE=assistant\1;;
+ s;^X-MANAGER(\;|:);RELATED\;VALUE=text\;TYPE=manager\1;;
+ s;^X-EVOLUTION-MANAGER(\;|:);RELATED\;VALUE=text\;TYPE=manager\1;;
+ s;^X-KADDRESSBOOK-X-MANAGERSNAME(\;|:);RELATED\;VALUE=text\;TYPE=manager\1;;
+ s;^X-SPOUSE(\;|:);RELATED\;VALUE=text\;TYPE=spouse\1;;
+ s;^X-EVOLUTION-SPOUSE(\;|:);RELATED\;VALUE=text\;TYPE=spouse\1;;
+ s;^X-KADDRESSBOOK-X-SPOUSENAME(\;|:);RELATED\;VALUE=text\;TYPE=spouse\1;;
+
+ # === Normalise obsolete vendor IM properties ===
+ s;^X-AIM((\;[A-z0-9-]+|\;[A-z0-9-]+=([^;,:"]+|"[^"]+")(,[^;,:"]+|,"[^"]+")*)*):;IMPP\1:aim:;;
+ s;^X-ICQ((\;[A-z0-9-]+|\;[A-z0-9-]+=([^;,:"]+|"[^"]+")(,[^;,:"]+|,"[^"]+")*)*):;IMPP\1:aim:;;
+ s;^X-GOOGLE-TALK((\;[A-z0-9-]+|\;[A-z0-9-]+=([^;,:"]+|"[^"]+")(,[^;,:"]+|,"[^"]+")*)*):;IMPP\1:xmpp:;;
+ s;^X-JABBER((\;[A-z0-9-]+|\;[A-z0-9-]+=([^;,:"]+|"[^"]+")(,[^;,:"]+|,"[^"]+")*)*):;IMPP\1:xmpp:;;
+ s;^X-MSN((\;[A-z0-9-]+|\;[A-z0-9-]+=([^;,:"]+|"[^"]+")(,[^;,:"]+|,"[^"]+")*)*):;IMPP\1:msn:;;
+ s;^X-YAHOO((\;[A-z0-9-]+|\;[A-z0-9-]+=([^;,:"]+|"[^"]+")(,[^;,:"]+|,"[^"]+")*)*):;IMPP\1:ymsgr:;;
+ s;^X-SIP((\;[A-z0-9-]+|\;[A-z0-9-]+=([^;,:"]+|"[^"]+")(,[^;,:"]+|,"[^"]+")*)*):(sip:)?;IMPP\1:sip:;;
- s:^RELATED:025 RELATED:p;
- s:^AGENT:025 RELATED\;TYPE=agent:p;
- s:^X-ASSISTANT:025 RELATED\;TYPE=assistant;VALUE=text:p;
- s:^X-EVOLUTION-ASSISTANT:025 RELATED\;TYPE=assistant;VALUE=text:p;
- s:^X-KADDRESSBOOK-X-AssistantsName:025 RELATED\;TYPE=assistant;VALUE=text:p;
- s:^X-MANAGER:025 RELATED\;TYPE=manager;VALUE=text:p;
- s:^X-EVOLUTION-MANAGER:025 RELATED\;TYPE=manager;VALUE=text:p;
- s:^X-KADDRESSBOOK-X-ManagersName:025 RELATED\;TYPE=manager;VALUE=text:p;
- s:^X-SPOUSE:025 RELATED\;TYPE=spouse;VALUE=text:p;
- s:^X-EVOLUTION-SPOUSE:025 RELATED\;TYPE=spouse;VALUE=text:p;
- s:^X-KADDRESSBOOK-X-SpouseName:025 RELATED\;TYPE=spouse;VALUE=text:p;
+ # === Update obsolete LABEL property ===
+ s;^LABEL((\;[A-z0-9-]+|\;[A-z0-9-]+=([^;,:"]+|"[^"]+")(,[^;,:"]+|,"[^"]+")*)*):(.*)$;ADR\1\;LABEL="\5":;;
- s:^BEGIN.*$:000 &:p;
- s:^CALADRURI.*$::;
- s:^CALURI.*$::;
- s:^CLASS.*$::;
- s:^CLIENTPIDMAP.*$::;
- s:^END.*$:100 &:p;
- s:^FBURL.*$::;
- s:^GEO.*$::;
- s:^MAILER.*$::;
- s:^NAME.*$::;
- s:^PRODID.*$::;
- s:^PROFILE.*$::;
- s:^REV.*$::;
- s:^SORT-STRING.*$::;
- s:^SOURCE.*$::;
- s:^TZ.*$::;
- s:^VERSION.*$:000 VERSION\:4.0:p;
- s:^XML.*$::;
+ /^([A-Z0-9-]+)((\;[A-z0-9-]+=?|\;[A-z0-9-]+=([^;,:"]+|"[^"]+")(,[^;,:"]+|,"[^"]+")*)*):(.*)$/{
+ h;
+ s;^([A-Z0-9-]+)((\;[A-z0-9-]+=?|\;[A-z0-9-]+=([^;,:"]+|"[^"]+")(,[^;,:"]+|,"[^"]+")*)*):(.*)$;key='\''\1'\'';;
+ H; g;
- s:^([A-Z].*)$:024 \1:p;
+ s;^([A-Z0-9-]+)((\;[A-z0-9-]+=?|\;[A-z0-9-]+=([^;,:"]+|"[^"]+")(,[^;,:"]+|,"[^"]+")*)*):([^\n]*)\n.*$;\2;;
+ s;'\'';'\'\\\\\'\'';g
+ s;\;([A-Z0-9-]+)(=|=(([^\;,:"]+|"[^"]+")(,[^\;,:"]+|,"[^"]+")*))?;\ntag[\1]='\''\3'\'';g
+ s;^\n+;;; s;\n+$;;;
+ /^.+$/H; g;
+
+ s;^([A-Z0-9-]+)((\;[A-z0-9-]+=?|\;[A-z0-9-]+=([^;,:"]+|"[^"]+")(,[^;,:"]+|,"[^"]+")*)*):([^\n]*)\n.*$;\6;;
+ # :X s;((^|[^\\])(\\\\)*)[;,];\1\n;g; s;((^|[^\\])(\\\\)*)\\([;,]);\4;g; tX;
+ s;'\'';'\'\\\\\'\'';g
+ s;^.*$;value='\''&'\'';
+ H; g;
+
+ s;^[^\n]*[\n ]+;;
+ s;\n+$;;;
+
+ p;
+ }
' \
- |sort |while read -r line; do
- case "$line" in
- 00[012]*)
- echo -E "$line"
- ;;
- 003*)
- fn=$(echo -E "$line" |sed -r 's:^[0-9]{3} ([^;\:]+)(;[^"\:]+|;"[^"]+")*\:(.*)$:\3:g' |tr -d '\r')
- ;;
- 004*)
- n=$(echo -E "$line" \
- |sed -rn 's:^([0-9]{3} )([^;\:]+)(;[^"\:]+|;"[^"]+")*\:([^;]*)(\;[^;]*)(\;[^;]*)?(\;[^;]*)?(\;[^;]*)?$:\7 \5 \6 \4 \8:gp' \
- |sed -r 's:,: :;s:\;: :g;s: +: :g' \
- |tr -d '\r'
- )
- echo -E "$line"
- ;;
- 005*)
- nick=$(echo -E "$line" |sed -r 's:^[0-9]{3} ([^;\:]+)(;[^"\:]+|;"[^"]+")*\:(.*)$:\3:g' |tr -d '\r')
- echo -E "$line"
- ;;
- *)
- [ -n "$n" ] && fn="$n"
- #[ -n "$fn" -a -n "$nick" ] && fn="$fn aka. $nick"
- [ -n "$fn" ] && echo -E "003 FN:$fn" \
- || echo -E "003 FN:$nick"
- echo -E "$line"
- cat
- ;;
- esac
- done |tr -d '\r' \
- | sed -r 's:^[0-9]{3} ([^;\:]+)(;[^"\:]+|;"[^"]+")*\:(.*)$:key="\1"\nvalue="\3"\ntag=\2:g' \
| while read -r line; do
+ declare -A tag
case "$line" in
- key=*) echo -E "$line"
- ;;
- value=*) echo -E "$line"
- ;;
- tag=*) ot=''
- echo -E "$line" \
- | sed -r 's:^tag=::;s:\;([A-Z+_-]+="[^"]+"|[A-Z+_-]+=[^\;]+):\n\1:g;' \
- | sed -r 's:([A-Z+_-]+)="?(.*)"?:tag\[\1\]="\2":g' \
- | sed -r '/^ *$/d' \
- | sort |while read -r tag; do
- nt="$(echo -E "$tag" |sed -r 's:^tag\[([A-Z+_-]+)\]="(.*)"$:\1:')"
- nv="$(echo -E "$tag" |sed -r 's:^tag\[([A-Z+_-]+)\]="(.*)"$:\2:')"
- [ "$nt" = "$ot" ] && vl="$nv,$vl" || vl="$nv"
- echo -E "tag[$nt]=\"$vl\""
- ot="$nt"
- done
- ;;
+ value*) eval "$line";;
+ tag*) eval "$line";;
+ key*)
+ if [ -z "$key" ]; then
+ eval "$line"
+ else
+ printf '%s\n' "$value" |sed -rn '
+ :X
+ s;((^|[^\\])(\\\\)*),;\1\n;g;
+ tX;
+ s;\\,;,;g;
+ p
+ ' \
+ | while read -r val; do
+ while [ -n "${values[$key$n]+x}" ]; do n=$((${n=-1} + 1)); done
+ if printf '%s\n' "$val" |grep -qE '((^|[^\\])(\\\\)*)\;'; then
+ m=0
+ values[${key}${n}]="${val}"
+ printf '%s\n' "$val" |sed -rn '
+ :X
+ s;((^|[^\\])(\\\\)*)\;;\1\n;g;
+ tX;
+ s;\\\;;\;;g;
+ p
+ ' \
+ | while read -r v; do
+ values[${key}${n}+${m}]="$(
+ printf %s\\n "${v}" \
+ | sed -rn '
+ :X
+ s;((^|[^\\])(\\\\)*)\\n;\1\n;g;
+ tX;
+ s;\\\\;\\;g;
+ p
+ '
+ )"
+ m=$(($m + 1))
+ done
+ else
+ values[${key}${n}]="$(
+ printf %s\\n "${val}" \
+ | sed -rn '
+ :X
+ s;((^|[^\\])(\\\\)*)\\n;\1\n;g;
+ tX;
+ s;\\\;;\;;g;
+ s;\\\\;\\;g;
+ p
+ '
+ )"
+ fi
+ for t in ${(k)tag}; do
+ values[${key}${n}_${t}]="${tag[$t]}"
+ done
+ done
+
+ eval "$line"
+ unset n
+ while [ -n "${values[$key$n]+x}" ]; do n=$((${n=-1} + 1)); done
+ unset value
+ unset tag
+ fi
+ ;;
esac
- done \
- | sed -r 's:[\\$`]:\\&:g'
+ done
}
view_card() { #Parameter: Cardfile
id="$1"
cardfile="$_DATA/vcard/${id}"
cachefile="$_DATA/cache/${id}.cache"
- unset key
if [ "$cachefile" -nt "$cardfile" ]; then
cat "$cachefile"
else
- declare -A tags
declare -A values
- vcf_parse "$cardfile" |while read -r line; do
- declare -A tag
- case "$line" in
- value*) eval "$line";;
- tag*) eval "$line";;
- key*)
- if [ -z "$key" ]; then
- eval "$line"
- else
- values[$key]="$value"
- for t in ${(k)tag}; do
- tags[${key}_$t]="$tag[$t]"
- done
- eval "$line"
- if [ -n "$values[$key]" ]; then
- n=0
- while [ -n "$values[$key$n]" ]; do n=$(($n + 1)); done
- key=$key$n
- fi
- unset value
- unset tag
- fi
- ;;
- esac
- done
- . $view_card |tee "$cachefile"
+ vcf_parse "$cardfile"
+ . "$_EXEC/templates/view_card.sh" |tee "$cachefile"
fi
}
cardfile="$_DATA/vcard/$id"
tempfile="$_DATA/temp/$id"
[ -f "$tempfile" ] && cardfile="$tempfile"
- unset key
- declare -A tags
declare -A values
- vcf_parse "$cardfile" |while read -r line; do
- declare -A tag
- case "$line" in
- value*) eval "$line";;
- tag*) eval "$line";;
- key*)
- if [ -z "$key" ]; then
- eval "$line"
- else
- [ -n "$value" ] && values[$key]="$value" || values[$key]='\r'
- for t in ${(k)tag}; do
- tags[${key}_$t]="$tag[$t]"
- done
- eval "$line"
- if [ -n "$values[$key]" ]; then
- n=0
- while [ -n "$values[${key}${n}]" ]; do n=$(($n + 1)); done
- key=$key$n
- fi
- unset value
- unset tag
- fi
- ;;
- esac
- done
- . $edit_card
+ vcf_parse "$cardfile"
+ force_items $FORCE_ITEMS
+ . "$_EXEC/templates/edit_card.sh"
}