syncevolution/test/synccompare.pl

1166 lines
44 KiB
Perl
Raw Normal View History

#! /usr/bin/env perl
#
# Copyright (C) 2008 Funambol, Inc.
# Copyright (C) 2008-2009 Patrick Ohly <patrick.ohly@gmx.de>
# Copyright (C) 2009 Intel Corporation
#
# Usage: <file>
# <left file> <right file>
# Either normalizes a file or compares two of them in a side-by-side
# diff.
#
# Checks environment variables:
#
# CLIENT_TEST_SERVER=funambol|scheduleworld|egroupware|synthesis
# Enables code which simplifies the text files just like
# certain well-known servers do. This is useful for testing
# to ignore the data loss introduced by these servers or (for
# users) to simulate the effect of these servers on their data.
#
# CLIENT_TEST_CLIENT=evolution|addressbook (Mac OS X/iPhone)
# Same as for servers this replicates the effect of storing
# data in the clients.
#
# CLIENT_TEST_LEFT_NAME="before sync"
# CLIENT_TEST_RIGHT_NAME="after sync"
# CLIENT_TEST_REMOVED="removed during sync"
# CLIENT_TEST_ADDED="added during sync"
# Setting these variables changes the default legend
# print above the left and right file during a
# comparison.
#
# CLIENT_TEST_COMPARISON_FAILED=1
# Overrides the default error code when changes are found.
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU Lesser General Public
# License as published by the Free Software Foundation; either
# version 2.1 of the License, or (at your option) version 3.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public
# License along with this library; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
# 02110-1301 USA
use strict;
synccompare: avoid segfault in Perl interpreter by limiting UTF-8 support The Perl version in Ubuntu Lucid had even more problems with synccompare than other versions. Compiling the latest stable Perl also didn't help (stack backtrace below). This patch disables UTF-8 support in string operations. The user-visible effect is that line length calculation is wrong when multi-byte characters are involved. This is better than not getting any output when the installed Perl is bad. File input and stdout are still using UTF-8, so UTF-8 content is passed through okay. *** glibc detected *** /usr/local/bin/perl-synccompare: double free or corruption (!prev): 0x08203278 *** ======= Backtrace: ========= /lib/tls/i686/cmov/libc.so.6(+0x6b591)[0xf7616591] /lib/tls/i686/cmov/libc.so.6(+0x6cde8)[0xf7617de8] /lib/tls/i686/cmov/libc.so.6(cfree+0x6d)[0xf761aecd] /usr/local/bin/perl-synccompare(Perl_offer_nice_chunk+0x3d)[0x80d2b0d] /usr/local/bin/perl-synccompare[0x80c426a] /usr/local/bin/perl-synccompare(Perl_hv_common+0xafa)[0x80c628a] /usr/local/bin/perl-synccompare(Perl_pp_helem+0x24a)[0x80d138a] /usr/local/bin/perl-synccompare(Perl_runops_standard+0x13)[0x80c9623] /usr/local/bin/perl-synccompare[0x80f74b0] /usr/local/bin/perl-synccompare(Perl_pp_require+0x11da)[0x80f9c7a] /usr/local/bin/perl-synccompare(Perl_runops_standard+0x13)[0x80c9623] /usr/local/bin/perl-synccompare(Perl_call_sv+0x165)[0x8077705] /usr/local/bin/perl-synccompare(Perl_swash_init+0x1d6)[0x8128686] /usr/local/bin/perl-synccompare(Perl_to_utf8_case+0x213)[0x812a723] /usr/local/bin/perl-synccompare(Perl_to_utf8_lower+0x37)[0x812a7f7] /usr/local/bin/perl-synccompare[0x8121280] /usr/local/bin/perl-synccompare(Perl_regexec_flags+0xb58)[0x8126b48] /usr/local/bin/perl-synccompare(Perl_pp_subst+0x207)[0x80d1917] /usr/local/bin/perl-synccompare(Perl_runops_standard+0x13)[0x80c9623] /usr/local/bin/perl-synccompare(perl_run+0x2bd)[0x807823d] /usr/local/bin/perl-synccompare(main+0x115)[0x8062c35] /lib/tls/i686/cmov/libc.so.6(__libc_start_main+0xe6)[0xf75c1bd6] /usr/local/bin/perl-synccompare[0x8062a81] ======= Memory map: ======== 08048000-0815e000 r-xp 00000000 08:05 2007582 /home/nightly/perl-5.12.3-lucid-i386/bin/perl
2011-05-05 14:08:32 +02:00
# Various crashes have been encountered in the Perl interpreter
# executable when enabling UTF-8. It is only needed for nicer
# side-by-side comparison of changes (correct column width),
# so not much functionality is lost by disabling this.
# use encoding 'utf8';
# Instead enable writing the result as UTF-8. Input
# files are read as UTF-8 via PerlIO parameters in open().
binmode(STDOUT, ":utf8");
use Algorithm::Diff;
use MIME::Base64;
use Digest::MD5 qw(md5 md5_hex md5_base64);
# ignore differences caused by specific servers or local backends?
my $server = $ENV{CLIENT_TEST_SERVER};
my $client = $ENV{CLIENT_TEST_CLIENT} || "evolution";
my $scheduleworld = $server =~ /scheduleworld/;
my $synthesis = $server =~ /synthesis/;
my $zyb = $server =~ /zyb/;
my $mobical = $server =~ /mobical/;
my $memotoo = $server =~ /memotoo/;
my $nokia_7210c = $server =~ /nokia_7210c/;
my $ovi = $server =~ /Ovi/;
my $unique_uid = $ENV{CLIENT_TEST_UNIQUE_UID};
my $full_timezones = $ENV{CLIENT_TEST_FULL_TIMEZONES}; # do not simplify VTIMEZONE definitions
my $no_timezones = $ENV{CLIENT_TEST_NO_TIMEZONES};
# TODO: this hack ensures that any synchronization is limited to
# properties supported by Synthesis. Remove this again.
# $synthesis = 1;
my $exchange = $server =~ /exchange/; # Exchange via ActiveSync
my $egroupware = $server =~ /egroupware/;
my $funambol = $server =~ /funambol/;
my $googlesyncml = $server eq "google";
my $googlecaldav = $server eq "googlecalendar";
my $googlecarddav = $server eq "googlecontacts";
my $googleeas = $server eq "googleeas";
my $google_valarm = $ENV{CLIENT_TEST_GOOGLE_VALARM};
my $yahoo = $server =~ /yahoo/;
my $davical = $server =~ /davical/;
my $apple = $server =~ /apple/;
my $oracle = $server =~ /oracle/;
my $radicale = $server =~ /radicale/;
my $zimbra = $server =~ /zimbra/;
my $evolution = $client =~ /evolution/;
my $addressbook = $client =~ /addressbook/;
my $akonadi = $server =~ /kde/;
sub Usage {
print "$0 <vcards.vcf\n";
print " normalizes one file (stdin or single argument), prints to stdout\n";
print "$0 vcards1.vcf vcards2.vcf\n";
print " compares the two files\n";
print "Also works for iCalendar files.\n";
}
sub uppercase {
my $text = shift;
$text =~ tr/a-z/A-Z/;
return $text;
}
sub sortlist {
my $list = shift;
return join(",", sort(split(/,/, $list)));
}
sub splitvalue {
my $prop = shift;
my $values = shift;
my $eol = shift;
my @res = ();
foreach my $val (split (/;/, $values)) {
push(@res, $prop, ":", $val, $eol);
}
return join("", @res);
}
# normalize the DATE-TIME duration unless the VALUE isn't a duration
sub NormalizeTrigger {
my $value = shift;
$value =~ /([+-]?)P(?:(\d*)D)?T(?:(\d*)H)?(?:(\d*)M)?(?:(\d*)S)?/;
my ($sign, $days, $hours, $minutes, $seconds) = ($1, int($2), int($3), int($4), int($5));
while ($seconds >= 60) {
$minutes++;
$seconds -= 60;
}
while ($minutes >= 60) {
$hours++;
$minutes -= 60;
}
while ($hours >= 24) {
$days++;
$hours -= 24;
}
$value = $sign;
$value .= ($days . "D") if $days;
$value .= ($hours . "H") if $hours;
$value .= ($minutes . "M") if $minutes;
$value .= ($seconds . "S") if $seconds;
return $value;
}
# decode base64 string, return size and hash
sub describeBase64 {
my $data = decode_base64($1);
return sprintf("%d b64 characters = %d bytes, %s md5sum", length($1), length($data), md5_hex($data));
}
# called for one VCALENDAR (with single VEVENT/VTODO/VJOURNAL) or VCARD,
# returns normalized one
sub NormalizeItem {
my $width = shift;
$_ = shift;
# Reduce \N to \n (both are allowed in vCard 3.0).
# Using a regular expression is a bit too broad
# because it also matches \\N, which must not be
# changed.
s/\\N/\\n/g;
# Ignore blank lines. Akonadi inserts them.
s/\n{2,}/\n/s;
# undo line continuation
s/\n\s//gs;
# ignore charset specifications, assume UTF-8
s/;CHARSET="?UTF-8"?//g;
# UID may differ, but only in vCards and journal entries:
# in calendar events the UID needs to be preserved to handle
# meeting invitations/replies correctly
s/((VCARD|VJOURNAL).*)^UID:[^\n]*\n/$1/msg;
# intentional changes to UID are acceptable when running with CLIENT_TEST_UNIQUE_UID
if ($unique_uid) {
s/UID:UNIQUE-UID-\d+-/UID:/g;
}
# merge all CATEGORIES properties into one comma-separated one
while ( s/^CATEGORIES:([^\n]*)\n(.*)^CATEGORIES:([^\n]*)\n/CATEGORIES:$1,$3\n$2/ms ) {}
# exact order of categories is irrelevant
s/^CATEGORIES:(\S+)/"CATEGORIES:" . sortlist($1)/mge;
# expand <foo> shortcuts to TYPE=<foo>
while (s/^(ADR|EMAIL|TEL)([^:\n]*);(HOME|OTHER|WORK|PARCEL|INTERNET|CAR|VOICE|CELL|PAGER)/$1;TYPE=$3/mg) {}
# the distinction between an empty and a missing property
# is vague and handled differently, so ignore empty properties
s/^[^:\n]*:;*\n//mg;
# use separate TYPE= fields
while( s/^(\w*[^:\n]*);TYPE=(\w*),(\w*)/$1;TYPE=$2;TYPE=$3/mg ) {}
# make TYPE uppercase (in vCard 3.0 at least those parameters are case-insensitive)
while( s/^(\w*[^:\n]*);TYPE=(\w*?[a-z]\w*?)([;:])/ $1 . ";TYPE=" . uppercase($2) . $3 /mge ) {}
# Replace parameters with a sorted parameter list. Cannot be done with
# a regular expression because of quoted strings. While we know exact
# parameter values, normalize them to use quoted strings if and only
# if the content is more complex than alphanumeric plus underscore and
# hyphen.
my @lines;
my ($propname, $sep, $rest);
foreach (split /\n/) {
($propname, $sep, $rest) = /^([^;:]+)([:;])(.*)/;
if ($sep eq ";") {
my @params;
my $c;
my $i = 0;
my $n = length($rest);
my $quoted = 0;
my $start = 0;
while ($i < $n) {
$c = substr($rest, $i, 1);
$i++;
if ($quoted) {
if ($c eq '"') {
$quoted = 0;
}
} else {
if ($c eq '"') {
$quoted = 1;
} elsif ($c eq ';' || $c eq ':') {
my $param = substr($rest, $start, $i - $start - 1);
my ($name, $value) = $param =~ /^([^=]*)="?([^"]*)"?$/;
if ($value =~ /^[a-zA-Z0-9_-]*$/) {
$param = $name . '=' . $value;
} else {
$param = $name . '="' . $value . '"';
}
push @params, $param;
$start = $i;
if ($c eq ':') {
last;
}
}
}
}
$_ = $propname . ';' . join(";", sort(@params)) . ':' . substr($rest, $start);
}
push @lines, $_;
}
$_ = join("\n", @lines);
# VALUE=DATE is the default, no need to show it
s/^(EXDATE|BDAY);VALUE=DATE:/\1:/mg;
# default opacity is OPAQUE
s/^TRANSP:OPAQUE\r?\n?//gm;
# multiple EXDATEs may be joined into one, use separate properties as normal form
s/^(EXDATE[^:]*):(.*)(\r?\n)/splitvalue($1, $2, $3)/mge;
# sort value lists of specific properties
s!^(RRULE.*):(.*)!$1 . ":" . join(';',sort(split(/;/, $2)))!meg;
# INTERVAL=1 is the default and thus can be removed
s/^RRULE(.*?);INTERVAL=1(;|$)/RRULE$1$2/mg;
# Ignore remaining "other" email, address and telephone type - this is
# an Evolution specific extension which might not be preserved.
s/^(ADR|EMAIL|TEL)([^:\n]*);TYPE=OTHER/$1$2/mg;
# TYPE=PREF on the other hand is not used by Evolution, but
# might be sent back.
s/^(ADR|EMAIL)([^:\n]*);TYPE=PREF/$1$2/mg;
# Evolution does not need TYPE=INTERNET for email
s/^(EMAIL)([^:\n]*);TYPE=INTERNET/$1$2/mg;
# ignore TYPE=PREF in address, does not matter in Evolution
s/^((ADR|LABEL)[^:\n]*);TYPE=PREF/$1/mg;
# ignore extra separators in multi-value fields
s/^((ORG|N|(ADR[^:\n]*?)):.*?);*$/$1/mg;
# the type of certain fields is ignore by Evolution
s/^X-(AIM|GROUPWISE|ICQ|YAHOO);TYPE=HOME/X-$1/gm;
# Evolution ignores an additional pager type
s/^TEL;TYPE=PAGER;TYPE=WORK/TEL;TYPE=PAGER/gm;
# PAGER property is sent by Evolution, but otherwise ignored
s/^LABEL[;:].*\n//mg;
# TYPE=VOICE is the default in Evolution and may or may not appear in the vcard;
# this simplification is a bit too agressive and hides the problematic
# TYPE=PREF,VOICE combination which Evolution does not handle :-/
s/^TEL([^:\n]*);TYPE=VOICE,([^:\n]*):/TEL$1;TYPE=$2:/mg;
s/^TEL([^:\n]*);TYPE=([^;:\n]*),VOICE([^:\n]*):/TEL$1;TYPE=$2$3:/mg;
s/^TEL([^:\n]*);TYPE=VOICE([^:\n]*):/TEL$1$2:/mg;
# don't care about the TYPE property of PHOTOs
s/^PHOTO;(.*)TYPE=[A-Z]*/PHOTO;$1/mg;
# encoding is not case sensitive, skip white space in the middle of binary data
if (s/^PHOTO;.*?ENCODING=(b|B|BASE64).*?:\s*/PHOTO;ENCODING=B: /mgi) {
if ($memotoo) {
# transcodes image data, can't compare it
s/(^PHOTO.*:).*/$1<stripped by synccompare>/mg;
} else {
while (s/^PHOTO(.*?): (\S+)[\t ]+(\S+)/PHOTO$1: $2$3/mg) {}
}
}
# Don't show base64 encoded PHOTO data (makes diff very long). Instead
# decode and show size + hash.
s/^PHOTO;ENCODING=B: (.*)$/"PHOTO: " . describeBase64($1)/mge;
# special case for the inlining of the local test case PHOTO
s!^PHOTO;;VALUE=uri:file://testcases/local.png$!PHOTO;;VALUE=uri:<local.png>!m;
s!^PHOTO;ENCODING=B: iVBORw0KGgoAAAANSUh.*UQOVkeH/aKBSLM04QlMqAAFNBTl\+CjN9AAAAAElFTkSuQmCC$!PHOTO;;VALUE=uri:<local.png>!m;
# ignore extra day factor in front of weekday
s/^RRULE:(.*)BYDAY=\+?1(\D)/RRULE:$1BYDAY=$2/mg;
# remove default VALUE=DATE-TIME
s/^(DTSTART|DTEND)([^:\n]*);VALUE=DATE-TIME/$1$2/mg;
# remove default LANGUAGE=en-US
s/^([^:\n]*);LANGUAGE=en-US/$1/mg;
# normalize values which look like a date to YYYYMMDD because the hyphen is optional
s/:(\d{4})-(\d{2})-(\d{2})/:$1$2$3/g;
# mailto is case insensitive
s/^((ATTENDEE|ORGANIZER).*):[Mm][Aa][Ii][Ll][Tt][Oo]:/$1:mailto:/mg;
# remove fields which may differ
s/^(PRODID|CREATED|DTSTAMP|LAST-MODIFIED|REV)(;X-VOBJ-FLOATINGTIME-ALLOWED=(TRUE|FALSE))?:.*\r?\n?//gm;
# remove optional properties and parameters
s/^(METHOD|X-WSS-[A-Z]*|X-WR-[A-Z]*|CALSCALE|X-KDE-ICAL-IMPLEMENTATION-VERSION|X-KDE-KCALCORE-ENABLED):.*\r?\n?//gm;
s/^(ATTENDEE[^:]*);X-UID=[^;:]*/$1/mg;
# trailing line break(s) in a DESCRIPTION may or may not be
# removed or added by servers
s/^DESCRIPTION:(.*?)(\\n)+$/DESCRIPTION:$1/gm;
# use the shorter property name when there are alternatives,
# but avoid duplicates
foreach my $i ("SPOUSE", "MANAGER", "ASSISTANT", "ANNIVERSARY") {
if (/^X-\Q$i\E:(.*?)$/m) {
s/^X-EVOLUTION-\Q$i\E:\Q$1\E\n//m;
}
}
s/^X-EVOLUTION-(SPOUSE|MANAGER|ASSISTANT|ANNIVERSARY)/X-$1/gm;
# some properties are always lost because we don't transmit them
if ($ENV{CLIENT_TEST_SERVER}) {
s/^(X-FOOBAR-EXTENSION|X-TEST)(;[^:;\n]*)*:.*\r?\n?//gm;
}
# if there is no DESCRIPTION in a VJOURNAL, then use the
# summary: that's what is done when exchanging such a
# VJOURNAL as plain text
if (/^BEGIN:VJOURNAL$/m && !/^DESCRIPTION/m) {
s/^SUMMARY:(.*)$/SUMMARY:$1\nDESCRIPTION:$1/m;
}
# strip configurable X- parameters or properties
my $strip = $ENV{CLIENT_TEST_STRIP_PROPERTIES};
if ($strip) {
s/^$strip(;[^:;\n]*)*:.*\r?\n?//gm;
}
$strip = $ENV{CLIENT_TEST_STRIP_PARAMETERS};
if ($strip) {
while (s/^(\w+)([^:\n]*);$strip=\d+/$1$2/mg) {}
}
testing: renamed LinkedItems tests, added "no ID" variants Numbering Client::Source::LinkedItems_xxx with xxx being a number is confusing, in particular when the same number stands for different test data. Now each set of linked items has an additional, unique name which is used for Client::Source::LinkedItems<Name>. Done in combination with adding more linked item tests and slightly reorganizing the logic for adding them: - a default set with VTIMEZONE is added in all cases - some SyncML servers override that default set - others, in particular peers accessed via their own backend, enable additional Client::Source tests on a case-by-case basis Exchange is only tested with its own default set (with "Standard Timezone" as TZID) and the all-day recurring set (as before). All other CalDAV servers are now also tested with the all-day set (previously exclusive to Exchange) and local floating time (= no TZID, new). Google CalDAV can't be tested with local time because it converts such events into the time zone of the current user. All-day events need special test data because Google adds a time to the UNTIL clause (http://code.google.com/p/google-caldav-issues/issues/detail?id=63). synccompare also needs to ignore that Google adds a redundant VTIMEZONE to the all-day test cases. Finally, Client::Source tests for updating a child event (with and without parent) without UID and RECURRENCE-ID inside the payload were added. These properties are removed via text operations. The expectation is that the source is able to add them back (if needed) based on the meta information that it has about the existing item. The file source is unable to do that. When using it in an HTTP server, the server will look to peers like a peer which doesn't support the semantic (which indeed it doesn't) and thus the client will add back the fields.
2011-11-02 12:11:48 +01:00
# strip redundant VTIMEZONE definitions (happen to be
# added by Google CalDAV server when storing an all-day event
# which doesn't need any time zone definition)
# http://code.google.com/p/google-caldav-issues/issues/detail?id=63
#
# Also strip all definitions if requested.
testing: renamed LinkedItems tests, added "no ID" variants Numbering Client::Source::LinkedItems_xxx with xxx being a number is confusing, in particular when the same number stands for different test data. Now each set of linked items has an additional, unique name which is used for Client::Source::LinkedItems<Name>. Done in combination with adding more linked item tests and slightly reorganizing the logic for adding them: - a default set with VTIMEZONE is added in all cases - some SyncML servers override that default set - others, in particular peers accessed via their own backend, enable additional Client::Source tests on a case-by-case basis Exchange is only tested with its own default set (with "Standard Timezone" as TZID) and the all-day recurring set (as before). All other CalDAV servers are now also tested with the all-day set (previously exclusive to Exchange) and local floating time (= no TZID, new). Google CalDAV can't be tested with local time because it converts such events into the time zone of the current user. All-day events need special test data because Google adds a time to the UNTIL clause (http://code.google.com/p/google-caldav-issues/issues/detail?id=63). synccompare also needs to ignore that Google adds a redundant VTIMEZONE to the all-day test cases. Finally, Client::Source tests for updating a child event (with and without parent) without UID and RECURRENCE-ID inside the payload were added. These properties are removed via text operations. The expectation is that the source is able to add them back (if needed) based on the meta information that it has about the existing item. The file source is unable to do that. When using it in an HTTP server, the server will look to peers like a peer which doesn't support the semantic (which indeed it doesn't) and thus the client will add back the fields.
2011-11-02 12:11:48 +01:00
while (m/(BEGIN:VTIMEZONE.*?TZID:([^\n]*)\n.*?END:VTIMEZONE\n)/gs) {
my $def = $1;
my $tzid = $2;
# Strip all, or not used as parameter?
if ($no_timezones || ! m/;TZID="?\Q$tzid\E"?/) {
testing: renamed LinkedItems tests, added "no ID" variants Numbering Client::Source::LinkedItems_xxx with xxx being a number is confusing, in particular when the same number stands for different test data. Now each set of linked items has an additional, unique name which is used for Client::Source::LinkedItems<Name>. Done in combination with adding more linked item tests and slightly reorganizing the logic for adding them: - a default set with VTIMEZONE is added in all cases - some SyncML servers override that default set - others, in particular peers accessed via their own backend, enable additional Client::Source tests on a case-by-case basis Exchange is only tested with its own default set (with "Standard Timezone" as TZID) and the all-day recurring set (as before). All other CalDAV servers are now also tested with the all-day set (previously exclusive to Exchange) and local floating time (= no TZID, new). Google CalDAV can't be tested with local time because it converts such events into the time zone of the current user. All-day events need special test data because Google adds a time to the UNTIL clause (http://code.google.com/p/google-caldav-issues/issues/detail?id=63). synccompare also needs to ignore that Google adds a redundant VTIMEZONE to the all-day test cases. Finally, Client::Source tests for updating a child event (with and without parent) without UID and RECURRENCE-ID inside the payload were added. These properties are removed via text operations. The expectation is that the source is able to add them back (if needed) based on the meta information that it has about the existing item. The file source is unable to do that. When using it in an HTTP server, the server will look to peers like a peer which doesn't support the semantic (which indeed it doesn't) and thus the client will add back the fields.
2011-11-02 12:11:48 +01:00
# no, remove definition
s!\Q$def\E!!s;
testing: renamed LinkedItems tests, added "no ID" variants Numbering Client::Source::LinkedItems_xxx with xxx being a number is confusing, in particular when the same number stands for different test data. Now each set of linked items has an additional, unique name which is used for Client::Source::LinkedItems<Name>. Done in combination with adding more linked item tests and slightly reorganizing the logic for adding them: - a default set with VTIMEZONE is added in all cases - some SyncML servers override that default set - others, in particular peers accessed via their own backend, enable additional Client::Source tests on a case-by-case basis Exchange is only tested with its own default set (with "Standard Timezone" as TZID) and the all-day recurring set (as before). All other CalDAV servers are now also tested with the all-day set (previously exclusive to Exchange) and local floating time (= no TZID, new). Google CalDAV can't be tested with local time because it converts such events into the time zone of the current user. All-day events need special test data because Google adds a time to the UNTIL clause (http://code.google.com/p/google-caldav-issues/issues/detail?id=63). synccompare also needs to ignore that Google adds a redundant VTIMEZONE to the all-day test cases. Finally, Client::Source tests for updating a child event (with and without parent) without UID and RECURRENCE-ID inside the payload were added. These properties are removed via text operations. The expectation is that the source is able to add them back (if needed) based on the meta information that it has about the existing item. The file source is unable to do that. When using it in an HTTP server, the server will look to peers like a peer which doesn't support the semantic (which indeed it doesn't) and thus the client will add back the fields.
2011-11-02 12:11:48 +01:00
}
}
if (!$full_timezones) {
# Strip trailing digits from TZID. They are appended by
# Evolution and SyncEvolution to distinguish VTIMEZONE
# definitions which have the same TZID, but different rules.
# This appending of digits may even get repeated, leading to:
# TZID=EST/EDT 1 1
s/(^TZID:|;TZID=)([^;:]*?)( \d+)+/$1$2/gm;
# Strip trailing -(Standard) from TZID. Evolution 2.24.5 adds
# that (not sure exactly where that comes from).
s/(^TZID:|;TZID=)([^;:]*?)-\(Standard\)/$1$2/gm;
# VTIMEZONE and TZID do not have to be preserved verbatim as long
# as the replacement is still representing the same timezone.
# Reduce TZIDs which specify a proper location
# to their location part and strip the VTIMEZONE - makes the
# diff shorter, too.
my $location = "[^\n]*((?:Africa|America|Antarctica|Arctic|Asia|Atlantic|Australia|Brazil|Canada|Chile|Egypt|Eire|Europe|Hongkong|Iceland|India|Iran|Israel|Jamaica|Japan|Kwajalein|Libya|Mexico|Mideast|Navajo|Pacific|Poland|Portugal|Singapore|Turkey|Zulu)[-a-zA-Z0-9_/]*)";
s;^BEGIN:VTIMEZONE.*?^TZID:$location.*^END:VTIMEZONE;BEGIN:VTIMEZONE\n TZID:$1 [...]\nEND:VTIMEZONE;gms;
s;TZID="?$location"?;TZID=$1;gm;
}
# normalize iCalendar 2.0
if (/^BEGIN:(VEVENT|VTODO|VJOURNAL)$/m) {
# CLASS=PUBLIC is the default, no need to show it
s/^CLASS:PUBLIC\r?\n//m;
# RELATED=START is the default behavior
s/^TRIGGER([^\n:]*);RELATED=START/TRIGGER$1/mg;
# VALUE=DURATION is the default behavior
s/^TRIGGER([^\n:]*);VALUE=DURATION/TRIGGER$1/mg;
s/^(TRIGGER.*):(\S*)/$1 . ":" . NormalizeTrigger($2)/mge;
# INDIVIDUAL is default for CUTYPE.
s/;CUTYPE=INDIVIDUAL([;:])/$1/mg;
# Print without quotation marks (probably not save in general, but okay for our test data).
s/;CN="([^;]*)"/;CN=$1/g;
}
synccompare: avoid segfault in Perl interpreter by limiting UTF-8 support The Perl version in Ubuntu Lucid had even more problems with synccompare than other versions. Compiling the latest stable Perl also didn't help (stack backtrace below). This patch disables UTF-8 support in string operations. The user-visible effect is that line length calculation is wrong when multi-byte characters are involved. This is better than not getting any output when the installed Perl is bad. File input and stdout are still using UTF-8, so UTF-8 content is passed through okay. *** glibc detected *** /usr/local/bin/perl-synccompare: double free or corruption (!prev): 0x08203278 *** ======= Backtrace: ========= /lib/tls/i686/cmov/libc.so.6(+0x6b591)[0xf7616591] /lib/tls/i686/cmov/libc.so.6(+0x6cde8)[0xf7617de8] /lib/tls/i686/cmov/libc.so.6(cfree+0x6d)[0xf761aecd] /usr/local/bin/perl-synccompare(Perl_offer_nice_chunk+0x3d)[0x80d2b0d] /usr/local/bin/perl-synccompare[0x80c426a] /usr/local/bin/perl-synccompare(Perl_hv_common+0xafa)[0x80c628a] /usr/local/bin/perl-synccompare(Perl_pp_helem+0x24a)[0x80d138a] /usr/local/bin/perl-synccompare(Perl_runops_standard+0x13)[0x80c9623] /usr/local/bin/perl-synccompare[0x80f74b0] /usr/local/bin/perl-synccompare(Perl_pp_require+0x11da)[0x80f9c7a] /usr/local/bin/perl-synccompare(Perl_runops_standard+0x13)[0x80c9623] /usr/local/bin/perl-synccompare(Perl_call_sv+0x165)[0x8077705] /usr/local/bin/perl-synccompare(Perl_swash_init+0x1d6)[0x8128686] /usr/local/bin/perl-synccompare(Perl_to_utf8_case+0x213)[0x812a723] /usr/local/bin/perl-synccompare(Perl_to_utf8_lower+0x37)[0x812a7f7] /usr/local/bin/perl-synccompare[0x8121280] /usr/local/bin/perl-synccompare(Perl_regexec_flags+0xb58)[0x8126b48] /usr/local/bin/perl-synccompare(Perl_pp_subst+0x207)[0x80d1917] /usr/local/bin/perl-synccompare(Perl_runops_standard+0x13)[0x80c9623] /usr/local/bin/perl-synccompare(perl_run+0x2bd)[0x807823d] /usr/local/bin/perl-synccompare(main+0x115)[0x8062c35] /lib/tls/i686/cmov/libc.so.6(__libc_start_main+0xe6)[0xf75c1bd6] /usr/local/bin/perl-synccompare[0x8062a81] ======= Memory map: ======== 08048000-0815e000 r-xp 00000000 08:05 2007582 /home/nightly/perl-5.12.3-lucid-i386/bin/perl
2011-05-05 14:08:32 +02:00
# Added by EDS >= 2.32, presumably to cache some internal computation.
# Because it can be recreated, it doesn't have to be preserved during
# sync and such changes can be ignored:
#
# RRULE:BYDAY=SU;COUNT=10;FREQ=WEEKLY | RRULE;X-EVOLUTION-ENDDATE=20080608T
# > 070000Z:BYDAY=SU;COUNT=10;FREQ=WEEK
# > LY
s/^(\w+)([^:\n]*);X-EVOLUTION-ENDDATE=[0-9TZ]*/$1$2/mg;
if ($scheduleworld || $egroupware || $synthesis || $addressbook || $funambol ||$googlesyncml || $googleeas || $googlecarddav || $mobical || $memotoo || $zimbra) {
# does not preserve X-EVOLUTION-UI-SLOT=
s/^(\w+)([^:\n]*);X-EVOLUTION-UI-SLOT=\d+/$1$2/mg;
}
if ($scheduleworld) {
# cannot distinguish EMAIL types
s/^EMAIL;TYPE=\w*/EMAIL/mg;
# replaces certain TZIDs with more up-to-date ones
s;TZID(=|:)/(scheduleworld.com|softwarestudio.org)/Olson_\d+_\d+/;TZID$1/foo.com/Olson_20000101_1/;mg;
}
if ($synthesis || $mobical) {
# only preserves ORG "Company", but loses "Department" and "Office"
s/^ORG:([^;:\n]+)(;[^\n]*)/ORG:$1/mg;
}
if ($funambol) {
# only preserves ORG "Company";"Department", but loses "Office"
s/^ORG:([^;:\n]+)(;[^;:\n]*)(;[^\n]*)/ORG:$1$2/mg;
myFUNAMBOL looses some data that was preserved by Funambol 3.0. synccompare.pl now simplifies the test data so that the Client::Sync::vcard21::testItems passes again. For an example of what gets lost see the failed test: BEGIN:VCARD BEGIN:VCARD N:Doe;John N:Doe;John ADR;TYPE=HOME:Test Box #1;;Test Drive ADR;TYPE=HOME:Test Box #1;;Test Drive 1;Test Village;Lower Test County;123 1;Test Village;Lower Test County;123 45;Testovia 45;Testovia ADR;TYPE=PARCEL:Test Box #3;;Test Dri | ADR;TYPE=HOME:Test Box #3;;Test Drive ve 3;Test Megacity;Test County;12347; | 3;Test Megacity;Test County;12347;Ne New Testonia | w Testonia ADR;TYPE=WORK:Test Box #2;;Test Drive ADR;TYPE=WORK:Test Box #2;;Test Drive 2;Test Town;Upper Test County;12346; 2;Test Town;Upper Test County;12346; Old Testovia Old Testovia BDAY:2006-01-08 BDAY:2006-01-08 CALURI:calender < CATEGORIES:TEST CATEGORIES:TEST EMAIL;TYPE=HOME;X-EVOLUTION-UI-SLOT=2 | EMAIL:john.doe@other.world :john.doe@home.priv | EMAIL;TYPE=HOME:john.doe@home.priv EMAIL;TYPE=WORK;X-EVOLUTION-UI-SLOT=1 | EMAIL;TYPE=WORK:john.doe@work.com :john.doe@work.com < EMAIL;X-EVOLUTION-UI-SLOT=3:john.doe@ < other.world < EMAIL;X-EVOLUTION-UI-SLOT=4:john.doe@ < yet.another.world < FBURL:free/busy < NICKNAME:user1 NICKNAME:user1 NOTE:This is a test case which uses a NOTE:This is a test case which uses a lmost all Evolution fields. lmost all Evolution fields. ORG:Test Inc.;Testing ORG:Test Inc.;Testing ROLE:professional test case ROLE:professional test case TEL;TYPE=CAR;X-EVOLUTION-UI-SLOT=7:ca | TEL;TYPE=CAR:car 7 r 7 | TEL;TYPE=CELL:mobile 3 TEL;TYPE=CELL;X-EVOLUTION-UI-SLOT=3:m | TEL;TYPE=FAX;TYPE=HOME:homefax 5 obile 3 | TEL;TYPE=FAX;TYPE=WORK:businessfax 4 TEL;TYPE=FAX;TYPE=HOME;X-EVOLUTION-UI | TEL;TYPE=HOME:home 2 -SLOT=5:homefax 5 | TEL;TYPE=PAGER:pager 6 TEL;TYPE=FAX;TYPE=WORK;X-EVOLUTION-UI | TEL;TYPE=PREF:primary 8 -SLOT=4:businessfax 4 | TEL;TYPE=WORK:business 1 TEL;TYPE=HOME;X-EVOLUTION-UI-SLOT=2:h < ome 2 < TEL;TYPE=PAGER;X-EVOLUTION-UI-SLOT=6: < pager 6 < TEL;TYPE=PREF;X-EVOLUTION-UI-SLOT=8:p < rimary 8 < TEL;TYPE=WORK;X-EVOLUTION-UI-SLOT=1:b < usiness 1 < TITLE:Senior Tester TITLE:Senior Tester URL:http://john.doe.com URL:http://john.doe.com VERSION:3.0 VERSION:3.0 X-AIM;X-EVOLUTION-UI-SLOT=1:AIM JOHN < X-EVOLUTION-ANNIVERSARY:2006-01-09 < X-EVOLUTION-ASSISTANT:John Doe Junior < X-EVOLUTION-BLOG-URL:web log < X-EVOLUTION-FILE-AS:Doe\, John X-EVOLUTION-FILE-AS:Doe\, John X-EVOLUTION-MANAGER:John Doe Senior < X-EVOLUTION-SPOUSE:Joan Doe < X-EVOLUTION-VIDEO-URL:chat < X-GROUPWISE;X-EVOLUTION-UI-SLOT=4:GRO < UPWISE DOE < X-ICQ;X-EVOLUTION-UI-SLOT=3:ICQ JD < X-YAHOO;X-EVOLUTION-UI-SLOT=2:YAHOO J < DOE < END:VCARD END:VCARD ------------------------------------------------------------------------------- BEGIN:VCARD BEGIN:VCARD N:breaks;line N:breaks;line ADR;TYPE=HOME:;Address Line 2\nAddres | ADR;TYPE=HOME:;;Address Line 1 s Line 3;Address Line 1 < NICKNAME:user7 NICKNAME:user7 NOTE:This test case uses line breaks. NOTE:This test case uses line breaks. This is line 1.\nLine 2.\n\nLine bre This is line 1.\nLine 2.\n\nLine bre aks in vcard 2.1 are encoded as =0D=0 aks in vcard 2.1 are encoded as =0D=0 A.\nThat means the = has to be encode A.\nThat means the = has to be encode d itself... d itself... VERSION:3.0 VERSION:3.0 X-EVOLUTION-FILE-AS:breaks\, line X-EVOLUTION-FILE-AS:breaks\, line END:VCARD END:VCARD git-svn-id: https://core.forge.funambol.org/svn/core/top-level/branches/b_v65/3x/client-api/native@20761 e8e8ed6c-164c-0410-afcf-9e9a7c7d8c10
2007-11-07 22:30:36 +01:00
# drops the second address line
s/^ADR(.*?):([^;]*?);[^;]*?;/ADR$1:$2;;/mg;
# has no concept of "preferred" phone number
s/^(TEL.*);TYPE=PREF/$1/mg;
myFUNAMBOL looses some data that was preserved by Funambol 3.0. synccompare.pl now simplifies the test data so that the Client::Sync::vcard21::testItems passes again. For an example of what gets lost see the failed test: BEGIN:VCARD BEGIN:VCARD N:Doe;John N:Doe;John ADR;TYPE=HOME:Test Box #1;;Test Drive ADR;TYPE=HOME:Test Box #1;;Test Drive 1;Test Village;Lower Test County;123 1;Test Village;Lower Test County;123 45;Testovia 45;Testovia ADR;TYPE=PARCEL:Test Box #3;;Test Dri | ADR;TYPE=HOME:Test Box #3;;Test Drive ve 3;Test Megacity;Test County;12347; | 3;Test Megacity;Test County;12347;Ne New Testonia | w Testonia ADR;TYPE=WORK:Test Box #2;;Test Drive ADR;TYPE=WORK:Test Box #2;;Test Drive 2;Test Town;Upper Test County;12346; 2;Test Town;Upper Test County;12346; Old Testovia Old Testovia BDAY:2006-01-08 BDAY:2006-01-08 CALURI:calender < CATEGORIES:TEST CATEGORIES:TEST EMAIL;TYPE=HOME;X-EVOLUTION-UI-SLOT=2 | EMAIL:john.doe@other.world :john.doe@home.priv | EMAIL;TYPE=HOME:john.doe@home.priv EMAIL;TYPE=WORK;X-EVOLUTION-UI-SLOT=1 | EMAIL;TYPE=WORK:john.doe@work.com :john.doe@work.com < EMAIL;X-EVOLUTION-UI-SLOT=3:john.doe@ < other.world < EMAIL;X-EVOLUTION-UI-SLOT=4:john.doe@ < yet.another.world < FBURL:free/busy < NICKNAME:user1 NICKNAME:user1 NOTE:This is a test case which uses a NOTE:This is a test case which uses a lmost all Evolution fields. lmost all Evolution fields. ORG:Test Inc.;Testing ORG:Test Inc.;Testing ROLE:professional test case ROLE:professional test case TEL;TYPE=CAR;X-EVOLUTION-UI-SLOT=7:ca | TEL;TYPE=CAR:car 7 r 7 | TEL;TYPE=CELL:mobile 3 TEL;TYPE=CELL;X-EVOLUTION-UI-SLOT=3:m | TEL;TYPE=FAX;TYPE=HOME:homefax 5 obile 3 | TEL;TYPE=FAX;TYPE=WORK:businessfax 4 TEL;TYPE=FAX;TYPE=HOME;X-EVOLUTION-UI | TEL;TYPE=HOME:home 2 -SLOT=5:homefax 5 | TEL;TYPE=PAGER:pager 6 TEL;TYPE=FAX;TYPE=WORK;X-EVOLUTION-UI | TEL;TYPE=PREF:primary 8 -SLOT=4:businessfax 4 | TEL;TYPE=WORK:business 1 TEL;TYPE=HOME;X-EVOLUTION-UI-SLOT=2:h < ome 2 < TEL;TYPE=PAGER;X-EVOLUTION-UI-SLOT=6: < pager 6 < TEL;TYPE=PREF;X-EVOLUTION-UI-SLOT=8:p < rimary 8 < TEL;TYPE=WORK;X-EVOLUTION-UI-SLOT=1:b < usiness 1 < TITLE:Senior Tester TITLE:Senior Tester URL:http://john.doe.com URL:http://john.doe.com VERSION:3.0 VERSION:3.0 X-AIM;X-EVOLUTION-UI-SLOT=1:AIM JOHN < X-EVOLUTION-ANNIVERSARY:2006-01-09 < X-EVOLUTION-ASSISTANT:John Doe Junior < X-EVOLUTION-BLOG-URL:web log < X-EVOLUTION-FILE-AS:Doe\, John X-EVOLUTION-FILE-AS:Doe\, John X-EVOLUTION-MANAGER:John Doe Senior < X-EVOLUTION-SPOUSE:Joan Doe < X-EVOLUTION-VIDEO-URL:chat < X-GROUPWISE;X-EVOLUTION-UI-SLOT=4:GRO < UPWISE DOE < X-ICQ;X-EVOLUTION-UI-SLOT=3:ICQ JD < X-YAHOO;X-EVOLUTION-UI-SLOT=2:YAHOO J < DOE < END:VCARD END:VCARD ------------------------------------------------------------------------------- BEGIN:VCARD BEGIN:VCARD N:breaks;line N:breaks;line ADR;TYPE=HOME:;Address Line 2\nAddres | ADR;TYPE=HOME:;;Address Line 1 s Line 3;Address Line 1 < NICKNAME:user7 NICKNAME:user7 NOTE:This test case uses line breaks. NOTE:This test case uses line breaks. This is line 1.\nLine 2.\n\nLine bre This is line 1.\nLine 2.\n\nLine bre aks in vcard 2.1 are encoded as =0D=0 aks in vcard 2.1 are encoded as =0D=0 A.\nThat means the = has to be encode A.\nThat means the = has to be encode d itself... d itself... VERSION:3.0 VERSION:3.0 X-EVOLUTION-FILE-AS:breaks\, line X-EVOLUTION-FILE-AS:breaks\, line END:VCARD END:VCARD git-svn-id: https://core.forge.funambol.org/svn/core/top-level/branches/b_v65/3x/client-api/native@20761 e8e8ed6c-164c-0410-afcf-9e9a7c7d8c10
2007-11-07 22:30:36 +01:00
}
if($googlesyncml || $googleeas || $googlecarddav) {
# ignore the PHOTO encoding data
s/^PHOTO(.*?): .*\n/PHOTO$1: [...]\n/mg;
}
if($googlesyncml || $googlecarddav) {
# FN property gets synthesized by Google.
s/^FN:.*\n/FN$1: [...]\n/mg;
}
CardDAV: use Apple/Google/CardDAV vCard flavor In principle, CardDAV servers support arbitrary vCard 3.0 data. Extensions can be different and need to be preserved. However, when multiple different clients or the server's Web UI interpret the vCards, they need to agree on the semantic of these vCard extensions. In practice, CardDAV was pushed by Apple and Apple clients are probably the most common clients of CardDAV services. When the Google Contacts Web UI creates or edits a contact, Google CardDAV will send that data using the vCard flavor used by Apple. Therefore it makes sense to exchange contacts with *all* CardDAV servers using that format. This format could be made configurable in SyncEvolution on a case-by-case basis; at the moment, it is hard-coded. During syncing, SyncEvolution takes care to translate between the vCard flavor used internally (based on Evolution) and the CardDAV vCard flavor. This mapping includes: X-AIM/JABBER/... <-> IMPP + X-SERVICE-TYPE Any IMPP property declared as X-SERVICE-TYPE=AIM will get mapped to X-AIM. Same for others. Some IMPP service types have no known X- property extension; they are stored in EDS as IMPP. X- property extensions without a known X-SERVICE-TYPE (for example, GaduGadu and Groupwise) are stored with X-SERVICE-TYPE values chosen by SyncEvolution so that Google CardDAV preserves them (GroupWise with mixed case got translated by Google into Groupwise, so the latter is used). Google always sends an X-ABLabel:Other for IMPP. This is ignored because the service type overrides it. The value itself also gets transformed during the mapping. IMPP uses an URI as value, with a chat protocol (like "aim" or "xmpp") and some protocol specific identifier. For each X- extension the protocol is determined by the property name and the value is the protocol specific identifier without URL encoding. X-SPOUSE/MANAGER/ASSISTANT <-> X-ABRELATEDNAMES + X-ABLabel The mapping is based on the X-ABLabel property attached to the X-ABRELATEDNAMES property. This depends on the English words "Spouse", "Manager", "Assistant" that Google CardDAV and Apple devices seem to use regardless of the configured language. As with IMPP, only the subset of related names which have a corresponding X- property extension get mapped. The rest is stored in EDS using the X-ABRELATEDNAMES property. X-ANNIVERSARY <-> X-ABDATE Same here, with X-ABLabel:Anniversary as the special case which gets mapped. X-ABLabel parameter <-> property CardDAV vCards have labels attached to arbitrary other properties (TEL, ADR, X-ABDATE, X-ABRELATEDNAMES, ...) via vCard group tags: item1.X-ABDATE:2010-01-01 item1.X-ABLabel:Anniversary The advantage is that property values can contain arbitrary characters, including line breaks and double quotation marks, which is not possible in property parameters. Neither EDS nor KDE (judging from the lack of responses on the KDE-PIM mailing list) support custom labels. SyncEvolution could have used grouping as it is done in CardDAV, but grouping is not used much (not at all?) by the UIs working with the vCards in EDS and KDE. It seemed easier to use a new X-ABLabel parameter. Characters which cannot be stored in a parameter get converted (double space to single space, line break to space, etc.) during syncing. In practice, these characters don't appear in X-ABLabel properties anyway because neither Apple nor Google UIs allow entering them for custom labels. The "Other" label is used by Google even in case where it adds no information. For example, all XMPP properties have an associated X-ABLabel=Other although the Web UI does not provide a means to edit or show such a label. Editing the text before the value in the UI changes the X-SERVICE-TYPE parameter value, not the X-ABLabel as for other fields. Therefore the "Other" label is ignored by removing it during syncing. X-EVOLUTION-UI-SLOT (the parameter used in Evolution to determine the order of properties in the UI) gets stored in CardDAV. The only exception is Google CardDAV which got confused when an IMPP property had both X-SERVICE-TYPE and X-EVOLUTION-UI-SLOT parameters set. For Google, X-EVOLUTION-UI-SLOT is only sent on other properties and thus ordering of chat information can get lost when syncing with Google. CardDAV needs to use test data with the new CardDAV vCard flavor. Most CardDAV servers can store EDS vCards and thus Client::Source::*::testImport passed (with minor tweaks in synccompare) when using the default eds_contact.vcf, but Client::Sync::*::testItems fails when comparing synced data with test cases when the synced data uses the native format and the test cases are still the ones from EDS. A libsynthesis with URLENCODE/DECODE() and sharedfield parameter for <property> is needed.
2014-05-16 11:25:00 +02:00
# Properties and parameters are case-insensitive. ownCloud uses
# X-ABLABEL while everyone else uses X-ABLabel.
s/X-ABLABEL/X-ABLabel/g;
if ($googlesyncml) {
# Not support car type in telephone
s!^TEL\;TYPE=CAR(.*)\n!TEL$1\n!mg;
# some properties are lost
s/^(X-EVOLUTION-FILE-AS|NICKNAME|BDAY|CATEGORIES|CALURI|FBURL|GEO|ROLE|URL|X-AIM|X-EVOLUTION-UI-SLOT|X-ANNIVERSARY|X-ASSISTANT|X-EVOLUTION-BLOG-URL|X-EVOLUTION-VIDEO-URL|X-GROUPWISE|X-ICQ|X-GADUGADU|X-JABBER|X-MSN|X-SIP|X-SKYPE|X-MANAGER|X-SPOUSE|X-MOZILLA-HTML|X-YAHOO)(;[^:;\n]*)*:.*\r?\n?//gm;
}
if ($googlecaldav) {
#several properties are not preserved by Google in icalendar2.0 format
s/^(SEQUENCE|X-EVOLUTION-ALARM-UID|TRANSP)(;[^:;\n]*)*:.*\r?\n?//gm;
# Google adds calendar owner as attendee of meetings, regardless
# whether it was on the original attendee list. Ignore this
# during testing by removing all attendees with @googlemail.com
# email address.
s/^ATTENDEE.*googlemail.com\r?\n//gm;
}
if ($apple) {
# remove some parameters added by Apple Calendar server in CalDAV
s/^(ORGANIZER[^:]*);SCHEDULE-AGENT=NONE/$1/gm;
s/^(ORGANIZER[^:]*);SCHEDULE-STATUS=5.3/$1/gm;
# seems to require a fixed number of recurrences; hmm, okay...
s/^RRULE:COUNT=400;FREQ=DAILY/RRULE:FREQ=DAILY/gm;
}
if ($oracle) {
# remove extensions added by server
s/^(X-S1CS-RECURRENCE-COUNT)(;[^:;\n]*)*:.*\r?\n?//gm;
# ignore loss of LANGUAGE=xxx property in ATTENDEE
s/^ATTENDEE([^\n:]*);LANGUAGE=([^\n;:]*)/ATTENDEE$1/mg;
}
if ($radicale) {
# remove extensions added by server
s/^(X-RADICALE-NAME)(;[^:;\n]*)*:.*\r?\n?//gm;
}
if ($googlecaldav || $yahoo) {
# default status is CONFIRMED
s/^STATUS:CONFIRMED\r?\n?//gm;
}
# Ignore VALARM ACTION:NONE. This has to be added to avoid default alarms in Google CalDAV.
s/^BEGIN:VALARM\r?\n.*?^ACTION:NONE\r?\n.*?^END:VALARM\r?\n//ms;
if ($yahoo) {
s/^(X-MICROSOFT-[-A-Z0-9]*)(;[^:;\n]*)*:.*\r?\n?//gm;
# some properties cannot be stored
s/^(FN)(;[^:;\n]*)*:.*\r?\n?//gm;
}
if ($addressbook) {
# some properties cannot be stored
s/^(X-MOZILLA-HTML|X-EVOLUTION-FILE-AS|X-EVOLUTION-ANNIVERSARY|X-EVOLUTION-BLOG-URL|X-EVOLUTION-VIDEO-URL|X-GROUPWISE|ROLE|CATEGORIES|FBURL|CALURI|FN)(;[^:;\n]*)*:.*\r?\n?//gm;
# only some parts of ADR are preserved
my $type;
s/^ADR(.*?)\:(.*)/$type=($1 || ""); @_ = split(\/(?<!\\);\/, $2); "ADR:;;" . ($_[2] || "") . ";" . ($_[3] || "") . ";" . ($_[4] || "") . ";" . ($_[5] || "") . ";" . ($_[6] || "")/gme;
# TYPE=CAR not supported
s/;TYPE=CAR//g;
}
if ($synthesis) {
# does not preserve certain properties
s/^(FN|GEO|BDAY|X-MOZILLA-HTML|X-EVOLUTION-FILE-AS|X-AIM|NICKNAME|UID|PHOTO|CALURI|SEQUENCE|TRANSP|ORGANIZER|ROLE|FBURL|X-ANNIVERSARY|X-ASSISTANT|X-EVOLUTION-BLOG-URL|X-EVOLUTION-VIDEO-URL|X-GADUGADU|X-GROUPWISE|X-ICQ|X-JABBER|X-MANAGER|X-MSN|X-SIP|X-SKYPE|X-SPOUSE|X-YAHOO)(;[^:;\n]*)*:.*\r?\n?//gm;
# default ADR is HOME
s/^ADR;TYPE=HOME/ADR/gm;
# only some parts of N are preserved
s/^N((?:;[^;:]*)*)\:(.*)/@_ = split(\/(?<!\\);\/, $2); "N$1:$_[0];" . ($_[1] || "") . ";;" . ($_[3] || "")/gme;
# breaks lines at semicolons, which adds white space
while( s/^ADR:(.*); +/ADR:$1;/gm ) {}
# no attributes stored for ATTENDEEs
s/^ATTENDEE;.*?:/ATTENDEE:/msg;
}
if ($synthesis) {
# VALARM not supported
s/^BEGIN:VALARM.*?END:VALARM\r?\n?//msg;
}
if ($egroupware) {
# CLASS:PUBLIC is added if none exists (as in our test cases),
# several properties not preserved
s/^(BDAY|CATEGORIES|FBURL|PHOTO|FN|X-[A-Z-]*|CALURI|CLASS|NICKNAME|UID|TRANSP|PRIORITY|SEQUENCE)(;[^:;\n]*)*:.*\r?\n?//gm;
# org gets truncated
s/^ORG:([^;:\n]*);.*/ORG:$1/gm;
}
if ($funambol) {
# several properties are not preserved
s/^(CALURI|FBURL|GEO|X-MOZILLA-HTML|X-EVOLUTION-FILE-AS|X-AIM|X-EVOLUTION-BLOG-URL|X-EVOLUTION-VIDEO-URL|X-GROUPWISE|X-ICQ|X-YAHOO|X-GADUGADU|X-JABBER|X-MSN|X-SIP|X-SKYPE|X-ASSISTANT)(;[^:;\n]*)*:.*\r?\n?//gm;
# quoted-printable line breaks are =0D=0A, not just single =0A
s/(?<!=0D)=0A/=0D=0A/g;
myFUNAMBOL looses some data that was preserved by Funambol 3.0. synccompare.pl now simplifies the test data so that the Client::Sync::vcard21::testItems passes again. For an example of what gets lost see the failed test: BEGIN:VCARD BEGIN:VCARD N:Doe;John N:Doe;John ADR;TYPE=HOME:Test Box #1;;Test Drive ADR;TYPE=HOME:Test Box #1;;Test Drive 1;Test Village;Lower Test County;123 1;Test Village;Lower Test County;123 45;Testovia 45;Testovia ADR;TYPE=PARCEL:Test Box #3;;Test Dri | ADR;TYPE=HOME:Test Box #3;;Test Drive ve 3;Test Megacity;Test County;12347; | 3;Test Megacity;Test County;12347;Ne New Testonia | w Testonia ADR;TYPE=WORK:Test Box #2;;Test Drive ADR;TYPE=WORK:Test Box #2;;Test Drive 2;Test Town;Upper Test County;12346; 2;Test Town;Upper Test County;12346; Old Testovia Old Testovia BDAY:2006-01-08 BDAY:2006-01-08 CALURI:calender < CATEGORIES:TEST CATEGORIES:TEST EMAIL;TYPE=HOME;X-EVOLUTION-UI-SLOT=2 | EMAIL:john.doe@other.world :john.doe@home.priv | EMAIL;TYPE=HOME:john.doe@home.priv EMAIL;TYPE=WORK;X-EVOLUTION-UI-SLOT=1 | EMAIL;TYPE=WORK:john.doe@work.com :john.doe@work.com < EMAIL;X-EVOLUTION-UI-SLOT=3:john.doe@ < other.world < EMAIL;X-EVOLUTION-UI-SLOT=4:john.doe@ < yet.another.world < FBURL:free/busy < NICKNAME:user1 NICKNAME:user1 NOTE:This is a test case which uses a NOTE:This is a test case which uses a lmost all Evolution fields. lmost all Evolution fields. ORG:Test Inc.;Testing ORG:Test Inc.;Testing ROLE:professional test case ROLE:professional test case TEL;TYPE=CAR;X-EVOLUTION-UI-SLOT=7:ca | TEL;TYPE=CAR:car 7 r 7 | TEL;TYPE=CELL:mobile 3 TEL;TYPE=CELL;X-EVOLUTION-UI-SLOT=3:m | TEL;TYPE=FAX;TYPE=HOME:homefax 5 obile 3 | TEL;TYPE=FAX;TYPE=WORK:businessfax 4 TEL;TYPE=FAX;TYPE=HOME;X-EVOLUTION-UI | TEL;TYPE=HOME:home 2 -SLOT=5:homefax 5 | TEL;TYPE=PAGER:pager 6 TEL;TYPE=FAX;TYPE=WORK;X-EVOLUTION-UI | TEL;TYPE=PREF:primary 8 -SLOT=4:businessfax 4 | TEL;TYPE=WORK:business 1 TEL;TYPE=HOME;X-EVOLUTION-UI-SLOT=2:h < ome 2 < TEL;TYPE=PAGER;X-EVOLUTION-UI-SLOT=6: < pager 6 < TEL;TYPE=PREF;X-EVOLUTION-UI-SLOT=8:p < rimary 8 < TEL;TYPE=WORK;X-EVOLUTION-UI-SLOT=1:b < usiness 1 < TITLE:Senior Tester TITLE:Senior Tester URL:http://john.doe.com URL:http://john.doe.com VERSION:3.0 VERSION:3.0 X-AIM;X-EVOLUTION-UI-SLOT=1:AIM JOHN < X-EVOLUTION-ANNIVERSARY:2006-01-09 < X-EVOLUTION-ASSISTANT:John Doe Junior < X-EVOLUTION-BLOG-URL:web log < X-EVOLUTION-FILE-AS:Doe\, John X-EVOLUTION-FILE-AS:Doe\, John X-EVOLUTION-MANAGER:John Doe Senior < X-EVOLUTION-SPOUSE:Joan Doe < X-EVOLUTION-VIDEO-URL:chat < X-GROUPWISE;X-EVOLUTION-UI-SLOT=4:GRO < UPWISE DOE < X-ICQ;X-EVOLUTION-UI-SLOT=3:ICQ JD < X-YAHOO;X-EVOLUTION-UI-SLOT=2:YAHOO J < DOE < END:VCARD END:VCARD ------------------------------------------------------------------------------- BEGIN:VCARD BEGIN:VCARD N:breaks;line N:breaks;line ADR;TYPE=HOME:;Address Line 2\nAddres | ADR;TYPE=HOME:;;Address Line 1 s Line 3;Address Line 1 < NICKNAME:user7 NICKNAME:user7 NOTE:This test case uses line breaks. NOTE:This test case uses line breaks. This is line 1.\nLine 2.\n\nLine bre This is line 1.\nLine 2.\n\nLine bre aks in vcard 2.1 are encoded as =0D=0 aks in vcard 2.1 are encoded as =0D=0 A.\nThat means the = has to be encode A.\nThat means the = has to be encode d itself... d itself... VERSION:3.0 VERSION:3.0 X-EVOLUTION-FILE-AS:breaks\, line X-EVOLUTION-FILE-AS:breaks\, line END:VCARD END:VCARD git-svn-id: https://core.forge.funambol.org/svn/core/top-level/branches/b_v65/3x/client-api/native@20761 e8e8ed6c-164c-0410-afcf-9e9a7c7d8c10
2007-11-07 22:30:36 +01:00
# only three email addresses, fourth one from test case gets lost
s/^EMAIL:john.doe\@yet.another.world\n\r?//mg;
# this particular type is not preserved
s/ADR;TYPE=PARCEL:Test Box #3/ADR;TYPE=HOME:Test Box #3/;
}
if ($funambol) {
#several properties are not preserved by funambol server in icalendar2.0 format
s/^(UID|SEQUENCE|TRANSP|LAST-MODIFIED|X-EVOLUTION-ALARM-UID)(;[^:;\n]*)*:.*\r?\n?//gm;
if (/^BEGIN:VEVENT/m ) {
#several properties are not preserved by funambol server in itodo2.0 format and
s/^(RECURRENCE-ID|ATTENDEE)(;[^:;\n]*)*:.*\r?\n?//gm;
#REPEAT:0 is added by funambol server so ignore it
s/^(REPEAT:0).*\r?\n?//gm;
#CN parameter is lost by funambol server
s/^ORGANIZER([^:\n]*);CN=([^:\n]*)(;[^:\n])*:(.*\r?\n?)/ORGANIZER$1$3:$4/mg;
}
if (/^BEGIN:VTODO/m ) {
#several properties are not preserved by funambol server in itodo2.0 format and
s/^(STATUS|URL)(;[^:;\n]*)*:.*\r?\n?//gm;
#some new properties are added by funambol server
s/^(CLASS:PUBLIC|PERCENT-COMPLETE:0).*\r?\n?//gm;
}
}
2009-12-11 10:52:22 +01:00
if($nokia_7210c) {
if (/BEGIN:VCARD/m) {
#ignore PREF, as it will added by default
s/^TEL([^:\n]*);TYPE=PREF/TEL$1/mg;
#remove non-digit prefix in TEL
s/^TEL([^:\n]*):(\D*)/TEL$1:/mg;
#properties N mismatch, sometimes lost part of components
s/^(N|X-EVOLUTION-FILE-AS):.*\r?\n?/$1:[...]\n/gm;
#strip spaces in 'NOTE'
while (s/^(NOTE|DESCRIPTION):(\S+)[\t ]+(\S+)/$1:$2$3/mg) {}
#preserve 80 chars in NOTE
s/^NOTE:(.{70}).*\r?\n?/NOTE:$1\n/mg;
#preserve one ADDR
# ignore the PHOTO encoding data, sometimes it add a default photo
s/^PHOTO(.*?): .*\n//mg;
#s/^(ADR)([^:;\n]*)(;TYPE=[^:\n]*)?:.*\r?\n?/$1:$4\n/mg;
#lost properties
s/^(NICKNAME|CATEGORIES|CALURI|FBURL|ROLE|X-AIM|X-ANNIVERSARY|X-ASSISTANT|X-EVOLUTION-BLOG-URL|X-EVOLUTION-VIDEO-URL|X-GROUPWISE|X-ICQ|X-MANAGER|X-SPOUSE|X-MOZILLA-HTML|X-YAHOO)(;[^:;\n]*)*:.*\r?\n?//gm;
}
if (/^BEGIN:VEVENT/m ) {
#The properties phones add by default
s/^(PRIORITY|CATEGORIES)(;[^:;\n]*)*:.*\r?\n?//gm;
#strip spaces in 'DESCRIPTION'
while (s/^DESCRIPTION:(\S+)[\t ]+(\S+)/DESCRIPTION:$1$2/mg) {}
}
if (/^BEGIN:VTODO/m) {
#mismatch properties
s/^(PRIORITY)(;[^:;\n]*)*:.*\r?\n?/$1:[...]\n/gm;
#lost properties
s/^(STATUS|DTSTART|CATEGORIES)(;[^:;\n]*)*:.*\r?\n?//gm;
}
#Testing with phones using vcalendar, do not support UID
s/^(UID|CLASS|SEQUENCE|TRANSP)(;[^:;\n]*)*:.*\r?\n?//gm;
}
if ($ovi) {
if (/^BEGIN:VCARD/m) {
#lost properties
s/^(X-AIM|CALURI|URL|FBURL|PHOTO|EMAIL)(;[^:;\n]*)*:.*\r?\n?//gm;
#FN value mismatch (reordring and adding , by the server)
s/^FN:.*\r?\n?/FN:[...]\n/gm;
#X-EVOLUTION-FILE-AS adding '\' by the server
while (s/^X-EVOLUTION-FILE-AS:(.*)\\(.*)/X-EVOLUTION-FILE-AS:$1$2/gm) {}
# does not preserve X-EVOLUTION-UI-SLOT=
s/^(\w+)([^:\n]*);X-EVOLUTION-UI-SLOT=\d+/$1$2/mg;
# does not preserve third ADR
s/^ADR:Test Box #3.*\n\r?//mg;
}
if (/^BEGIN:VEVENT/m) {
#Testing with vcalendar, do not support UID
s/^(UID|SEQUENCE|TRANSP)(;[^:;\n]*)*:.*\r?\n?//gm;
#Add PRORITY by default
s/^(PRIORITY)(;[^:;\n]*)*:.*\r?\n?//gm;
# VALARM not supported
s/^BEGIN:VALARM.*?END:VALARM\r?\n?//msg;
}
if (/^BEGIN:VTODO/m) {
#Testing with vcalendar, do not support UID
s/^(UID|SEQUENCE|PERCENT-COMPLETE)(;[^:;\n]*)*:.*\r?\n?//gm;
#Mismatch DTSTART, COMPLETED
s/^(DTSTART|COMPLETED)(;[^:;\n]*)*:.*\r?\n?/$1:[...]\n/gm;
}
}
2009-12-11 10:52:22 +01:00
if ($funambol || $egroupware || $nokia_7210c) {
# NOTE may be truncated due to length resistrictions
s/^(NOTE(;[^:;\n]*)*:.{0,160}).*(\r?\n?)/$1$3/gm;
}
if ($memotoo) {
if (/^BEGIN:VCARD/m ) {
s/^(FN|FBURL|GEO|CALURI|ROLE|X-MOZILLA-HTML|X-EVOLUTION-BLOG-URL|X-EVOLUTION-VIDEO-URL|X-GADUGADU|X-JABBER|X-MSN|X-SIP|X-SKYPE|X-GROUPWISE)(;[^:;\n]*)*:.*\r?\n?//gm;
# s/^(FN|FBURL|CALURI|CATEGORIES|ROLE|X-MOZILLA-HTML|X-EVOLUTION-FILE-AS|X-EVOLUTION-BLOG-URL|X-EVOLUTION-VIDEO-URL|X-GADUGADU|X-JABBER|X-MSN|X-SIP|X-SKYPE|X-GROUPWISE)(;[^:;\n]*)*:.*\r?\n?//gm;
# strip 'TYPE=HOME'
s/^URL([^\n:]*);TYPE=HOME/URL$1/mg;
s/^EMAIL([^\n:]*);TYPE=HOME/EMAIL$1/mg;
}
if (/^BEGIN:VEVENT/m ) {
s/^(UID|SEQUENCE|TRANSP|RECURRENCE-ID|X-EVOLUTION-ALARM-UID|ORGANIZER)(;[^:;\n]*)*:.*\r?\n?//gm;
# some parameters of 'ATTENDEE' will be lost by server
s/^ATTENDEE([^\n:]*);CUTYPE=([^\n;:]*)/ATTENDEE$1/mg;
s/^ATTENDEE([^\n:]*);LANGUAGE=([^\n;:]*)/ATTENDEE$1/mg;
s/^ATTENDEE([^\n:]*);ROLE=([^\n;:]*)/ATTENDEE$1/mg;
s/^ATTENDEE([^\n:]*);RSVP=([^\n;:]*)/ATTENDEE$1/mg;
s/^ATTENDEE([^\n:]*);CN=([^\n;:]*)/ATTENDEE$1/mg;
s/^ATTENDEE([^\n:]*);PARTSTAT=([^\n;:]*)/ATTENDEE$1/mg;
if (/^BEGIN:VALARM/m ) {
s/^(DESCRIPTION)(;[^:;\n]*)*:.*\r?\n?//mg;
}
}
if (/^BEGIN:VTODO/m ) {
s/^(UID|SEQUENCE|URL|CLASS|PRIORITY)(;[^:;\n]*)*:.*\r?\n?//gm;
s/^PERCENT-COMPLETE:0\r?\n?//gm;
}
}
if ($mobical) {
s/^(CALURI|CATEGORIES|FBURL|GEO|NICKNAME|X-MOZILLA-HTML|X-EVOLUTION-FILE-AS|X-ANNIVERSARY|X-ASSISTANT|X-EVOLUTION-BLOG-URL|X-EVOLUTION-VIDEO-URL|X-GROUPWISE|X-ICQ|X-GADUGADU|X-JABBER|X-MSN|X-SIP|X-SKYPE|X-MANAGER|X-SPOUSE|X-YAHOO|X-AIM)(;[^:;\n]*)*:.*\r?\n?//gm;
# some workrounds here for mobical's bug
s/^(FN|BDAY)(;[^:;\n]*)*:.*\r?\n?//gm;
if (/^BEGIN:VEVENT/m ) {
s/^(UID|SEQUENCE|CLASS|TRANSP|RECURRENCE-ID|ATTENDEE|ORGANIZER|AALARM|DALARM)(;[^:;\n]*)*:.*\r?\n?//gm;
}
if (/^BEGIN:VTODO/m ) {
s/^(UID|SEQUENCE|DTSTART|URL|PERCENT-COMPLETE|CLASS)(;[^:;\n]*)*:.*\r?\n?//gm;
s/^PRIORITY:0\r?\n?//gm;
}
}
if ($zyb) {
s/^(CALURI|CATEGORIES|FBURL|NICKNAME|X-MOZILLA-HTML|PHOTO|X-EVOLUTION-FILE-AS|X-ANNIVERSARY|X-ASSISTANT|X-EVOLUTION-BLOG-URL|X-EVOLUTION-VIDEO-URL|X-GROUPWISE|X-ICQ|X-MANAGER|X-SPOUSE|X-YAHOO|X-AIM)(;[^:;\n]*)*:.*\r?\n?//gm;
}
if ($exchange) {
# unsupported properties
s/^(SEQUENCE|X-EVOLUTION-ALARM-UID)(;[^:;\n]*)*:.*\r?\n?//gm;
# added properties which can be ignored (?)
s/^(X-MEEGO-ACTIVESYNCD-[a-zA-Z]*)(;[^:;\n]*)*:.*\r?\n?//gm;
# ORGANIZER added - remove and thus ignore if we have no ATTENDEEs
if (!/^ATTENDEE/m) {
s/^(ORGANIZER)(;[^:;\n]*)*:.*\r?\n?//gm;
}
# ignore added VALARM DESCRIPTION
s/^DESCRIPTION:Reminder\n//m;
}
if ($googleeas) {
# properties not supported by Google
s/^(X-EVOLUTION-FILE-AS|CATEGORIES)(;[^:;\n]*)*:.*\r?\n?//gm;
}
if ($googleeas || $exchange) {
# properties not supported by ActiveSync
s/^(FN)(;[^:;\n]*)*:.*\r?\n?//gm;
}
if ($googleeas || $exchange) {
# properties not supported by ActiveSync and/or activesyncd
s/^(GEO)(;[^:;\n]*)*:.*\r?\n?//gm;
}
if ($akonadi) {
# Akonadi adds empty GEO propery....
s/^(GEO)(;[^:;\n]*)*:0+\.0+;0+\.0+\r?\n?//gm;
# ... and rounds other values.
s/^(GEO(?:;[^:;\n]*)*):([-+]?\d+)\.\d+;([-+]?\d+)\.\d+/$1:$2;$3/gm;
# does not preserve X-EVOLUTION-UI-SLOT=
s/^(\w+)([^:\n]*);X-EVOLUTION-UI-SLOT=\d+/$1$2/mg;
}
if ($googleeas || $exchange) {
# temporarily ignore modified properties
s/^(BDAY|X-ANNIVERSARY)(;[^:;\n]*)*:.*\r?\n?//gm;
}
# treat X-MOZILLA-HTML=FALSE as if the property didn't exist
s/^X-MOZILLA-HTML:FALSE\r?\n?//gm;
my @formatted = ();
# Modify lines to cover not more than
# $width characters by folding lines (as done for the N or SUMMARY above),
# but also indent each inner BEGIN/END block by 2 spaces
# and finally sort the lines.
# We need to keep a stack of open blocks in @formatted:
# - BEGIN creates another open block
# - END closes it, sorts it, and adds as single string to the parent block
push @formatted, [];
foreach $_ (split /\n/, $_) {
if (/^BEGIN:/) {
# start a new block
push @formatted, [];
}
my $spaces = " " x ($#formatted - 1);
# Ignore group tags during folding, add back before indenting.
/^([^.:;]+\.)?(.*)/s;
my $tag = $1;
$_ = $2;
my $thiswidth = $width -1 - length($spaces);
$thiswidth = 1 if $thiswidth <= 0;
s/(.{$thiswidth})(?!$)/$1\n /g;
$_ = $tag . $_;
s/^(.*)$/$spaces$1/mg;
push @{$formatted[$#formatted]}, $_;
if (/^\s*END:/) {
my $block = pop @formatted;
my $begin = shift @{$block};
my $end = pop @{$block};
# Keep begin/end as first/last line,
# inbetween sort, but so that N or SUMMARY are
# at the top. This ensures that the order of items
# is the same, even if individual properties differ.
# Also put indented blocks at the end, not the top.
sub numspaces {
my $str = shift;
$str =~ /^(\s*)/;
return length($1);
}
# Sort lines without group tag before lines without group tag.
# When both lines have group tags, sort based on line without
# group tag, then regroup related items after sorting.
sub cmplines {
my $a = shift;
my $b = shift;
$a =~ s/^[^.:;]+\.//;
$b =~ s/^[^.:;]+\.//;
return $a cmp $b;
}
my @body;
my $isimportant;
foreach $_ (@{$block}) {
$isimportant = ($_ =~ /^\s*(N|SUMMARY):/);
/^(\s*)([^.:;]+\.)?(.*)/s;
push @body, [$isimportant, length($1), $2, $3];
}
my @sorted = sort( { ($a->[1] - $b->[1]) || # Compare indention, more indented last.
($b->[0] - $a->[0]) || # Compare importance, less important last.
$a->[3] cmp $b->[3] } # Compare property name, parameters and value without group tag.
@body );
# Combine lines with the same group tag.
my %tags;
my @tagged;
my $tag;
my $entry;
my $index;
foreach (@sorted) {
$tag = $_->[2];
# Has a line a group tag?
if ($tag) {
# Same as one found before?
$index = $tags{$tag};
if (defined($index)) {
# Append to previous instance of the tag, keeping tag indices the same.
push @{$tagged[$index]}, $_;
} else {
# Add at end, remember index for next line with the same tag.
push @tagged, $_;
$tags{$tag} = $#tagged;
}
} else {
push @tagged, $_;
}
}
# Convert back into individual, indented text lines.
my @expanded;
foreach (@tagged) {
if ($_->[2]) {
if ($#{$_} == 4) {
# Simplify IMPP + X-ABLabel:Other to just IMPP without group tag.
# For the sake of simplicity we only do that if the number of
# grouped properties is exactly two. Otherwise we would have
# to search in the list of extra properties.
if ($_->[3] =~ /^IMPP[;:]/ &&
$_->[4][3] =~ /^X-ABLabel:Other$/) {
splice(@{$_}, 4);
}
}
if ($#{$_} == 3) {
# If the last remaining property is X-ABLabel, then ignore it.
# We ignore empty properties, which can cause their labels to
# be left as redundant information (happens with Google CardDAV
# when sending an empty URL).
if ($_->[3] =~ "X-ABLabel:") {
next;
}
# Remove redundant group tag from other properties.
$_->[2] = "";
}
}
push @expanded, (" " x $_->[1]) . ($_->[2] ? "- " : "") . $_->[3];
if ($#{$_} > 3) {
foreach ($_->[4,-1]) {
push @expanded, (" " x $_->[1]) . " " . $_->[3];
}
}
}
# Create one BEGIN/END block.
$_ = join("\n",
$begin,
@expanded,
$end);
push @{$formatted[$#formatted]}, $_;
}
}
return ${$formatted[0]}[0];
}
# parameters: text, width to use for reformatted lines
# returns list of lines without line breaks
sub Normalize {
$_ = shift;
my $width = shift;
s/\r//g;
my @items = ();
# split into individual items
foreach $_ ( split( /(?:(?<=\nEND:VCARD)|(?<=\nEND:VCALENDAR))\n*/ ) ) {
if (/END:VEVENT\s+BEGIN:VEVENT/s) {
# remove multiple events from calendar item
s/(BEGIN:VEVENT.*END:VEVENT\n)//s;
my $events = $1;
my $calendar = $_;
my $event;
# inject every single one back into the calendar and process the result
foreach $event ( split ( /(?:(?<=\nEND:VEVENT))\n*/, $events ) ) {
$_ = $calendar;
s/\nEND:VCALENDAR/\n$event\nEND:VCALENDAR/;
push @items, NormalizeItem($width, $_);
}
} else {
# already a single item
push @items, NormalizeItem($width, $_);
}
}
return split( /\n/, join( "\n\n", sort @items ));
}
# number of columns available for output:
# try tput without printing the shells error if not found,
# default to 80
my $columns = `which tput >/dev/null 2>/dev/null && tput 2>/dev/null && tput cols`;
if ($? || !$columns) {
$columns = 80;
}
if($#ARGV > 1) {
# error
Usage();
exit 1;
} elsif($#ARGV == 1) {
# comparison
my ($file1, $file2) = ($ARGV[0], $ARGV[1]);
my $singlewidth = int(($columns - 3) / 2);
$columns = $singlewidth * 2 + 3;
my @normal1;
my @normal2;
if (-d $file1 && -d $file2) {
# Both "files" are really directories of individual files.
# Don't include files in the comparison which are known
# to be identical because the refer to the same inode.
# - build map from inode to filename(s) (each inode might be used more than once!)
my %files1;
my %files2;
my @content1;
my @content2;
my $inode;
my $fullname;
my $entry;
opendir(my $dh, $file1) || die "cannot read $file1: $!";
foreach $entry (grep { -f "$file1/$_" } readdir($dh)) {
$fullname = "$file1/$entry";
$inode = (stat($fullname))[1];
if (!$files1{$inode}) {
$files1{$inode} = [];
}
push(@{$files1{$inode}}, $entry);
}
closedir($dh);
# - remove common files, read others
opendir(my $dh, $file2) || die "cannot read $file2: $!";
foreach $entry (grep { -f "$file2/$_" } readdir($dh)) {
$fullname = "$file2/$entry";
$inode = (stat($fullname))[1];
if (@{$files1{$inode}}) {
# randomly match against the last file
pop @{$files1{$inode}};
} else {
open(IN, "<:utf8", "$fullname") || die "$fullname: $!";
push @content2, <IN>;
}
}
# - read remaining entries from first dir
foreach my $array (values %files1) {
foreach $entry (@{$array}) {
$fullname = "$file1/$entry";
open(IN, "<:utf8", "$fullname") || die "$fullname: $!";
push @content1, <IN>;
}
}
my $content1 = join("", @content1);
my $content2 = join("", @content2);
@normal1 = Normalize($content1, $singlewidth);
@normal2 = Normalize($content2, $singlewidth);
} else {
if (-d $file1) {
open(IN1, "-|:utf8", "find $file1 -type f -print0 | xargs -0 cat") || die "$file1: $!";
} else {
open(IN1, "<:utf8", $file1) || die "$file1: $!";
}
if (-d $file2) {
open(IN2, "-|:utf8", "find $file2 -type f -print0 | xargs -0 cat") || die "$file2: $!";
} else {
open(IN2, "<:utf8", $file2) || die "$file2: $!";
}
my $buf1 = join("", <IN1>);
my $buf2 = join("", <IN2>);
@normal1 = Normalize($buf1, $singlewidth);
@normal2 = Normalize($buf2, $singlewidth);
close(IN1);
close(IN2);
}
# Produce output where each line is marked as old (aka remove) with o,
# as new (aka added) with n, and as unchanged with u at the beginning.
# This allows simpler processing below.
my $res = 0;
if (0) {
# $_ = `diff "--old-line-format=o %L" "--new-line-format=n %L" "--unchanged-line-format=u %L" "$normal1" "$normal2"`;
# $res = $?;
} else {
# convert into same format as diff above - this allows reusing the
# existing output formatting code
my $diffs_ref = Algorithm::Diff::sdiff(\@normal1, \@normal2);
@_ = ();
my $hunk;
foreach $hunk ( @{$diffs_ref} ) {
my ($type, $left, $right) = @{$hunk};
if ($type eq "-") {
push @_, "o $left";
$res = 1;
} elsif ($type eq "+") {
push @_, "n $right";
$res = 1;
} elsif ($type eq "c") {
push @_, "o $left";
push @_, "n $right";
$res = 1;
} else {
push @_, "u $left";
}
}
$_ = join("\n", @_);
}
if ($res) {
print $ENV{CLIENT_TEST_HEADER};
printf "%*s | %s\n", $singlewidth,
($ENV{CLIENT_TEST_LEFT_NAME} || "before sync"),
($ENV{CLIENT_TEST_RIGHT_NAME} || "after sync");
printf "%*s <\n", $singlewidth,
($ENV{CLIENT_TEST_REMOVED} || "removed during sync");
printf "%*s > %s\n", $singlewidth, "",
($ENV{CLIENT_TEST_ADDED} || "added during sync");
print "-" x $columns, "\n";
# fix confusing output like:
# BEGIN:VCARD BEGIN:VCARD
# > N:new;entry
# > FN:new
# > END:VCARD
# >
# > BEGIN:VCARD
# and replace it with:
# > BEGIN:VCARD
# > N:new;entry
# > FN:new
# > END:VCARD
#
# BEGIN:VCARD BEGIN:VCARD
#
# With the o/n/u markup this presents itself as:
# u BEGIN:VCARD
# n N:new;entry
# n FN:new
# n END:VCARD
# n
# n BEGIN:VCARD
#
# The alternative case is also possible:
# o END:VCARD
# o
# o BEGIN:VCARD
# o N:old;entry
# u END:VCARD
# case one above
while( s/^u BEGIN:(VCARD|VCALENDAR)\n((?:^n .*\n)+?)^n BEGIN:/n BEGIN:$1\n$2u BEGIN:/m) {}
# same for the other direction
while( s/^u BEGIN:(VCARD|VCALENDAR)\n((?:^o .*\n)+?)^o BEGIN:/o BEGIN:$1\n$2u BEGIN:/m) {}
# case two
while( s/^o END:(VCARD|VCALENDAR)\n((?:^o .*\n)+?)^u END:/u END:$1\n$2o END:/m) {}
while( s/^n END:(VCARD|VCALENDAR)\n((?:^n .*\n)+?)^u END:/u END:$1\n$2n END:/m) {}
# split at end of each record
my $spaces = " " x $singlewidth;
foreach $_ (split /(?:(?<=. END:VCARD\n)|(?<=. END:VCALENDAR\n))(?:^. \n)*/m, $_) {
# ignore unchanged records
if (!length($_) || /^((u [^\n]*\n)*(u [^\n]*?))$/s) {
next;
}
# make all lines equally long in terms of printable characters
s/^(.*)$/$1 . (" " x ($singlewidth + 2 - length($1)))/gme;
# convert into side-by-side output
my @buffer = ();
foreach $_ (split /\n/, $_) {
if (/^u (.*)/) {
print join(" <\n", @buffer), " <\n" if $#buffer >= 0;
@buffer = ();
print $1, " ", $1, "\n";
} elsif (/^o (.*)/) {
# preserve in buffer for potential merging with "n "
push @buffer, $1;
} else {
/^n (.*)/;
# have line to be merged with?
if ($#buffer >= 0) {
print shift @buffer, " | ", $1, "\n";
} else {
print join(" <\n", @buffer), " <\n" if $#buffer >= 0;
print $spaces, " > ", $1, "\n";
}
}
}
print join(" <\n", @buffer), " <\n" if $#buffer >= 0;
@buffer = ();
print "-" x $columns, "\n";
}
}
# unlink($normal1);
# unlink($normal2);
exit($res ? ((defined $ENV{CLIENT_TEST_COMPARISON_FAILED}) ? int($ENV{CLIENT_TEST_COMPARISON_FAILED}) : 1) : 0);
} else {
# normalize
my $in;
if( $#ARGV >= 0 ) {
my $file1 = $ARGV[0];
if (-d $file1) {
open(IN, "-|:utf8", "find $file1 -type f -print0 | xargs -0 cat") || die "$file1: $!";
} else {
open(IN, "<:utf8", $file1) || die "$file1: $!";
}
$in = *IN{IO};
} else {
$in = *STDIN{IO};
}
my $buf = join("", <$in>);
print STDOUT join("\n", Normalize($buf, $columns)), "\n";
}