[tryton-debian-vcs] tryton-modules-party-vcarddav branch upstream created. f9abbbffb1ebb8ff1de225fbde0eaa60171bc48b

Mathias Behrle tryton-debian-vcs at alioth.debian.org
Wed Nov 27 17:04:19 UTC 2013


The following commit has been merged in the upstream branch:
https://alioth.debian.org/plugins/scmgit/cgi-bin/gitweb.cgi/?p=tryton/tryton-modules-party-vcarddav.git;a=commitdiff;h=f9abbbffb1ebb8ff1de225fbde0eaa60171bc48b
commit f9abbbffb1ebb8ff1de225fbde0eaa60171bc48b
Author: Mathias Behrle <mathiasb at m9s.biz>
Date:   Sun Nov 24 17:27:24 2013 +0100

    Adding upstream version 3.0.0.

diff --git a/CHANGELOG b/CHANGELOG
index 7f34659..7d5a2bd 100644
--- a/CHANGELOG
+++ b/CHANGELOG
@@ -1,3 +1,6 @@
+Version 3.0.0 - 2013-10-21
+* Bug fixes (see mercurial logs for details)
+
 Version 2.8.0 - 2013-04-22
 * Bug fixes (see mercurial logs for details)
 
diff --git a/INSTALL b/INSTALL
index 1a36e89..4ece669 100644
--- a/INSTALL
+++ b/INSTALL
@@ -9,6 +9,7 @@ Prerequisites
  * trytond_party (http://www.tryton.org/)
  * vobject 0.8.0 or later (http://vobject.skyhouseconsulting.com/)
  * pywebdav 0.9.8 or later (http://sourceforge.net/projects/pywebdav/)
+ * python-sql (http://code.google.com/p/python-sql/)
 
 Installation
 ------------
diff --git a/PKG-INFO b/PKG-INFO
index 0fc8a56..e02ee4f 100644
--- a/PKG-INFO
+++ b/PKG-INFO
@@ -1,12 +1,12 @@
 Metadata-Version: 1.1
 Name: trytond_party_vcarddav
-Version: 2.8.0
+Version: 3.0.0
 Summary: Tryton module for CardDAV
 Home-page: http://www.tryton.org/
 Author: Tryton
 Author-email: UNKNOWN
 License: GPL-3
-Download-URL: http://downloads.tryton.org/2.8/
+Download-URL: http://downloads.tryton.org/3.0/
 Description: trytond_party_vcarddav
         ======================
         
@@ -61,6 +61,7 @@ Classifier: Natural Language :: English
 Classifier: Natural Language :: French
 Classifier: Natural Language :: German
 Classifier: Natural Language :: Russian
+Classifier: Natural Language :: Slovenian
 Classifier: Natural Language :: Spanish
 Classifier: Operating System :: OS Independent
 Classifier: Programming Language :: Python :: 2.6
diff --git a/locale/es_ES.po b/locale/es_ES.po
index e1adc11..1ada398 100644
--- a/locale/es_ES.po
+++ b/locale/es_ES.po
@@ -5,7 +5,7 @@ msgstr "Content-Type: text/plain; charset=utf-8\n"
 msgctxt "error:ir.action.report:"
 msgid "Invalid email definition on report \"%s\"."
 msgstr ""
-"La definición de correo electrónico sobre el informe \"%s\", no es correcta."
+"La definición de correo electrónico sobre el informe \"%s\" no es correcta."
 
 msgctxt "error:party.party:"
 msgid "The UUID of the party must be unique."
diff --git a/locale/sl_SI.po b/locale/sl_SI.po
new file mode 100644
index 0000000..cb74981
--- /dev/null
+++ b/locale/sl_SI.po
@@ -0,0 +1,23 @@
+# 
+msgid ""
+msgstr "Content-Type: text/plain; charset=utf-8\n"
+
+msgctxt "error:party.party:"
+msgid "The UUID of the party must be unique."
+msgstr "UUID stranke mora biti edinstven."
+
+msgctxt "field:party.party,uuid:"
+msgid "UUID"
+msgstr "UUID"
+
+msgctxt "field:party.party,vcard:"
+msgid "VCard"
+msgstr "VCard"
+
+msgctxt "help:party.party,uuid:"
+msgid "Universally Unique Identifier"
+msgstr "Vsesplošno edinstven identifikator"
+
+msgctxt "model:ir.action,name:report_party_vcard"
+msgid "VCard"
+msgstr "VCard"
diff --git a/party.py b/party.py
index 63712b7..2a9e860 100644
--- a/party.py
+++ b/party.py
@@ -5,7 +5,7 @@ import vobject
 
 from trytond.model import fields
 from trytond.report import Report
-from trytond.backend import TableHandler, FIELDS
+from trytond import backend
 from trytond.transaction import Transaction
 from trytond.pool import Pool, PoolMeta
 
@@ -29,18 +29,21 @@ class Party:
 
     @classmethod
     def __register__(cls, module_name):
+        TableHandler = backend.get('TableHandler')
         cursor = Transaction().cursor
         table = TableHandler(cursor, cls, module_name)
+        sql_table = cls.__table__()
 
         if not table.column_exist('uuid'):
             table.add_raw_column('uuid',
-                FIELDS[cls.uuid._type].sql_type(cls.uuid),
-                FIELDS[cls.uuid._type].sql_format, None, None)
-            cursor.execute('SELECT id FROM "' + cls._table + '"')
+                cls.uuid.sql_type(),
+                cls.uuid.sql_format, None, None)
+            cursor.execute(*sql_table.select(sql_table.id))
             for id, in cursor.fetchall():
-                cursor.execute('UPDATE "' + cls._table + '" '
-                    'SET "uuid" = %s WHERE id = %s',
-                    (cls.default_uuid(), id))
+                cursor.execute(*sql_table.update(
+                        columns=[sql_table.uuid],
+                        values=[cls.default_uuid()],
+                        where=sql_table.id == id))
         super(Party, cls).__register__(module_name)
 
     @staticmethod
diff --git a/setup.py b/setup.py
index b853d85..f188b3b 100644
--- a/setup.py
+++ b/setup.py
@@ -21,7 +21,7 @@ major_version, minor_version, _ = info.get('version', '0.0.1').split('.', 2)
 major_version = int(major_version)
 minor_version = int(minor_version)
 
-requires = ['vobject >= 0.8.0', 'PyWebDAV >= 0.9.8']
+requires = ['vobject >= 0.8.0', 'PyWebDAV >= 0.9.8', 'python-sql']
 for dep in info.get('depends', []):
     if not re.match(r'(ir|res|webdav)(\W|$)', dep):
         requires.append('trytond_%s >= %s.%s, < %s.%s' %
@@ -63,6 +63,7 @@ setup(name='trytond_party_vcarddav',
         'Natural Language :: French',
         'Natural Language :: German',
         'Natural Language :: Russian',
+        'Natural Language :: Slovenian',
         'Natural Language :: Spanish',
         'Operating System :: OS Independent',
         'Programming Language :: Python :: 2.6',
diff --git a/tryton.cfg b/tryton.cfg
index 273c8c4..802b318 100644
--- a/tryton.cfg
+++ b/tryton.cfg
@@ -1,5 +1,5 @@
 [tryton]
-version=2.8.0
+version=3.0.0
 depends:
     ir
     party
diff --git a/trytond_party_vcarddav.egg-info/PKG-INFO b/trytond_party_vcarddav.egg-info/PKG-INFO
index 7789dea..b08822f 100644
--- a/trytond_party_vcarddav.egg-info/PKG-INFO
+++ b/trytond_party_vcarddav.egg-info/PKG-INFO
@@ -1,12 +1,12 @@
 Metadata-Version: 1.1
 Name: trytond-party-vcarddav
-Version: 2.8.0
+Version: 3.0.0
 Summary: Tryton module for CardDAV
 Home-page: http://www.tryton.org/
 Author: Tryton
 Author-email: UNKNOWN
 License: GPL-3
-Download-URL: http://downloads.tryton.org/2.8/
+Download-URL: http://downloads.tryton.org/3.0/
 Description: trytond_party_vcarddav
         ======================
         
@@ -61,6 +61,7 @@ Classifier: Natural Language :: English
 Classifier: Natural Language :: French
 Classifier: Natural Language :: German
 Classifier: Natural Language :: Russian
+Classifier: Natural Language :: Slovenian
 Classifier: Natural Language :: Spanish
 Classifier: Operating System :: OS Independent
 Classifier: Programming Language :: Python :: 2.6
diff --git a/trytond_party_vcarddav.egg-info/SOURCES.txt b/trytond_party_vcarddav.egg-info/SOURCES.txt
index 6f88556..61ff6d3 100644
--- a/trytond_party_vcarddav.egg-info/SOURCES.txt
+++ b/trytond_party_vcarddav.egg-info/SOURCES.txt
@@ -22,6 +22,7 @@ locale/es_ES.po
 locale/fr_FR.po
 locale/nl_NL.po
 locale/ru_RU.po
+locale/sl_SI.po
 trytond_party_vcarddav.egg-info/PKG-INFO
 trytond_party_vcarddav.egg-info/SOURCES.txt
 trytond_party_vcarddav.egg-info/dependency_links.txt
diff --git a/trytond_party_vcarddav.egg-info/requires.txt b/trytond_party_vcarddav.egg-info/requires.txt
index 583404b..085c9e7 100644
--- a/trytond_party_vcarddav.egg-info/requires.txt
+++ b/trytond_party_vcarddav.egg-info/requires.txt
@@ -1,4 +1,5 @@
 vobject >= 0.8.0
 PyWebDAV >= 0.9.8
-trytond_party >= 2.8, < 2.9
-trytond >= 2.8, < 2.9
\ No newline at end of file
+python-sql
+trytond_party >= 3.0, < 3.1
+trytond >= 3.0, < 3.1
\ No newline at end of file
diff --git a/webdav.py b/webdav.py
index 9749c7a..fdab179 100644
--- a/webdav.py
+++ b/webdav.py
@@ -2,6 +2,10 @@
 #this repository contains the full copyright notices and license terms.
 from pywebdav.lib.errors import DAV_NotFound, DAV_Forbidden
 from pywebdav.lib.constants import COLLECTION, OBJECT
+from sql.functions import Extract
+from sql.aggregate import Max
+from sql.conditionals import Coalesce
+
 from trytond.tools import reduce_ids
 from trytond.transaction import Transaction
 from trytond.cache import Cache
@@ -170,6 +174,7 @@ class Collection:
     @classmethod
     def get_creationdate(cls, uri, cache=None):
         Party = Pool().get('party.party')
+        party = Party.__table__()
         party_id = cls.vcard(uri)
 
         cursor = Transaction().cursor
@@ -189,11 +194,10 @@ class Collection:
             res = None
             for i in range(0, len(ids), cursor.IN_MAX):
                 sub_ids = ids[i:i + cursor.IN_MAX]
-                red_sql, red_ids = reduce_ids('id', sub_ids)
-                cursor.execute('SELECT id, '
-                        'EXTRACT(epoch FROM create_date) '
-                    'FROM "' + Party._table + '" '
-                    'WHERE ' + red_sql, red_ids)
+                red_sql = reduce_ids(party.id, sub_ids)
+                cursor.execute(*party.select(party.id,
+                        Extract('EPOCH', party.create_date),
+                        where=red_sql))
                 for party_id2, date in cursor.fetchall():
                     if party_id2 == party_id:
                         res = date
@@ -210,6 +214,9 @@ class Collection:
         Party = pool.get('party.party')
         Address = pool.get('party.address')
         ContactMechanism = pool.get('party.contact_mechanism')
+        party = Party.__table__()
+        address = Address.__table__()
+        contact_mechanism = ContactMechanism.__table__()
 
         cursor = Transaction().cursor
 
@@ -227,21 +234,21 @@ class Collection:
             res = None
             for i in range(0, len(ids), cursor.IN_MAX):
                 sub_ids = ids[i:i + cursor.IN_MAX]
-                red_sql, red_ids = reduce_ids('p.id', sub_ids)
-                cursor.execute('SELECT p.id, '
-                        'MAX(EXTRACT(epoch FROM '
-                            'COALESCE(p.write_date, p.create_date))), '
-                        'MAX(EXTRACT(epoch FROM '
-                            'COALESCE(a.write_date, a.create_date))), '
-                        'MAX(EXTRACT(epoch FROM '
-                            'COALESCE(c.write_date, c.create_date))) '
-                    'FROM "' + Party._table + '" p '
-                    'LEFT JOIN "' + Address._table + '" a '
-                        'ON p.id = a.party '
-                    'LEFT JOIN "' + ContactMechanism._table + '" c '
-                        'ON p.id = c.party '
-                    'WHERE ' + red_sql + ' '
-                    'GROUP BY p.id', red_ids)
+                red_sql = reduce_ids(party.id, sub_ids)
+                cursor.execute(*party.join(address, 'LEFT',
+                        condition=party.id == address.party
+                        ).join(contact_mechanism, 'LEFT',
+                        condition=party.id == contact_mechanism.party
+                        ).select(party.id,
+                        Max(Extract('EPOCH', Coalesce(party.write_date,
+                                    party.create_date))),
+                        Max(Extract('EPOCH', Coalesce(address.write_date,
+                                    address.create_date))),
+                        Max(Extract('EPOCH', Coalesce(
+                                    contact_mechanism.write_date,
+                                    contact_mechanism.create_date))),
+                        where=red_sql,
+                        group_by=party.id))
                 for party_id2, date_p, date_a, date_c in cursor.fetchall():
                     date = max(date_p, date_a, date_c)
                     if party_id2 == party_id:
commit e742443d35b6e8309f2f6714b5974d7dc60ed0e3
Author: Mathias Behrle <mathiasb at m9s.biz>
Date:   Thu May 2 00:36:42 2013 +0200

    Adding upstream version 2.8.0.

diff --git a/CHANGELOG b/CHANGELOG
index 2a0dd2b..7f34659 100644
--- a/CHANGELOG
+++ b/CHANGELOG
@@ -1,3 +1,6 @@
+Version 2.8.0 - 2013-04-22
+* Bug fixes (see mercurial logs for details)
+
 Version 2.6.0 - 2012-10-22
 * Bug fixes (see mercurial logs for details)
 
diff --git a/COPYRIGHT b/COPYRIGHT
index 8ad4b79..a005001 100644
--- a/COPYRIGHT
+++ b/COPYRIGHT
@@ -1,6 +1,6 @@
-Copyright (C) 2009-2012 Cédric Krier.
-Copyright (C) 2009-2012 Bertrand Chenal.
-Copyright (C) 2009-2012 B2CK SPRL.
+Copyright (C) 2009-2013 Cédric Krier.
+Copyright (C) 2009-2013 Bertrand Chenal.
+Copyright (C) 2009-2013 B2CK SPRL.
 
 This program is free software: you can redistribute it and/or modify
 it under the terms of the GNU General Public License as published by
diff --git a/PKG-INFO b/PKG-INFO
index 728f9a4..0fc8a56 100644
--- a/PKG-INFO
+++ b/PKG-INFO
@@ -1,12 +1,12 @@
-Metadata-Version: 1.0
+Metadata-Version: 1.1
 Name: trytond_party_vcarddav
-Version: 2.6.0
+Version: 2.8.0
 Summary: Tryton module for CardDAV
 Home-page: http://www.tryton.org/
 Author: Tryton
 Author-email: UNKNOWN
 License: GPL-3
-Download-URL: http://downloads.tryton.org/2.6/
+Download-URL: http://downloads.tryton.org/2.8/
 Description: trytond_party_vcarddav
         ======================
         
@@ -54,6 +54,7 @@ Classifier: Intended Audience :: Legal Industry
 Classifier: Intended Audience :: Manufacturing
 Classifier: License :: OSI Approved :: GNU General Public License (GPL)
 Classifier: Natural Language :: Bulgarian
+Classifier: Natural Language :: Catalan
 Classifier: Natural Language :: Czech
 Classifier: Natural Language :: Dutch
 Classifier: Natural Language :: English
diff --git a/locale/bg_BG.po b/locale/bg_BG.po
index fbdab9b..52a3454 100644
--- a/locale/bg_BG.po
+++ b/locale/bg_BG.po
@@ -3,8 +3,8 @@ msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
 msgctxt "error:party.party:"
-msgid "The UUID of the party must be unique!"
-msgstr "UUID на партньор трабва да е уникален!"
+msgid "The UUID of the party must be unique."
+msgstr ""
 
 msgctxt "field:party.party,uuid:"
 msgid "UUID"
diff --git a/locale/ca_ES.po b/locale/ca_ES.po
index 27962bb..b22de37 100644
--- a/locale/ca_ES.po
+++ b/locale/ca_ES.po
@@ -2,9 +2,22 @@
 msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
+msgctxt "error:ir.action.report:"
+msgid "Invalid email definition on report \"%s\"."
+msgstr ""
+"La definició del correu electrònic sobre l'informe \"%s\", no és correcta."
+
 msgctxt "error:party.party:"
-msgid "The UUID of the party must be unique!"
-msgstr "L'UUID del tercer ha de ser únic."
+msgid "The UUID of the party must be unique."
+msgstr "El UUID del tercer ha de ser únic."
+
+msgctxt "error:webdav.collection:"
+msgid ""
+"You can not create a collection named \"%(parent)s\" in collection "
+"\"%(child)s\" because there is already a file with that name."
+msgstr ""
+"No podeu crear un directori amb el nom \"%(parent)s\" dins del directori "
+"\"%(child)s\" perquè ja existeix un altre amb aquest nom."
 
 msgctxt "field:party.party,uuid:"
 msgid "UUID"
diff --git a/locale/cs_CZ.po b/locale/cs_CZ.po
index 4cac0ef..de94f2b 100644
--- a/locale/cs_CZ.po
+++ b/locale/cs_CZ.po
@@ -3,7 +3,7 @@ msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
 msgctxt "error:party.party:"
-msgid "The UUID of the party must be unique!"
+msgid "The UUID of the party must be unique."
 msgstr ""
 
 msgctxt "field:party.party,uuid:"
diff --git a/locale/de_DE.po b/locale/de_DE.po
index 69e8741..7d81b41 100644
--- a/locale/de_DE.po
+++ b/locale/de_DE.po
@@ -3,8 +3,8 @@ msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
 msgctxt "error:party.party:"
-msgid "The UUID of the party must be unique!"
-msgstr "Die UUID für eine Partei kann nicht mehrfach vergeben werden!"
+msgid "The UUID of the party must be unique."
+msgstr "Die UUID einer Partei kann nur einmal vergeben werden."
 
 msgctxt "field:party.party,uuid:"
 msgid "UUID"
diff --git a/locale/es_AR.po b/locale/es_AR.po
index a38c31f..24aab1a 100644
--- a/locale/es_AR.po
+++ b/locale/es_AR.po
@@ -3,8 +3,8 @@ msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
 msgctxt "error:party.party:"
-msgid "The UUID of the party must be unique!"
-msgstr "¡El UUID de la entidad debe ser único!"
+msgid "The UUID of the party must be unique."
+msgstr "El UUID de la entidad debe ser único."
 
 msgctxt "field:party.party,uuid:"
 msgid "UUID"
diff --git a/locale/es_CO.po b/locale/es_CO.po
index 02242b0..0ff1a0f 100644
--- a/locale/es_CO.po
+++ b/locale/es_CO.po
@@ -3,8 +3,8 @@ msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
 msgctxt "error:party.party:"
-msgid "The UUID of the party must be unique!"
-msgstr "¡La UUID del tercero debe ser único!"
+msgid "The UUID of the party must be unique."
+msgstr ""
 
 msgctxt "field:party.party,uuid:"
 msgid "UUID"
diff --git a/locale/es_ES.po b/locale/es_ES.po
index 7579bc1..e1adc11 100644
--- a/locale/es_ES.po
+++ b/locale/es_ES.po
@@ -2,10 +2,23 @@
 msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
+msgctxt "error:ir.action.report:"
+msgid "Invalid email definition on report \"%s\"."
+msgstr ""
+"La definición de correo electrónico sobre el informe \"%s\", no es correcta."
+
 msgctxt "error:party.party:"
-msgid "The UUID of the party must be unique!"
+msgid "The UUID of the party must be unique."
 msgstr "El UUID del tercero debe ser único."
 
+msgctxt "error:webdav.collection:"
+msgid ""
+"You can not create a collection named \"%(parent)s\" in collection "
+"\"%(child)s\" because there is already a file with that name."
+msgstr ""
+"No puede crear un directorio con el nombre \"%(parent)s\" en el directorio "
+"\"%(child)s\" porque ya existe otro con ese nombre."
+
 msgctxt "field:party.party,uuid:"
 msgid "UUID"
 msgstr "UUID"
diff --git a/locale/fr_FR.po b/locale/fr_FR.po
index 3617a9d..b97725a 100644
--- a/locale/fr_FR.po
+++ b/locale/fr_FR.po
@@ -3,12 +3,8 @@ msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
 msgctxt "error:party.party:"
-msgid "The UUID of the party must be unique!"
-msgstr "L'UUID du tiers doit être unique !"
-
-msgctxt "error:party.party:"
-msgid "The UUID of the party must be unique!"
-msgstr "L'UUID du tiers doit être unique !"
+msgid "The UUID of the party must be unique."
+msgstr "L'UUID du tiers doit être unique."
 
 msgctxt "field:party.party,uuid:"
 msgid "UUID"
diff --git a/locale/nl_NL.po b/locale/nl_NL.po
index 4cac0ef..de94f2b 100644
--- a/locale/nl_NL.po
+++ b/locale/nl_NL.po
@@ -3,7 +3,7 @@ msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
 msgctxt "error:party.party:"
-msgid "The UUID of the party must be unique!"
+msgid "The UUID of the party must be unique."
 msgstr ""
 
 msgctxt "field:party.party,uuid:"
diff --git a/locale/ru_RU.po b/locale/ru_RU.po
index abbbe75..14b83db 100644
--- a/locale/ru_RU.po
+++ b/locale/ru_RU.po
@@ -3,22 +3,21 @@ msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
 msgctxt "error:party.party:"
-msgid "The UUID of the party must be unique!"
-msgstr ""
+msgid "The UUID of the party must be unique."
+msgstr "UUID контрагента должен быть уникальным."
 
 msgctxt "field:party.party,uuid:"
 msgid "UUID"
-msgstr ""
+msgstr "UUID"
 
 msgctxt "field:party.party,vcard:"
 msgid "VCard"
-msgstr ""
+msgstr "VCard"
 
-#, fuzzy
 msgctxt "help:party.party,uuid:"
 msgid "Universally Unique Identifier"
-msgstr "Универсальный уникальный идентификатор"
+msgstr "Универсальный уникальный идентификатор (UUID)"
 
 msgctxt "model:ir.action,name:report_party_vcard"
 msgid "VCard"
-msgstr ""
+msgstr "VCard"
diff --git a/party.py b/party.py
index 94aac3d..63712b7 100644
--- a/party.py
+++ b/party.py
@@ -24,7 +24,7 @@ class Party:
         super(Party, cls).__setup__()
         cls._sql_constraints += [
             ('uuid_uniq', 'UNIQUE(uuid)',
-                'The UUID of the party must be unique!'),
+                'The UUID of the party must be unique.'),
             ]
 
     @classmethod
@@ -48,13 +48,13 @@ class Party:
         return str(uuid.uuid4())
 
     @classmethod
-    def create(cls, vals):
+    def create(cls, vlist):
         Collection = Pool().get('webdav.collection')
 
-        party = super(Party, cls).create(vals)
+        parties = super(Party, cls).create(vlist)
         # Restart the cache for vcard
         Collection._vcard_cache.clear()
-        return party
+        return parties
 
     @classmethod
     def copy(cls, parties, default=None):
@@ -101,15 +101,22 @@ class Party:
             if hasattr(vcard, 'uid'):
                 res['uuid'] = vcard.uid.value
             res['addresses'] = []
+            to_create = []
             for adr in vcard.contents.get('adr', []):
                 vals = Address.vcard2values(adr)
-                res['addresses'].append(('create', vals))
+                to_create.append(vals)
+            if to_create:
+                res['addresses'].append(('create', to_create))
             res['contact_mechanisms'] = []
+            to_create = []
             for email in vcard.contents.get('email', []):
                 vals = {}
                 vals['type'] = 'email'
                 vals['value'] = email.value
-                res['contact_mechanisms'].append(('create', vals))
+                to_create.append(vals)
+            if to_create:
+                res['contact_mechanisms'].append(('create', to_create))
+            to_create = []
             for tel in vcard.contents.get('tel', []):
                 vals = {}
                 vals['type'] = 'phone'
@@ -117,7 +124,9 @@ class Party:
                         and 'cell' in tel.type_param.lower():
                     vals['type'] = 'mobile'
                 vals['value'] = tel.value
-                res['contact_mechanisms'].append(('create', vals))
+                to_create.append(vals)
+            if to_create:
+                res['contact_mechanisms'].append(('create', to_create))
         else:
             i = 0
             res['addresses'] = []
@@ -142,11 +151,14 @@ class Party:
                 new_addresses = vcard.contents.get('adr', [])[i:]
             except IndexError:
                 new_addresses = []
+            to_create = []
             for adr in new_addresses:
                 if not hasattr(adr, 'value'):
                     continue
                 vals = Address.vcard2values(adr)
-                res['addresses'].append(('create', vals))
+                to_create.append(vals)
+            if to_create:
+                res['addresses'].append(('create', to_create))
 
             i = 0
             res['contact_mechanisms'] = []
@@ -168,13 +180,16 @@ class Party:
                 new_emails = vcard.contents.get('email', [])[i:]
             except IndexError:
                 new_emails = []
+            to_create = []
             for email in new_emails:
                 if not hasattr(email, 'value'):
                     continue
                 vals = {}
                 vals['type'] = 'email'
                 vals['value'] = email.value
-                res['contact_mechanisms'].append(('create', vals))
+                to_create.append(vals)
+            if to_create:
+                res['contact_mechanisms'].append(('create', to_create))
 
             i = 0
             for cm in self.contact_mechanisms:
@@ -194,6 +209,7 @@ class Party:
                 new_tels = vcard.contents.get('tel', [])[i:]
             except IndexError:
                 new_tels = []
+            to_create = []
             for tel in new_tels:
                 if not hasattr(tel, 'value'):
                     continue
@@ -203,7 +219,9 @@ class Party:
                         and 'cell' in tel.type_param.lower():
                     vals['type'] = 'mobile'
                 vals['value'] = tel.value
-                res['contact_mechanisms'].append(('create', vals))
+                to_create.append(vals)
+            if to_create:
+                res['contact_mechanisms'].append(('create', to_create))
 
             if contact_mechanisms_todelete:
                 res['contact_mechanisms'].append(('delete',
@@ -271,7 +289,7 @@ class VCard(Report):
             raise Exception('Error', 'Report (%s) not find!' % cls.__name__)
         action_report = action_reports[0]
 
-        parties = Party(ids)
+        parties = Party.browse(ids)
 
         data = ''.join(cls.create_vcard(party).serialize()
             for party in parties)
@@ -309,7 +327,7 @@ class VCard(Report):
             if not hasattr(adr, 'value'):
                 adr.value = vobject.vcard.Address()
             adr.value.street = address.street and address.street + (
-                address.streetbis and  (" " + address.streetbis) or '') or ''
+                address.streetbis and (" " + address.streetbis) or '') or ''
             adr.value.city = address.city or ''
             if address.subdivision:
                 adr.value.region = address.subdivision.name or ''
diff --git a/setup.py b/setup.py
index 495c64c..b853d85 100644
--- a/setup.py
+++ b/setup.py
@@ -25,10 +25,10 @@ requires = ['vobject >= 0.8.0', 'PyWebDAV >= 0.9.8']
 for dep in info.get('depends', []):
     if not re.match(r'(ir|res|webdav)(\W|$)', dep):
         requires.append('trytond_%s >= %s.%s, < %s.%s' %
-                (dep, major_version, minor_version, major_version,
-                    minor_version + 1))
+            (dep, major_version, minor_version, major_version,
+                minor_version + 1))
 requires.append('trytond >= %s.%s, < %s.%s' %
-        (major_version, minor_version, major_version, minor_version + 1))
+    (major_version, minor_version, major_version, minor_version + 1))
 
 setup(name='trytond_party_vcarddav',
     version=info.get('version', '0.0.1'),
@@ -36,15 +36,15 @@ setup(name='trytond_party_vcarddav',
     long_description=read('README'),
     author='Tryton',
     url='http://www.tryton.org/',
-    download_url="http://downloads.tryton.org/" + \
-        info.get('version', '0.0.1').rsplit('.', 1)[0] + '/',
+    download_url=("http://downloads.tryton.org/" +
+        info.get('version', '0.0.1').rsplit('.', 1)[0] + '/'),
     package_dir={'trytond.modules.party_vcarddav': '.'},
     packages=[
         'trytond.modules.party_vcarddav',
         ],
     package_data={
-        'trytond.modules.party_vcarddav': info.get('xml', []) \
-            + ['tryton.cfg', 'locale/*.po'],
+        'trytond.modules.party_vcarddav': (info.get('xml', [])
+            + ['tryton.cfg', 'locale/*.po']),
         },
     classifiers=[
         'Development Status :: 5 - Production/Stable',
@@ -56,6 +56,7 @@ setup(name='trytond_party_vcarddav',
         'Intended Audience :: Manufacturing',
         'License :: OSI Approved :: GNU General Public License (GPL)',
         'Natural Language :: Bulgarian',
+        'Natural Language :: Catalan',
         'Natural Language :: Czech',
         'Natural Language :: Dutch',
         'Natural Language :: English',
diff --git a/tryton.cfg b/tryton.cfg
index e9fe934..273c8c4 100644
--- a/tryton.cfg
+++ b/tryton.cfg
@@ -1,5 +1,5 @@
 [tryton]
-version=2.6.0
+version=2.8.0
 depends:
     ir
     party
diff --git a/trytond_party_vcarddav.egg-info/PKG-INFO b/trytond_party_vcarddav.egg-info/PKG-INFO
index 4ec2583..7789dea 100644
--- a/trytond_party_vcarddav.egg-info/PKG-INFO
+++ b/trytond_party_vcarddav.egg-info/PKG-INFO
@@ -1,12 +1,12 @@
-Metadata-Version: 1.0
+Metadata-Version: 1.1
 Name: trytond-party-vcarddav
-Version: 2.6.0
+Version: 2.8.0
 Summary: Tryton module for CardDAV
 Home-page: http://www.tryton.org/
 Author: Tryton
 Author-email: UNKNOWN
 License: GPL-3
-Download-URL: http://downloads.tryton.org/2.6/
+Download-URL: http://downloads.tryton.org/2.8/
 Description: trytond_party_vcarddav
         ======================
         
@@ -54,6 +54,7 @@ Classifier: Intended Audience :: Legal Industry
 Classifier: Intended Audience :: Manufacturing
 Classifier: License :: OSI Approved :: GNU General Public License (GPL)
 Classifier: Natural Language :: Bulgarian
+Classifier: Natural Language :: Catalan
 Classifier: Natural Language :: Czech
 Classifier: Natural Language :: Dutch
 Classifier: Natural Language :: English
diff --git a/trytond_party_vcarddav.egg-info/requires.txt b/trytond_party_vcarddav.egg-info/requires.txt
index 83ba174..583404b 100644
--- a/trytond_party_vcarddav.egg-info/requires.txt
+++ b/trytond_party_vcarddav.egg-info/requires.txt
@@ -1,4 +1,4 @@
 vobject >= 0.8.0
 PyWebDAV >= 0.9.8
-trytond_party >= 2.6, < 2.7
-trytond >= 2.6, < 2.7
\ No newline at end of file
+trytond_party >= 2.8, < 2.9
+trytond >= 2.8, < 2.9
\ No newline at end of file
diff --git a/webdav.py b/webdav.py
index ee65adf..9749c7a 100644
--- a/webdav.py
+++ b/webdav.py
@@ -1,6 +1,7 @@
 #This file is part of Tryton.  The COPYRIGHT file at the top level of
 #this repository contains the full copyright notices and license terms.
 from pywebdav.lib.errors import DAV_NotFound, DAV_Forbidden
+from pywebdav.lib.constants import COLLECTION, OBJECT
 from trytond.tools import reduce_ids
 from trytond.transaction import Transaction
 from trytond.cache import Cache
@@ -153,7 +154,6 @@ class Collection:
 
     @classmethod
     def get_resourcetype(cls, uri, cache=None):
-        from DAV.constants import COLLECTION, OBJECT
         party_id = cls.vcard(uri)
         if party_id:
             return OBJECT
@@ -190,10 +190,10 @@ class Collection:
             for i in range(0, len(ids), cursor.IN_MAX):
                 sub_ids = ids[i:i + cursor.IN_MAX]
                 red_sql, red_ids = reduce_ids('id', sub_ids)
-                cursor.execute('SELECT id, ' \
-                            'EXTRACT(epoch FROM create_date) ' \
-                        'FROM "' + Party._table + '" ' \
-                        'WHERE ' + red_sql, red_ids)
+                cursor.execute('SELECT id, '
+                        'EXTRACT(epoch FROM create_date) '
+                    'FROM "' + Party._table + '" '
+                    'WHERE ' + red_sql, red_ids)
                 for party_id2, date in cursor.fetchall():
                     if party_id2 == party_id:
                         res = date
@@ -262,7 +262,7 @@ class Collection:
         if party_id:
             val = Vcard.execute([party_id],
                 {'id': party_id, 'ids': [party_id]})
-            return val[1]
+            return str(val[1])
         return super(Collection, cls).get_data(uri, cache=cache)
 
     @classmethod
@@ -285,7 +285,7 @@ class Collection:
             vcard = vobject.readOne(data)
             values = Party().vcard2values(vcard)
             try:
-                party_id = Party.create(values)
+                party_id, = Party.create([values])
             except Exception:
                 raise DAV_Forbidden
             party = Party(party_id)
commit 594b72ab9c4eac0c0fe713152ca28d98b1be286e
Author: Mathias Behrle <mathiasb at m9s.biz>
Date:   Tue Oct 23 19:53:37 2012 +0200

    Adding upstream version 2.6.0.

diff --git a/CHANGELOG b/CHANGELOG
index e6fec4c..2a0dd2b 100644
--- a/CHANGELOG
+++ b/CHANGELOG
@@ -1,4 +1,4 @@
-Version 2.4.1 - 2012-09-02
+Version 2.6.0 - 2012-10-22
 * Bug fixes (see mercurial logs for details)
 
 Version 2.4.0 - 2012-04-24
diff --git a/MANIFEST.in b/MANIFEST.in
index 32879df..97a9596 100644
--- a/MANIFEST.in
+++ b/MANIFEST.in
@@ -4,6 +4,7 @@ include TODO
 include COPYRIGHT
 include CHANGELOG
 include LICENSE
+include tryton.cfg
 include *.xml
 include *.odt
 include locale/*.po
diff --git a/PKG-INFO b/PKG-INFO
index 9dc963c..728f9a4 100644
--- a/PKG-INFO
+++ b/PKG-INFO
@@ -1,13 +1,49 @@
 Metadata-Version: 1.0
 Name: trytond_party_vcarddav
-Version: 2.4.1
-Summary: Add CardDAV on parties
+Version: 2.6.0
+Summary: Tryton module for CardDAV
 Home-page: http://www.tryton.org/
-Author: B2CK
-Author-email: info at b2ck.com
+Author: Tryton
+Author-email: UNKNOWN
 License: GPL-3
-Download-URL: http://downloads.tryton.org/2.4/
-Description: UNKNOWN
+Download-URL: http://downloads.tryton.org/2.6/
+Description: trytond_party_vcarddav
+        ======================
+        
+        The party_vcarddav module of the Tryton application platform.  See
+        __tryton__.py
+        
+        Installing
+        ----------
+        
+        See INSTALL
+        
+        Support
+        -------
+        
+        If you encounter any problems with Tryton, please don't hesitate to ask
+        questions on the Tryton bug tracker, mailing list, wiki or IRC channel:
+        
+          http://bugs.tryton.org/
+          http://groups.tryton.org/
+          http://wiki.tryton.org/
+          irc://irc.freenode.net/tryton
+        
+        License
+        -------
+        
+        See LICENSE
+        
+        Copyright
+        ---------
+        
+        See COPYRIGHT
+        
+        
+        For more information please visit the Tryton web site:
+        
+          http://www.tryton.org/
+        
 Platform: UNKNOWN
 Classifier: Development Status :: 5 - Production/Stable
 Classifier: Environment :: Plugins
diff --git a/__init__.py b/__init__.py
index 82686c6..ee29b01 100644
--- a/__init__.py
+++ b/__init__.py
@@ -1,6 +1,19 @@
 #This file is part of Tryton.  The COPYRIGHT file at the top level of
 #this repository contains the full copyright notices and license terms.
 
+from trytond.pool import Pool
 from . import carddav
 from .webdav import *
 from .party import *
+
+
+def register():
+    Pool.register(
+        Collection,
+        Party,
+        Address,
+        ActionReport,
+        module='party_vcarddav', type_='model')
+    Pool.register(
+        VCard,
+        module='party_vcarddav', type_='report')
diff --git a/__tryton__.py b/__tryton__.py
deleted file mode 100644
index 14f9973..0000000
--- a/__tryton__.py
+++ /dev/null
@@ -1,46 +0,0 @@
-# -*- coding: utf-8 -*-
-#This file is part of Tryton.  The COPYRIGHT file at the top level of
-#this repository contains the full copyright notices and license terms.
-{
-    'name': 'Party - vCardDAV',
-    'name_bg_BG': 'Фирма - vCardDAV',
-    'name_ca_ES': 'Tercers - vCardDAV',
-    'name_de_DE': 'Parteien vCardDAV',
-    'name_es_AR': 'Entidad - vCardDAV',
-    'name_es_CO': 'vCardDAV de Compañía',
-    'name_es_ES': 'Tercero - vCardDAV',
-    'name_fr_FR': 'Tiers - vCardDAV',
-    'version': '2.4.1',
-    'author': 'B2CK',
-    'email': 'info at b2ck.com',
-    'website': 'http://www.tryton.org/',
-    'description': 'Add CardDAV on parties',
-    'description_bg_BG': 'Добавя към партньор поддръжка на CardDAV',
-    'description_ca_ES': 'Afegeix suport per CardDAV als Tercers',
-    'description_de_DE': 'Ermöglicht CardDAV für Parteien',
-    'description_es_AR': 'Añade soporte de CardDAV para entidades',
-    'description_es_CO': 'Soporte de CardDAV para terceros',
-    'description_es_ES': 'Añade soporte de CardDAV para terceros',
-    'description_fr_FR': 'Ajoute le support CardDAV pour les tiers',
-    'depends': [
-        'ir',
-        'res',
-        'party',
-        'webdav',
-    ],
-    'xml': [
-        'party.xml',
-    ],
-    'translation': [
-        'locale/bg_BG.po',
-        'locale/ca_ES.po',
-        'locale/cs_CZ.po',
-        'locale/de_DE.po',
-        'locale/es_AR.po',
-        'locale/es_CO.po',
-        'locale/es_ES.po',
-        'locale/fr_FR.po',
-        'locale/nl_NL.po',
-        'locale/ru_RU.po',
-    ],
-}
diff --git a/carddav.py b/carddav.py
index f8c37dc..58b8b0d 100644
--- a/carddav.py
+++ b/carddav.py
@@ -38,16 +38,15 @@ def _get_carddav_address_data(self, uri):
         raise DAV_NotFound
     pool = Pool(Transaction().cursor.database_name)
     try:
-        collection_obj = pool.get('webdav.collection')
+        Collection = pool.get('webdav.collection')
     except KeyError:
         raise DAV_NotFound
     try:
-        res = collection_obj.get_address_data(dburi, cache=CACHE)
+        return Collection.get_address_data(dburi, cache=CACHE)
     except DAV_Error:
         raise
     except Exception:
         raise DAV_Error(500)
-    return res
 
 TrytonDAVInterface._get_carddav_address_data = _get_carddav_address_data
 TrytonDAVInterface._get_carddav_addressbook_data = _get_carddav_address_data
diff --git a/locale/ca_ES.po b/locale/ca_ES.po
index 59fa348..27962bb 100644
--- a/locale/ca_ES.po
+++ b/locale/ca_ES.po
@@ -4,7 +4,7 @@ msgstr "Content-Type: text/plain; charset=utf-8\n"
 
 msgctxt "error:party.party:"
 msgid "The UUID of the party must be unique!"
-msgstr "L'UUID del tercer ha de ser únic"
+msgstr "L'UUID del tercer ha de ser únic."
 
 msgctxt "field:party.party,uuid:"
 msgid "UUID"
@@ -16,7 +16,7 @@ msgstr "vCard"
 
 msgctxt "help:party.party,uuid:"
 msgid "Universally Unique Identifier"
-msgstr "Identificador universal únic"
+msgstr "Identificador universal únic."
 
 msgctxt "model:ir.action,name:report_party_vcard"
 msgid "VCard"
diff --git a/locale/es_AR.po b/locale/es_AR.po
index c6b26fd..a38c31f 100644
--- a/locale/es_AR.po
+++ b/locale/es_AR.po
@@ -16,7 +16,7 @@ msgstr "VCard"
 
 msgctxt "help:party.party,uuid:"
 msgid "Universally Unique Identifier"
-msgstr "Identificador universal único"
+msgstr "Identificador Universal Único"
 
 msgctxt "model:ir.action,name:report_party_vcard"
 msgid "VCard"
diff --git a/locale/es_CO.po b/locale/es_CO.po
index 6130afe..02242b0 100644
--- a/locale/es_CO.po
+++ b/locale/es_CO.po
@@ -4,7 +4,7 @@ msgstr "Content-Type: text/plain; charset=utf-8\n"
 
 msgctxt "error:party.party:"
 msgid "The UUID of the party must be unique!"
-msgstr "¡La UUID del tercero debe ser única!"
+msgstr "¡La UUID del tercero debe ser único!"
 
 msgctxt "field:party.party,uuid:"
 msgid "UUID"
diff --git a/party.py b/party.py
index 74d2a12..94aac3d 100644
--- a/party.py
+++ b/party.py
@@ -1,98 +1,95 @@
 #This file is part of Tryton.  The COPYRIGHT file at the top level of
 #this repository contains the full copyright notices and license terms.
-import copy
 import uuid
-from trytond.model import ModelSQL, ModelView, fields
+import vobject
+
+from trytond.model import fields
 from trytond.report import Report
 from trytond.backend import TableHandler, FIELDS
 from trytond.transaction import Transaction
-from trytond.pool import Pool
+from trytond.pool import Pool, PoolMeta
+
+__all__ = ['Party', 'Address', 'ActionReport', 'VCard']
+__metaclass__ = PoolMeta
 
 
-class Party(ModelSQL, ModelView):
-    _name = 'party.party'
+class Party:
+    __name__ = 'party.party'
     uuid = fields.Char('UUID', required=True,
             help='Universally Unique Identifier')
     vcard = fields.Binary('VCard')
 
-    def __init__(self):
-        super(Party, self).__init__()
-        self._sql_constraints += [
-                ('uuid_uniq', 'UNIQUE(uuid)',
-                    'The UUID of the party must be unique!'),
-        ]
+    @classmethod
+    def __setup__(cls):
+        super(Party, cls).__setup__()
+        cls._sql_constraints += [
+            ('uuid_uniq', 'UNIQUE(uuid)',
+                'The UUID of the party must be unique!'),
+            ]
 
-    def init(self, module_name):
+    @classmethod
+    def __register__(cls, module_name):
         cursor = Transaction().cursor
-        table = TableHandler(cursor, self, module_name)
+        table = TableHandler(cursor, cls, module_name)
 
         if not table.column_exist('uuid'):
             table.add_raw_column('uuid',
-                    FIELDS[self.uuid._type].sql_type(self.uuid),
-                    FIELDS[self.uuid._type].sql_format, None, None)
-            cursor.execute('SELECT id FROM "' + self._table + '"')
+                FIELDS[cls.uuid._type].sql_type(cls.uuid),
+                FIELDS[cls.uuid._type].sql_format, None, None)
+            cursor.execute('SELECT id FROM "' + cls._table + '"')
             for id, in cursor.fetchall():
-                cursor.execute('UPDATE "' + self._table + '" ' \
-                        'SET "uuid" = %s WHERE id = %s',
-                        (self.default_uuid(), id))
-        super(Party, self).init(module_name)
+                cursor.execute('UPDATE "' + cls._table + '" '
+                    'SET "uuid" = %s WHERE id = %s',
+                    (cls.default_uuid(), id))
+        super(Party, cls).__register__(module_name)
 
-    def default_uuid(self):
+    @staticmethod
+    def default_uuid():
         return str(uuid.uuid4())
 
-    def create(self, vals):
-        collection_obj = Pool().get('webdav.collection')
+    @classmethod
+    def create(cls, vals):
+        Collection = Pool().get('webdav.collection')
 
-        res = super(Party, self).create(vals)
+        party = super(Party, cls).create(vals)
         # Restart the cache for vcard
-        collection_obj.vcard.reset()
-        return res
-
-    def copy(self, ids, default=None):
-        int_id = isinstance(ids, (int, long))
-        if int_id:
-            ids = [ids]
+        Collection._vcard_cache.clear()
+        return party
 
+    @classmethod
+    def copy(cls, parties, default=None):
         if default is None:
             default = {}
-
-        new_ids = []
-        for party_id in ids:
+        new_parties = []
+        for party in parties:
             current_default = default.copy()
-            current_default['uuid'] = self.default_uuid()
-            new_id = super(Party, self).copy(party_id, default=current_default)
-            new_ids.append(new_id)
-
-        if int_id:
-            return new_ids[0]
-        return new_ids
+            current_default['uuid'] = cls.default_uuid()
+            new_party, = super(Party, cls).copy([party],
+                default=current_default)
+            new_parties.append(new_party)
+        return new_parties
 
-    def write(self, ids, vals):
-        collection_obj = Pool().get('webdav.collection')
+    @classmethod
+    def write(cls, parties, vals):
+        Collection = Pool().get('webdav.collection')
 
-        res = super(Party, self).write(ids, vals)
+        super(Party, cls).write(parties, vals)
         # Restart the cache for vcard
-        collection_obj.vcard.reset()
-        return res
+        Collection._vcard_cache.clear()
 
-    def delete(self, ids):
-        collection_obj = Pool().get('webdav.collection')
+    @classmethod
+    def delete(cls, parties):
+        Collection = Pool().get('webdav.collection')
 
-        res = super(Party, self).delete(ids)
+        super(Party, cls).delete(parties)
         # Restart the cache for vcard
-        collection_obj.vcard.reset()
-        return res
+        Collection._vcard_cache.clear()
 
-    def vcard2values(self, party_id, vcard):
+    def vcard2values(self, vcard):
         '''
         Convert vcard to values for create or write
-
-        :param party_id: the party id for write or None for create
-        :param vcard: a vcard instance of vobject
-        :return: a dictionary with values
         '''
-        import vobject
-        address_obj = Pool().get('party.address')
+        Address = Pool().get('party.address')
 
         res = {}
         res['name'] = vcard.fn.value
@@ -100,12 +97,12 @@ class Party(ModelSQL, ModelView):
             vcard.add('n')
             vcard.n.value = vobject.vcard.Name(vcard.fn.value)
         res['vcard'] = vcard.serialize()
-        if not party_id:
+        if not self.id:
             if hasattr(vcard, 'uid'):
                 res['uuid'] = vcard.uid.value
             res['addresses'] = []
             for adr in vcard.contents.get('adr', []):
-                vals = address_obj.vcard2values(adr)
+                vals = Address.vcard2values(adr)
                 res['addresses'].append(('create', vals))
             res['contact_mechanisms'] = []
             for email in vcard.contents.get('email', []):
@@ -122,11 +119,10 @@ class Party(ModelSQL, ModelView):
                 vals['value'] = tel.value
                 res['contact_mechanisms'].append(('create', vals))
         else:
-            party = self.browse(party_id)
             i = 0
             res['addresses'] = []
             addresses_todelete = []
-            for address in party.addresses:
+            for address in self.addresses:
                 try:
                     adr = vcard.contents.get('adr', [])[i]
                 except IndexError:
@@ -137,8 +133,8 @@ class Party(ModelSQL, ModelView):
                     addresses_todelete.append(address.id)
                     i += 1
                     continue
-                vals = address_obj.vcard2values(adr)
-                res['addresses'].append(('write', address.id, vals))
+                vals = Address.vcard2values(adr)
+                res['addresses'].append(('write', [address.id], vals))
                 i += 1
             if addresses_todelete:
                 res['addresses'].append(('delete', addresses_todelete))
@@ -149,13 +145,13 @@ class Party(ModelSQL, ModelView):
             for adr in new_addresses:
                 if not hasattr(adr, 'value'):
                     continue
-                vals = address_obj.vcard2values(adr)
+                vals = Address.vcard2values(adr)
                 res['addresses'].append(('create', vals))
 
             i = 0
             res['contact_mechanisms'] = []
             contact_mechanisms_todelete = []
-            for cm in party.contact_mechanisms:
+            for cm in self.contact_mechanisms:
                 if cm.type != 'email':
                     continue
                 try:
@@ -181,7 +177,7 @@ class Party(ModelSQL, ModelView):
                 res['contact_mechanisms'].append(('create', vals))
 
             i = 0
-            for cm in party.contact_mechanisms:
+            for cm in self.contact_mechanisms:
                 if cm.type not in ('phone', 'mobile'):
                     continue
                 try:
@@ -214,89 +210,79 @@ class Party(ModelSQL, ModelView):
                     contact_mechanisms_todelete))
         return res
 
-Party()
-
 
-class Address(ModelSQL, ModelView):
-    _name = 'party.address'
+class Address:
+    __name__ = 'party.address'
 
     def vcard2values(self, adr):
         '''
         Convert adr from vcard to values for create or write
-
-        :param adr: a adr from vcard instance of vobject
-        :return: a dictionary with values
         '''
-        country_obj = Pool().get('country.country')
-        subdivision_obj = Pool().get('country.subdivision')
+        pool = Pool()
+        Country = pool.get('country.country')
+        Subdivision = pool.get('country.subdivision')
 
         vals = {}
         vals['street'] = adr.value.street or ''
         vals['city'] = adr.value.city or ''
         vals['zip'] = adr.value.code or ''
         if adr.value.country:
-            country_ids = country_obj.search([
-                ('rec_name', '=', adr.value.country),
-                ], limit=1)
-            if country_ids:
-                vals['country'] = country_ids[0]
+            countries = Country.search([
+                    ('rec_name', '=', adr.value.country),
+                    ], limit=1)
+            if countries:
+                country, = countries
+                vals['country'] = country.id
                 if adr.value.region:
-                    subdivision_ids = subdivision_obj.search([
+                    subdivisions = Subdivision.search([
                             ('rec_name', '=', adr.value.region),
-                            ('country', '=', country_ids[0]),
+                            ('country', '=', country.id),
                             ], limit=1)
-                    if subdivision_ids:
-                        vals['subdivision'] = subdivision_ids[0]
+                    if subdivisions:
+                        subdivision, = subdivisions
+                        vals['subdivision'] = subdivision.id
         return vals
 
-Address()
-
 
-class ActionReport(ModelSQL, ModelView):
-    _name = 'ir.action.report'
+class ActionReport:
+    __name__ = 'ir.action.report'
 
-    def __init__(self):
-        super(ActionReport, self).__init__()
+    @classmethod
+    def __setup__(cls):
+        super(ActionReport, cls).__setup__()
         new_ext = ('vcf', 'VCard file')
-        if new_ext not in self.extension.selection:
-            self.extension = copy.copy(self.extension)
-            self.extension.selection = copy.copy(self.extension.selection)
-            self.extension.selection.append(new_ext)
-            self._reset_columns()
-
-ActionReport()
+        if new_ext not in cls.extension.selection:
+            cls.extension.selection.append(new_ext)
 
 
 class VCard(Report):
-    _name = 'party_vcarddav.party.vcard'
+    __name__ = 'party_vcarddav.party.vcard'
 
-    def execute(self, ids, datas):
-        party_obj = Pool().get('party.party')
-        action_report_obj = Pool().get('ir.action.report')
+    @classmethod
+    def execute(cls, ids, data):
+        pool = Pool()
+        Party = pool.get('party.party')
+        ActionReport = pool.get('ir.action.report')
 
-        action_report_ids = action_report_obj.search([
-            ('report_name', '=', self._name)
-            ])
-        if not action_report_ids:
-            raise Exception('Error', 'Report (%s) not find!' % self._name)
-        action_report = action_report_obj.browse(action_report_ids[0])
+        action_reports = ActionReport.search([
+                ('report_name', '=', cls.__name__)
+                ])
+        if not action_reports:
+            raise Exception('Error', 'Report (%s) not find!' % cls.__name__)
+        action_report = action_reports[0]
 
-        parties = party_obj.browse(ids)
+        parties = Party(ids)
 
-        data = ''.join(self.create_vcard(party).serialize()
+        data = ''.join(cls.create_vcard(party).serialize()
             for party in parties)
 
         return ('vcf', buffer(data), False, action_report.name)
 
-    def create_vcard(self, party):
+    @classmethod
+    def create_vcard(cls, party):
         '''
         Return a vcard instance of vobject for the party
-
-        :param party: a BrowseRecord of party.party
-        :return: a vcard instance of vobject
         '''
-        import vobject
-
         if party.vcard:
             vcard = vobject.readOne(str(party.vcard))
         else:
@@ -387,5 +373,3 @@ class VCard(Report):
             vcard.contents['tel'].remove(tel)
 
         return vcard
-
-VCard()
diff --git a/setup.py b/setup.py
index a321362..495c64c 100644
--- a/setup.py
+++ b/setup.py
@@ -4,8 +4,19 @@
 
 from setuptools import setup
 import re
+import os
+import ConfigParser
 
-info = eval(open('__tryton__.py').read())
+
+def read(fname):
+    return open(os.path.join(os.path.dirname(__file__), fname)).read()
+
+config = ConfigParser.ConfigParser()
+config.readfp(open('tryton.cfg'))
+info = dict(config.items('tryton'))
+for key in ('depends', 'extras_depend', 'xml'):
+    if key in info:
+        info[key] = info[key].strip().splitlines()
 major_version, minor_version, _ = info.get('version', '0.0.1').split('.', 2)
 major_version = int(major_version)
 minor_version = int(minor_version)
@@ -21,20 +32,20 @@ requires.append('trytond >= %s.%s, < %s.%s' %
 
 setup(name='trytond_party_vcarddav',
     version=info.get('version', '0.0.1'),
-    description=info.get('description', ''),
-    author=info.get('author', ''),
-    author_email=info.get('email', ''),
-    url=info.get('website', ''),
+    description='Tryton module for CardDAV',
+    long_description=read('README'),
+    author='Tryton',
+    url='http://www.tryton.org/',
     download_url="http://downloads.tryton.org/" + \
-            info.get('version', '0.0.1').rsplit('.', 1)[0] + '/',
+        info.get('version', '0.0.1').rsplit('.', 1)[0] + '/',
     package_dir={'trytond.modules.party_vcarddav': '.'},
     packages=[
         'trytond.modules.party_vcarddav',
-    ],
+        ],
     package_data={
         'trytond.modules.party_vcarddav': info.get('xml', []) \
-                + info.get('translation', []),
-    },
+            + ['tryton.cfg', 'locale/*.po'],
+        },
     classifiers=[
         'Development Status :: 5 - Production/Stable',
         'Environment :: Plugins',
@@ -56,7 +67,7 @@ setup(name='trytond_party_vcarddav',
         'Programming Language :: Python :: 2.6',
         'Programming Language :: Python :: 2.7',
         'Topic :: Office/Business',
-    ],
+        ],
     license='GPL-3',
     install_requires=requires,
     zip_safe=False,
@@ -66,4 +77,4 @@ setup(name='trytond_party_vcarddav',
     """,
     test_suite='tests',
     test_loader='trytond.test_loader:Loader',
-)
+    )
diff --git a/tryton.cfg b/tryton.cfg
new file mode 100644
index 0000000..e9fe934
--- /dev/null
+++ b/tryton.cfg
@@ -0,0 +1,9 @@
+[tryton]
+version=2.6.0
+depends:
+    ir
+    party
+    res
+    webdav
+xml:
+    party.xml
diff --git a/trytond_party_vcarddav.egg-info/PKG-INFO b/trytond_party_vcarddav.egg-info/PKG-INFO
index 9e669c8..4ec2583 100644
--- a/trytond_party_vcarddav.egg-info/PKG-INFO
+++ b/trytond_party_vcarddav.egg-info/PKG-INFO
@@ -1,13 +1,49 @@
 Metadata-Version: 1.0
 Name: trytond-party-vcarddav
-Version: 2.4.1
-Summary: Add CardDAV on parties
+Version: 2.6.0
+Summary: Tryton module for CardDAV
 Home-page: http://www.tryton.org/
-Author: B2CK
-Author-email: info at b2ck.com
+Author: Tryton
+Author-email: UNKNOWN
 License: GPL-3
-Download-URL: http://downloads.tryton.org/2.4/
-Description: UNKNOWN
+Download-URL: http://downloads.tryton.org/2.6/
+Description: trytond_party_vcarddav
+        ======================
+        
+        The party_vcarddav module of the Tryton application platform.  See
+        __tryton__.py
+        
+        Installing
+        ----------
+        
+        See INSTALL
+        
+        Support
+        -------
+        
+        If you encounter any problems with Tryton, please don't hesitate to ask
+        questions on the Tryton bug tracker, mailing list, wiki or IRC channel:
+        
+          http://bugs.tryton.org/
+          http://groups.tryton.org/
+          http://wiki.tryton.org/
+          irc://irc.freenode.net/tryton
+        
+        License
+        -------
+        
+        See LICENSE
+        
+        Copyright
+        ---------
+        
+        See COPYRIGHT
+        
+        
+        For more information please visit the Tryton web site:
+        
+          http://www.tryton.org/
+        
 Platform: UNKNOWN
 Classifier: Development Status :: 5 - Production/Stable
 Classifier: Environment :: Plugins
diff --git a/trytond_party_vcarddav.egg-info/SOURCES.txt b/trytond_party_vcarddav.egg-info/SOURCES.txt
index 82f82db..6f88556 100644
--- a/trytond_party_vcarddav.egg-info/SOURCES.txt
+++ b/trytond_party_vcarddav.egg-info/SOURCES.txt
@@ -6,8 +6,8 @@ MANIFEST.in
 README
 party.xml
 setup.py
+tryton.cfg
 ./__init__.py
-./__tryton__.py
 ./carddav.py
 ./party.py
 ./webdav.py
diff --git a/trytond_party_vcarddav.egg-info/requires.txt b/trytond_party_vcarddav.egg-info/requires.txt
index 6e82de2..83ba174 100644
--- a/trytond_party_vcarddav.egg-info/requires.txt
+++ b/trytond_party_vcarddav.egg-info/requires.txt
@@ -1,4 +1,4 @@
 vobject >= 0.8.0
 PyWebDAV >= 0.9.8
-trytond_party >= 2.4, < 2.5
-trytond >= 2.4, < 2.5
\ No newline at end of file
+trytond_party >= 2.6, < 2.7
+trytond >= 2.6, < 2.7
\ No newline at end of file
diff --git a/webdav.py b/webdav.py
index 3c7f3de..ee65adf 100644
--- a/webdav.py
+++ b/webdav.py
@@ -1,53 +1,56 @@
 #This file is part of Tryton.  The COPYRIGHT file at the top level of
 #this repository contains the full copyright notices and license terms.
 from pywebdav.lib.errors import DAV_NotFound, DAV_Forbidden
-from trytond.model import ModelView, ModelSQL
 from trytond.tools import reduce_ids
 from trytond.transaction import Transaction
 from trytond.cache import Cache
-from trytond.pool import Pool
+from trytond.pool import Pool, PoolMeta
 
+__all__ = ['Collection']
+__metaclass__ = PoolMeta
 
-CARDDAV_NS = 'urn:ietf:params:xml:ns:carddav'
 
+CARDDAV_NS = 'urn:ietf:params:xml:ns:carddav'
 
-class Collection(ModelSQL, ModelView):
 
-    _name = "webdav.collection"
+class Collection:
+    __name__ = "webdav.collection"
+    _vcard_cache = Cache('webdav.collection.vcard', context=False)
 
-    @Cache('webdav_collection.vcard')
-    def vcard(self, uri):
+    @classmethod
+    def vcard(cls, uri):
         '''
         Return party ids of the vcard in uri or False
-
-        :param uri: the uri
-        :return: party id
-            or None if there is no party
-            or False if not in Contacts
         '''
-        party_obj = Pool().get('party.party')
+        Party = Pool().get('party.party')
+        party_id = cls._vcard_cache.get(uri, -1)
+        if party_id != -1:
+            return party_id
 
         if uri and uri.startswith('Contacts/'):
             uuid = uri[9:-4]
-            party_ids = party_obj.search([
-                ('uuid', '=', uuid),
-                ], limit=1)
-            if party_ids:
-                return party_ids[0]
-            return None
-        if uri == 'Contacts':
-            return None
-        return False
-
-    def _carddav_filter_domain(self, filter):
+            parties = Party.search([
+                    ('uuid', '=', uuid),
+                    ], limit=1)
+            if parties:
+                party_id = parties[0].id
+            else:
+                party_id = None
+        elif uri == 'Contacts':
+            party_id = None
+        else:
+            party_id = False
+        cls._vcard_cache.set(uri, party_id)
+        return party_id
+
+    @classmethod
+    def _carddav_filter_domain(cls, filter):
         '''
         Return a domain for the carddav filter
-
-        :param filter: the DOM Element of filter
-        :return: a list for domain
         '''
-        address_obj = Pool().get('party.address')
-        contact_mechanism_obj = Pool().get('party.contact_mechanism')
+        pool = Pool()
+        Address = pool.get('party.address')
+        ContactMechanism = pool.get('party.contact_mechanism')
 
         res = []
         if not filter:
@@ -110,60 +113,64 @@ class Collection(ModelSQL, ModelView):
                             res2.append((field, 'not ilike', value))
                     if name == 'adr':
                         domain = res2
-                        address_ids = address_obj.search(domain)
-                        res = [('addresses', 'in', address_ids)]
+                        addresses = Address.search(domain)
+                        res = [('addresses', 'in', [a.id for a in addresses])]
                     elif name in ('mail', 'tel'):
                         if name == 'mail':
                             type = ['email']
                         else:
                             type = ['phone', 'mobile']
                         domain = [('type', 'in', type), res2]
-                        contact_mechanism_ids = contact_mechanism_obj.search(
-                                domain)
+                        contact_mechanisms = ContactMechanism.search(
+                            domain)
                         res2 = [
-                            ('contact_mechanisms', 'in', contact_mechanism_ids)
+                            ('contact_mechanisms', 'in',
+                                [c.id for c in contact_mechanisms])
                             ]
                     res.append(res2)
         return res
 
-    def get_childs(self, uri, filter=None, cache=None):
-        party_obj = Pool().get('party.party')
+    @classmethod
+    def get_childs(cls, uri, filter=None, cache=None):
+        Party = Pool().get('party.party')
 
         if uri in ('Contacts', 'Contacts/'):
-            domain = self._carddav_filter_domain(filter)
-            party_ids = party_obj.search(domain)
-            parties = party_obj.browse(party_ids)
+            domain = cls._carddav_filter_domain(filter)
+            parties = Party.search(domain)
             if cache is not None:
                 cache.setdefault('_contact', {})
-                for party_id in party_ids:
-                    cache['_contact'][party_id] = {}
+                for party in parties:
+                    cache['_contact'][party.id] = {}
             return [x.uuid + '.vcf' for x in parties]
-        party_id = self.vcard(uri)
+        party_id = cls.vcard(uri)
         if party_id or party_id is None:
             return []
-        res = super(Collection, self).get_childs(uri, filter=filter,
+        res = super(Collection, cls).get_childs(uri, filter=filter,
                 cache=cache)
         if not uri and not filter:
             res.append('Contacts')
         return res
 
-    def get_resourcetype(self, uri, cache=None):
+    @classmethod
+    def get_resourcetype(cls, uri, cache=None):
         from DAV.constants import COLLECTION, OBJECT
-        party_id = self.vcard(uri)
+        party_id = cls.vcard(uri)
         if party_id:
             return OBJECT
         elif party_id is None:
             return COLLECTION
-        return super(Collection, self).get_resourcetype(uri, cache=cache)
+        return super(Collection, cls).get_resourcetype(uri, cache=cache)
 
-    def get_contenttype(self, uri, cache=None):
-        if self.vcard(uri):
+    @classmethod
+    def get_contenttype(cls, uri, cache=None):
+        if cls.vcard(uri):
             return 'text/x-vcard'
-        return super(Collection, self).get_contenttype(uri, cache=cache)
+        return super(Collection, cls).get_contenttype(uri, cache=cache)
 
-    def get_creationdate(self, uri, cache=None):
-        party_obj = Pool().get('party.party')
-        party_id = self.vcard(uri)
+    @classmethod
+    def get_creationdate(cls, uri, cache=None):
+        Party = Pool().get('party.party')
+        party_id = cls.vcard(uri)
 
         cursor = Transaction().cursor
 
@@ -185,7 +192,7 @@ class Collection(ModelSQL, ModelView):
                 red_sql, red_ids = reduce_ids('id', sub_ids)
                 cursor.execute('SELECT id, ' \
                             'EXTRACT(epoch FROM create_date) ' \
-                        'FROM "' + party_obj._table + '" ' \
+                        'FROM "' + Party._table + '" ' \
                         'WHERE ' + red_sql, red_ids)
                 for party_id2, date in cursor.fetchall():
                     if party_id2 == party_id:
@@ -195,17 +202,18 @@ class Collection(ModelSQL, ModelView):
                         cache['_contact'][party_id2]['creationdate'] = date
             if res is not None:
                 return res
-        return super(Collection, self).get_creationdate(uri, cache=cache)
+        return super(Collection, cls).get_creationdate(uri, cache=cache)
 
-    def get_lastmodified(self, uri, cache=None):
+    @classmethod
+    def get_lastmodified(cls, uri, cache=None):
         pool = Pool()
-        party_obj = pool.get('party.party')
-        address_obj = pool.get('party.address')
-        contact_mechanism_obj = pool.get('party.contact_mechanism')
+        Party = pool.get('party.party')
+        Address = pool.get('party.address')
+        ContactMechanism = pool.get('party.contact_mechanism')
 
         cursor = Transaction().cursor
 
-        party_id = self.vcard(uri)
+        party_id = cls.vcard(uri)
         if party_id:
             if cache is not None:
                 cache.setdefault('_contact', {})
@@ -227,10 +235,10 @@ class Collection(ModelSQL, ModelView):
                             'COALESCE(a.write_date, a.create_date))), '
                         'MAX(EXTRACT(epoch FROM '
                             'COALESCE(c.write_date, c.create_date))) '
-                    'FROM "' + party_obj._table + '" p '
-                    'LEFT JOIN "' + address_obj._table + '" a '
+                    'FROM "' + Party._table + '" p '
+                    'LEFT JOIN "' + Address._table + '" a '
                         'ON p.id = a.party '
-                    'LEFT JOIN "' + contact_mechanism_obj._table + '" c '
+                    'LEFT JOIN "' + ContactMechanism._table + '" c '
                         'ON p.id = c.party '
                     'WHERE ' + red_sql + ' '
                     'GROUP BY p.id', red_ids)
@@ -243,93 +251,98 @@ class Collection(ModelSQL, ModelView):
                         cache['_contact'][party_id2]['lastmodified'] = date
             if res is not None:
                 return res
-        return super(Collection, self).get_lastmodified(uri, cache=cache)
+        return super(Collection, cls).get_lastmodified(uri, cache=cache)
 
-    def get_data(self, uri, cache=None):
-        vcard_obj = Pool().get('party_vcarddav.party.vcard', type='report')
-        party_id = self.vcard(uri)
+    @classmethod
+    def get_data(cls, uri, cache=None):
+        Vcard = Pool().get('party_vcarddav.party.vcard', type='report')
+        party_id = cls.vcard(uri)
         if party_id is None:
             raise DAV_NotFound
         if party_id:
-            val = vcard_obj.execute([party_id],
-                    {'id': party_id, 'ids': [party_id]},
-                    )
+            val = Vcard.execute([party_id],
+                {'id': party_id, 'ids': [party_id]})
             return val[1]
-        return super(Collection, self).get_data(uri, cache=cache)
+        return super(Collection, cls).get_data(uri, cache=cache)
 
-    def get_address_data(self, uri, cache=None):
-        vcard_obj = Pool().get('party_vcarddav.party.vcard', type='report')
-        party_id = self.vcard(uri)
+    @classmethod
+    def get_address_data(cls, uri, cache=None):
+        Vcard = Pool().get('party_vcarddav.party.vcard', type='report')
+        party_id = cls.vcard(uri)
         if not party_id:
             raise DAV_NotFound
-        return vcard_obj.execute([party_id],
+        return Vcard.execute([party_id],
             {'id': party_id, 'ids': [party_id]},
             ).decode('utf-8')
 
-    def put(self, uri, data, content_type, cache=None):
+    @classmethod
+    def put(cls, uri, data, content_type, cache=None):
         import vobject
-        party_obj = Pool().get('party.party')
+        Party = Pool().get('party.party')
 
-        party_id = self.vcard(uri)
+        party_id = cls.vcard(uri)
         if party_id is None:
             vcard = vobject.readOne(data)
-            values = party_obj.vcard2values(None, vcard)
+            values = Party().vcard2values(vcard)
             try:
-                party_id = party_obj.create(values)
+                party_id = Party.create(values)
             except Exception:
                 raise DAV_Forbidden
-            party = party_obj.browse(party_id)
+            party = Party(party_id)
             return (Transaction().cursor.database_name + '/Contacts/' +
                     party.uuid + '.vcf')
         if party_id:
+            party = Party(party_id)
             vcard = vobject.readOne(data)
-            values = party_obj.vcard2values(party_id, vcard)
+            values = party.vcard2values(vcard)
             try:
-                party_obj.write(party_id, values)
+                Party.write([party], values)
             except Exception:
                 raise DAV_Forbidden
             return
-        return super(Collection, self).put(uri, data, content_type,
+        return super(Collection, cls).put(uri, data, content_type,
             cache=cache)
 
-    def mkcol(self, uri, cache=None):
-        party_id = self.vcard(uri)
+    @classmethod
+    def mkcol(cls, uri, cache=None):
+        party_id = cls.vcard(uri)
         if party_id is None:
             raise DAV_Forbidden
         if party_id:
             raise DAV_Forbidden
-        return super(Collection, self).mkcol(uri, cache=cache)
+        return super(Collection, cls).mkcol(uri, cache=cache)
 
-    def rmcol(self, uri, cache=None):
-        party_id = self.vcard(uri)
+    @classmethod
+    def rmcol(cls, uri, cache=None):
+        party_id = cls.vcard(uri)
         if party_id is None:
             raise DAV_Forbidden
         if party_id:
             raise DAV_Forbidden
-        return super(Collection, self).rmcol(uri, cache=cache)
+        return super(Collection, cls).rmcol(uri, cache=cache)
 
-    def rm(self, uri, cache=None):
-        party_obj = Pool().get('party.party')
+    @classmethod
+    def rm(cls, uri, cache=None):
+        Party = Pool().get('party.party')
 
-        party_id = self.vcard(uri)
+        party_id = cls.vcard(uri)
         if party_id is None:
             raise DAV_Forbidden
         if party_id:
             try:
-                party_obj.delete(party_id)
+                Party.delete([Party(party_id)])
             except Exception:
                 raise DAV_Forbidden
             return 200
-        return super(Collection, self).rm(uri, cache=cache)
+        return super(Collection, cls).rm(uri, cache=cache)
 
-    def exists(self, uri, cache=None):
-        party_id = self.vcard(uri)
+    @classmethod
+    def exists(cls, uri, cache=None):
+        party_id = cls.vcard(uri)
         if party_id is None or party_id:
             if party_id:
                 return 1
             if uri in ('Contacts', 'Contacts/'):
                 return 1
             return 0
-        return super(Collection, self).exists(uri, cache=cache)
-
-Collection()
+        return super(Collection, cls).exists(uri, cache=cache)
commit 08cbdba56c2bc88ea37a538046e8b21285d0fc02
Author: Mathias Behrle <mathiasb at m9s.biz>
Date:   Tue Sep 11 13:16:46 2012 +0200

    Adding upstream version 2.4.1.

diff --git a/CHANGELOG b/CHANGELOG
index 15c3081..e6fec4c 100644
--- a/CHANGELOG
+++ b/CHANGELOG
@@ -1,3 +1,6 @@
+Version 2.4.1 - 2012-09-02
+* Bug fixes (see mercurial logs for details)
+
 Version 2.4.0 - 2012-04-24
 * Bug fixes (see mercurial logs for details)
 
diff --git a/PKG-INFO b/PKG-INFO
index 87aca61..9dc963c 100644
--- a/PKG-INFO
+++ b/PKG-INFO
@@ -1,6 +1,6 @@
 Metadata-Version: 1.0
 Name: trytond_party_vcarddav
-Version: 2.4.0
+Version: 2.4.1
 Summary: Add CardDAV on parties
 Home-page: http://www.tryton.org/
 Author: B2CK
diff --git a/__tryton__.py b/__tryton__.py
index fd27138..14f9973 100644
--- a/__tryton__.py
+++ b/__tryton__.py
@@ -10,7 +10,7 @@
     'name_es_CO': 'vCardDAV de Compañía',
     'name_es_ES': 'Tercero - vCardDAV',
     'name_fr_FR': 'Tiers - vCardDAV',
-    'version': '2.4.0',
+    'version': '2.4.1',
     'author': 'B2CK',
     'email': 'info at b2ck.com',
     'website': 'http://www.tryton.org/',
diff --git a/party.py b/party.py
index 58c50ce..74d2a12 100644
--- a/party.py
+++ b/party.py
@@ -1,6 +1,5 @@
 #This file is part of Tryton.  The COPYRIGHT file at the top level of
 #this repository contains the full copyright notices and license terms.
-import base64
 import copy
 import uuid
 from trytond.model import ModelSQL, ModelView, fields
@@ -50,11 +49,23 @@ class Party(ModelSQL, ModelView):
         return res
 
     def copy(self, ids, default=None):
+        int_id = isinstance(ids, (int, long))
+        if int_id:
+            ids = [ids]
+
         if default is None:
             default = {}
-        default = default.copy()
-        default['uuid'] = self.default_uuid()
-        return super(Party, self).copy(ids, default=default)
+
+        new_ids = []
+        for party_id in ids:
+            current_default = default.copy()
+            current_default['uuid'] = self.default_uuid()
+            new_id = super(Party, self).copy(party_id, default=current_default)
+            new_ids.append(new_id)
+
+        if int_id:
+            return new_ids[0]
+        return new_ids
 
     def write(self, ids, vals):
         collection_obj = Pool().get('webdav.collection')
@@ -275,7 +286,7 @@ class VCard(Report):
         data = ''.join(self.create_vcard(party).serialize()
             for party in parties)
 
-        return ('vcf', base64.encodestring(data), False, action_report.name)
+        return ('vcf', buffer(data), False, action_report.name)
 
     def create_vcard(self, party):
         '''
@@ -287,7 +298,7 @@ class VCard(Report):
         import vobject
 
         if party.vcard:
-            vcard = vobject.readOne(party.vcard)
+            vcard = vobject.readOne(str(party.vcard))
         else:
             vcard = vobject.vCard()
         if not hasattr(vcard, 'n'):
diff --git a/trytond_party_vcarddav.egg-info/PKG-INFO b/trytond_party_vcarddav.egg-info/PKG-INFO
index 86d6550..9e669c8 100644
--- a/trytond_party_vcarddav.egg-info/PKG-INFO
+++ b/trytond_party_vcarddav.egg-info/PKG-INFO
@@ -1,6 +1,6 @@
 Metadata-Version: 1.0
 Name: trytond-party-vcarddav
-Version: 2.4.0
+Version: 2.4.1
 Summary: Add CardDAV on parties
 Home-page: http://www.tryton.org/
 Author: B2CK
diff --git a/webdav.py b/webdav.py
index 144748f..3c7f3de 100644
--- a/webdav.py
+++ b/webdav.py
@@ -1,6 +1,5 @@
 #This file is part of Tryton.  The COPYRIGHT file at the top level of
 #this repository contains the full copyright notices and license terms.
-import base64
 from pywebdav.lib.errors import DAV_NotFound, DAV_Forbidden
 from trytond.model import ModelView, ModelSQL
 from trytond.tools import reduce_ids
@@ -255,7 +254,7 @@ class Collection(ModelSQL, ModelView):
             val = vcard_obj.execute([party_id],
                     {'id': party_id, 'ids': [party_id]},
                     )
-            return base64.decodestring(val[1])
+            return val[1]
         return super(Collection, self).get_data(uri, cache=cache)
 
     def get_address_data(self, uri, cache=None):
@@ -263,11 +262,9 @@ class Collection(ModelSQL, ModelView):
         party_id = self.vcard(uri)
         if not party_id:
             raise DAV_NotFound
-        val = vcard_obj.execute([party_id],
-                {'id': party_id, 'ids': [party_id]},
-                )
-        res = base64.decodestring(val[1])
-        return res.decode('utf-8')
+        return vcard_obj.execute([party_id],
+            {'id': party_id, 'ids': [party_id]},
+            ).decode('utf-8')
 
     def put(self, uri, data, content_type, cache=None):
         import vobject
commit a0905fb320d3066042bcb0bc7de8737dacceed50
Author: Mathias Behrle <mathiasb at m9s.biz>
Date:   Tue Apr 24 19:30:47 2012 +0200

    Adding upstream version 2.4.0.

diff --git a/CHANGELOG b/CHANGELOG
index 06decad..15c3081 100644
--- a/CHANGELOG
+++ b/CHANGELOG
@@ -1,3 +1,6 @@
+Version 2.4.0 - 2012-04-24
+* Bug fixes (see mercurial logs for details)
+
 Version 2.2.0 - 2011-10-24
 * Bug fixes (see mercurial logs for details)
 
diff --git a/COPYRIGHT b/COPYRIGHT
index 1245dd5..8ad4b79 100644
--- a/COPYRIGHT
+++ b/COPYRIGHT
@@ -1,6 +1,6 @@
-Copyright (C) 2009-2011 Cédric Krier.
-Copyright (C) 2009-2011 Bertrand Chenal.
-Copyright (C) 2009-2011 B2CK SPRL.
+Copyright (C) 2009-2012 Cédric Krier.
+Copyright (C) 2009-2012 Bertrand Chenal.
+Copyright (C) 2009-2012 B2CK SPRL.
 
 This program is free software: you can redistribute it and/or modify
 it under the terms of the GNU General Public License as published by
diff --git a/INSTALL b/INSTALL
index 6db776c..1a36e89 100644
--- a/INSTALL
+++ b/INSTALL
@@ -4,11 +4,11 @@ Installing trytond_party_vcarddav
 Prerequisites
 -------------
 
- * Python 2.5 or later (http://www.python.org/)
+ * Python 2.6 or later (http://www.python.org/)
  * trytond (http://www.tryton.org/)
  * trytond_party (http://www.tryton.org/)
  * vobject 0.8.0 or later (http://vobject.skyhouseconsulting.com/)
- * pywebdav 0.9.3 or later (http://sourceforge.net/projects/pywebdav/)
+ * pywebdav 0.9.8 or later (http://sourceforge.net/projects/pywebdav/)
 
 Installation
 ------------
diff --git a/PKG-INFO b/PKG-INFO
index c00e768..87aca61 100644
--- a/PKG-INFO
+++ b/PKG-INFO
@@ -1,12 +1,12 @@
 Metadata-Version: 1.0
 Name: trytond_party_vcarddav
-Version: 2.2.0
+Version: 2.4.0
 Summary: Add CardDAV on parties
 Home-page: http://www.tryton.org/
 Author: B2CK
 Author-email: info at b2ck.com
 License: GPL-3
-Download-URL: http://downloads.tryton.org/2.2/
+Download-URL: http://downloads.tryton.org/2.4/
 Description: UNKNOWN
 Platform: UNKNOWN
 Classifier: Development Status :: 5 - Production/Stable
diff --git a/__init__.py b/__init__.py
index a021c2f..82686c6 100644
--- a/__init__.py
+++ b/__init__.py
@@ -1,6 +1,6 @@
 #This file is part of Tryton.  The COPYRIGHT file at the top level of
 #this repository contains the full copyright notices and license terms.
 
-import carddav
-from webdav import *
-from party import *
+from . import carddav
+from .webdav import *
+from .party import *
diff --git a/__tryton__.py b/__tryton__.py
index d6dce77..fd27138 100644
--- a/__tryton__.py
+++ b/__tryton__.py
@@ -2,35 +2,41 @@
 #This file is part of Tryton.  The COPYRIGHT file at the top level of
 #this repository contains the full copyright notices and license terms.
 {
-    'name' : 'Party - vCardDAV',
+    'name': 'Party - vCardDAV',
     'name_bg_BG': 'Фирма - vCardDAV',
+    'name_ca_ES': 'Tercers - vCardDAV',
     'name_de_DE': 'Parteien vCardDAV',
-    'name_es_CO' : 'vCardDAV de Compañía',
-    'name_es_ES' : 'Tercero - vCardDAV',
+    'name_es_AR': 'Entidad - vCardDAV',
+    'name_es_CO': 'vCardDAV de Compañía',
+    'name_es_ES': 'Tercero - vCardDAV',
     'name_fr_FR': 'Tiers - vCardDAV',
-    'version' : '2.2.0',
-    'author' : 'B2CK',
+    'version': '2.4.0',
+    'author': 'B2CK',
     'email': 'info at b2ck.com',
     'website': 'http://www.tryton.org/',
     'description': 'Add CardDAV on parties',
     'description_bg_BG': 'Добавя към партньор поддръжка на CardDAV',
+    'description_ca_ES': 'Afegeix suport per CardDAV als Tercers',
     'description_de_DE': 'Ermöglicht CardDAV für Parteien',
+    'description_es_AR': 'Añade soporte de CardDAV para entidades',
     'description_es_CO': 'Soporte de CardDAV para terceros',
     'description_es_ES': 'Añade soporte de CardDAV para terceros',
     'description_fr_FR': 'Ajoute le support CardDAV pour les tiers',
-    'depends' : [
+    'depends': [
         'ir',
         'res',
         'party',
         'webdav',
     ],
-    'xml' : [
+    'xml': [
         'party.xml',
     ],
     'translation': [
         'locale/bg_BG.po',
+        'locale/ca_ES.po',
         'locale/cs_CZ.po',
         'locale/de_DE.po',
+        'locale/es_AR.po',
         'locale/es_CO.po',
         'locale/es_ES.po',
         'locale/fr_FR.po',
diff --git a/carddav.py b/carddav.py
index 5bd9863..f8c37dc 100644
--- a/carddav.py
+++ b/carddav.py
@@ -1,13 +1,11 @@
 #This file is part of Tryton.  The COPYRIGHT file at the top level of
 #this repository contains the full copyright notices and license terms.
-from DAV import propfind
-from DAV.errors import *
+from pywebdav.lib import propfind
+from pywebdav.lib.errors import DAV_NotFound, DAV_Error
 from trytond.protocols.webdav import TrytonDAVInterface, CACHE
 from trytond.pool import Pool
 from trytond.transaction import Transaction
 
-_TRYTON_RELOAD = False
-
 TrytonDAVInterface.PROPS['urn:ietf:params:xml:ns:carddav'] = (
         'address-data',
         'addressbook-data',
@@ -16,6 +14,7 @@ TrytonDAVInterface.M_NS['urn:ietf:params:xml:ns:carddav'] = '_get_carddav'
 
 _mk_prop_response = propfind.PROPFIND.mk_prop_response
 
+
 def mk_prop_response(self, uri, good_props, bad_props, doc):
     res = _mk_prop_response(self, uri, good_props, bad_props, doc)
     dbname, uri = TrytonDAVInterface.get_dburi(uri)
@@ -32,6 +31,7 @@ def mk_prop_response(self, uri, good_props, bad_props, doc):
 
 propfind.PROPFIND.mk_prop_response = mk_prop_response
 
+
 def _get_carddav_address_data(self, uri):
     dbname, dburi = self._get_dburi(uri)
     if not dbname:
@@ -43,7 +43,7 @@ def _get_carddav_address_data(self, uri):
         raise DAV_NotFound
     try:
         res = collection_obj.get_address_data(dburi, cache=CACHE)
-    except (DAV_Error, DAV_NotFound, DAV_Secret, DAV_Forbidden):
+    except DAV_Error:
         raise
     except Exception:
         raise DAV_Error(500)
diff --git a/locale/bg_BG.po b/locale/bg_BG.po
index 60fcc7f..fbdab9b 100644
--- a/locale/bg_BG.po
+++ b/locale/bg_BG.po
@@ -2,19 +2,19 @@
 msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
-msgctxt "error:party.party:0"
+msgctxt "error:party.party:"
 msgid "The UUID of the party must be unique!"
 msgstr "UUID на партньор трабва да е уникален!"
 
-msgctxt "field:party.party,uuid:0"
+msgctxt "field:party.party,uuid:"
 msgid "UUID"
 msgstr "UUID"
 
-msgctxt "field:party.party,vcard:0"
+msgctxt "field:party.party,vcard:"
 msgid "VCard"
 msgstr "VCard"
 
-msgctxt "help:party.party,uuid:0"
+msgctxt "help:party.party,uuid:"
 msgid "Universally Unique Identifier"
 msgstr "Универсален иникален идентификатор"
 
diff --git a/locale/es_ES.po b/locale/ca_ES.po
similarity index 50%
copy from locale/es_ES.po
copy to locale/ca_ES.po
index 77b22bc..59fa348 100644
--- a/locale/es_ES.po
+++ b/locale/ca_ES.po
@@ -2,22 +2,22 @@
 msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
-msgctxt "error:party.party:0"
+msgctxt "error:party.party:"
 msgid "The UUID of the party must be unique!"
-msgstr "El UUID del tercero debe ser único"
+msgstr "L'UUID del tercer ha de ser únic"
 
-msgctxt "field:party.party,uuid:0"
+msgctxt "field:party.party,uuid:"
 msgid "UUID"
 msgstr "UUID"
 
-msgctxt "field:party.party,vcard:0"
+msgctxt "field:party.party,vcard:"
 msgid "VCard"
-msgstr "VCard"
+msgstr "vCard"
 
-msgctxt "help:party.party,uuid:0"
+msgctxt "help:party.party,uuid:"
 msgid "Universally Unique Identifier"
-msgstr "Identificador universal único"
+msgstr "Identificador universal únic"
 
 msgctxt "model:ir.action,name:report_party_vcard"
 msgid "VCard"
-msgstr "VCard"
+msgstr "vCard"
diff --git a/locale/cs_CZ.po b/locale/cs_CZ.po
index 05af030..4cac0ef 100644
--- a/locale/cs_CZ.po
+++ b/locale/cs_CZ.po
@@ -2,19 +2,19 @@
 msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
-msgctxt "error:party.party:0"
+msgctxt "error:party.party:"
 msgid "The UUID of the party must be unique!"
 msgstr ""
 
-msgctxt "field:party.party,uuid:0"
+msgctxt "field:party.party,uuid:"
 msgid "UUID"
 msgstr ""
 
-msgctxt "field:party.party,vcard:0"
+msgctxt "field:party.party,vcard:"
 msgid "VCard"
 msgstr ""
 
-msgctxt "help:party.party,uuid:0"
+msgctxt "help:party.party,uuid:"
 msgid "Universally Unique Identifier"
 msgstr ""
 
diff --git a/locale/de_DE.po b/locale/de_DE.po
index 6165913..69e8741 100644
--- a/locale/de_DE.po
+++ b/locale/de_DE.po
@@ -2,19 +2,19 @@
 msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
-msgctxt "error:party.party:0"
+msgctxt "error:party.party:"
 msgid "The UUID of the party must be unique!"
 msgstr "Die UUID für eine Partei kann nicht mehrfach vergeben werden!"
 
-msgctxt "field:party.party,uuid:0"
+msgctxt "field:party.party,uuid:"
 msgid "UUID"
 msgstr "UUID"
 
-msgctxt "field:party.party,vcard:0"
+msgctxt "field:party.party,vcard:"
 msgid "VCard"
 msgstr "VCard"
 
-msgctxt "help:party.party,uuid:0"
+msgctxt "help:party.party,uuid:"
 msgid "Universally Unique Identifier"
 msgstr "Universally Unique Identifier"
 
diff --git a/locale/es_ES.po b/locale/es_AR.po
similarity index 64%
copy from locale/es_ES.po
copy to locale/es_AR.po
index 77b22bc..c6b26fd 100644
--- a/locale/es_ES.po
+++ b/locale/es_AR.po
@@ -2,19 +2,19 @@
 msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
-msgctxt "error:party.party:0"
+msgctxt "error:party.party:"
 msgid "The UUID of the party must be unique!"
-msgstr "El UUID del tercero debe ser único"
+msgstr "¡El UUID de la entidad debe ser único!"
 
-msgctxt "field:party.party,uuid:0"
+msgctxt "field:party.party,uuid:"
 msgid "UUID"
 msgstr "UUID"
 
-msgctxt "field:party.party,vcard:0"
+msgctxt "field:party.party,vcard:"
 msgid "VCard"
 msgstr "VCard"
 
-msgctxt "help:party.party,uuid:0"
+msgctxt "help:party.party,uuid:"
 msgid "Universally Unique Identifier"
 msgstr "Identificador universal único"
 
diff --git a/locale/es_CO.po b/locale/es_CO.po
index 1ce74cf..6130afe 100644
--- a/locale/es_CO.po
+++ b/locale/es_CO.po
@@ -2,19 +2,19 @@
 msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
-msgctxt "error:party.party:0"
+msgctxt "error:party.party:"
 msgid "The UUID of the party must be unique!"
 msgstr "¡La UUID del tercero debe ser única!"
 
-msgctxt "field:party.party,uuid:0"
+msgctxt "field:party.party,uuid:"
 msgid "UUID"
 msgstr "UUID"
 
-msgctxt "field:party.party,vcard:0"
+msgctxt "field:party.party,vcard:"
 msgid "VCard"
 msgstr "VCard"
 
-msgctxt "help:party.party,uuid:0"
+msgctxt "help:party.party,uuid:"
 msgid "Universally Unique Identifier"
 msgstr "Identificador Único Universal"
 
diff --git a/locale/es_ES.po b/locale/es_ES.po
index 77b22bc..7579bc1 100644
--- a/locale/es_ES.po
+++ b/locale/es_ES.po
@@ -2,21 +2,21 @@
 msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
-msgctxt "error:party.party:0"
+msgctxt "error:party.party:"
 msgid "The UUID of the party must be unique!"
-msgstr "El UUID del tercero debe ser único"
+msgstr "El UUID del tercero debe ser único."
 
-msgctxt "field:party.party,uuid:0"
+msgctxt "field:party.party,uuid:"
 msgid "UUID"
 msgstr "UUID"
 
-msgctxt "field:party.party,vcard:0"
+msgctxt "field:party.party,vcard:"
 msgid "VCard"
 msgstr "VCard"
 
-msgctxt "help:party.party,uuid:0"
+msgctxt "help:party.party,uuid:"
 msgid "Universally Unique Identifier"
-msgstr "Identificador universal único"
+msgstr "Identificador universal único."
 
 msgctxt "model:ir.action,name:report_party_vcard"
 msgid "VCard"
diff --git a/locale/fr_FR.po b/locale/fr_FR.po
index d911b1e..3617a9d 100644
--- a/locale/fr_FR.po
+++ b/locale/fr_FR.po
@@ -2,19 +2,23 @@
 msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
-msgctxt "error:party.party:0"
+msgctxt "error:party.party:"
 msgid "The UUID of the party must be unique!"
 msgstr "L'UUID du tiers doit être unique !"
 
-msgctxt "field:party.party,uuid:0"
+msgctxt "error:party.party:"
+msgid "The UUID of the party must be unique!"
+msgstr "L'UUID du tiers doit être unique !"
+
+msgctxt "field:party.party,uuid:"
 msgid "UUID"
 msgstr "UUID"
 
-msgctxt "field:party.party,vcard:0"
+msgctxt "field:party.party,vcard:"
 msgid "VCard"
 msgstr "VCard"
 
-msgctxt "help:party.party,uuid:0"
+msgctxt "help:party.party,uuid:"
 msgid "Universally Unique Identifier"
 msgstr "Identificateur unique universel"
 
diff --git a/locale/nl_NL.po b/locale/nl_NL.po
index 05af030..4cac0ef 100644
--- a/locale/nl_NL.po
+++ b/locale/nl_NL.po
@@ -2,19 +2,19 @@
 msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
-msgctxt "error:party.party:0"
+msgctxt "error:party.party:"
 msgid "The UUID of the party must be unique!"
 msgstr ""
 
-msgctxt "field:party.party,uuid:0"
+msgctxt "field:party.party,uuid:"
 msgid "UUID"
 msgstr ""
 
-msgctxt "field:party.party,vcard:0"
+msgctxt "field:party.party,vcard:"
 msgid "VCard"
 msgstr ""
 
-msgctxt "help:party.party,uuid:0"
+msgctxt "help:party.party,uuid:"
 msgid "Universally Unique Identifier"
 msgstr ""
 
diff --git a/locale/ru_RU.po b/locale/ru_RU.po
index 08774e3..abbbe75 100644
--- a/locale/ru_RU.po
+++ b/locale/ru_RU.po
@@ -2,20 +2,20 @@
 msgid ""
 msgstr "Content-Type: text/plain; charset=utf-8\n"
 
-msgctxt "error:party.party:0"
+msgctxt "error:party.party:"
 msgid "The UUID of the party must be unique!"
 msgstr ""
 
-msgctxt "field:party.party,uuid:0"
+msgctxt "field:party.party,uuid:"
 msgid "UUID"
 msgstr ""
 
-msgctxt "field:party.party,vcard:0"
+msgctxt "field:party.party,vcard:"
 msgid "VCard"
 msgstr ""
 
 #, fuzzy
-msgctxt "help:party.party,uuid:0"
+msgctxt "help:party.party,uuid:"
 msgid "Universally Unique Identifier"
 msgstr "Универсальный уникальный идентификатор"
 
diff --git a/party.py b/party.py
index c3f8fd0..58c50ce 100644
--- a/party.py
+++ b/party.py
@@ -272,7 +272,8 @@ class VCard(Report):
 
         parties = party_obj.browse(ids)
 
-        data = ''.join(self.create_vcard(party).serialize() for party in parties)
+        data = ''.join(self.create_vcard(party).serialize()
+            for party in parties)
 
         return ('vcf', base64.encodestring(data), False, action_report.name)
 
diff --git a/setup.py b/setup.py
index bad90eb..a321362 100644
--- a/setup.py
+++ b/setup.py
@@ -10,9 +10,9 @@ major_version, minor_version, _ = info.get('version', '0.0.1').split('.', 2)
 major_version = int(major_version)
 minor_version = int(minor_version)
 
-requires = ['vobject >= 0.8.0', 'PyWebDAV >= 0.9.3']
+requires = ['vobject >= 0.8.0', 'PyWebDAV >= 0.9.8']
 for dep in info.get('depends', []):
-    if not re.match(r'(ir|res|workflow|webdav)(\W|$)', dep):
+    if not re.match(r'(ir|res|webdav)(\W|$)', dep):
         requires.append('trytond_%s >= %s.%s, < %s.%s' %
                 (dep, major_version, minor_version, major_version,
                     minor_version + 1))
diff --git a/trytond_party_vcarddav.egg-info/PKG-INFO b/trytond_party_vcarddav.egg-info/PKG-INFO
index 23e55a0..86d6550 100644
--- a/trytond_party_vcarddav.egg-info/PKG-INFO
+++ b/trytond_party_vcarddav.egg-info/PKG-INFO
@@ -1,12 +1,12 @@
 Metadata-Version: 1.0
 Name: trytond-party-vcarddav
-Version: 2.2.0
+Version: 2.4.0
 Summary: Add CardDAV on parties
 Home-page: http://www.tryton.org/
 Author: B2CK
 Author-email: info at b2ck.com
 License: GPL-3
-Download-URL: http://downloads.tryton.org/2.2/
+Download-URL: http://downloads.tryton.org/2.4/
 Description: UNKNOWN
 Platform: UNKNOWN
 Classifier: Development Status :: 5 - Production/Stable
diff --git a/trytond_party_vcarddav.egg-info/SOURCES.txt b/trytond_party_vcarddav.egg-info/SOURCES.txt
index 66fee8f..82f82db 100644
--- a/trytond_party_vcarddav.egg-info/SOURCES.txt
+++ b/trytond_party_vcarddav.egg-info/SOURCES.txt
@@ -13,8 +13,10 @@ setup.py
 ./webdav.py
 doc/index.rst
 locale/bg_BG.po
+locale/ca_ES.po
 locale/cs_CZ.po
 locale/de_DE.po
+locale/es_AR.po
 locale/es_CO.po
 locale/es_ES.po
 locale/fr_FR.po
diff --git a/trytond_party_vcarddav.egg-info/requires.txt b/trytond_party_vcarddav.egg-info/requires.txt
index 8e9bc42..6e82de2 100644
--- a/trytond_party_vcarddav.egg-info/requires.txt
+++ b/trytond_party_vcarddav.egg-info/requires.txt
@@ -1,4 +1,4 @@
 vobject >= 0.8.0
-PyWebDAV >= 0.9.3
-trytond_party >= 2.2, < 2.3
-trytond >= 2.2, < 2.3
\ No newline at end of file
+PyWebDAV >= 0.9.8
+trytond_party >= 2.4, < 2.5
+trytond >= 2.4, < 2.5
\ No newline at end of file
diff --git a/webdav.py b/webdav.py
index 68a6156..144748f 100644
--- a/webdav.py
+++ b/webdav.py
@@ -1,8 +1,7 @@
 #This file is part of Tryton.  The COPYRIGHT file at the top level of
 #this repository contains the full copyright notices and license terms.
 import base64
-import urlparse
-from DAV.errors import DAV_NotFound, DAV_Forbidden
+from pywebdav.lib.errors import DAV_NotFound, DAV_Forbidden
 from trytond.model import ModelView, ModelSQL
 from trytond.tools import reduce_ids
 from trytond.transaction import Transaction
@@ -78,13 +77,15 @@ class Collection(ModelSQL, ModelView):
                     field = 'value'
                 if field:
                     res2 = []
-                    if prop.hasAttribute('test') \
-                            and prop.addressbook_filter.getAttribute('test') == 'allof':
+                    if (prop.hasAttribute('test')
+                            and (prop.addressbook_filter.getAttribute('test')
+                                == 'allof')):
                         res2.append('AND')
                     else:
                         res2.append('OR')
-                    if prop.getElementsByTagNameNS(CARDDAV_NS, 'is-not-defined'):
-                        res2.append((field, '=', False))
+                    if prop.getElementsByTagNameNS(CARDDAV_NS,
+                            'is-not-defined'):
+                        res2.append((field, '=', None))
                     for text_match in prop.getElementsByTagNameNS(CARDDAV_NS,
                             'text-match'):
                         value = text_match.firstChild.data
@@ -220,20 +221,20 @@ class Collection(ModelSQL, ModelView):
             for i in range(0, len(ids), cursor.IN_MAX):
                 sub_ids = ids[i:i + cursor.IN_MAX]
                 red_sql, red_ids = reduce_ids('p.id', sub_ids)
-                cursor.execute('SELECT p.id, ' \
-                            'MAX(EXTRACT(epoch FROM ' \
-                                'COALESCE(p.write_date, p.create_date))), ' \
-                            'MAX(EXTRACT(epoch FROM ' \
-                                'COALESCE(a.write_date, a.create_date))), ' \
-                            'MAX(EXTRACT(epoch FROM ' \
-                                'COALESCE(c.write_date, c.create_date))) ' \
-                        'FROM "' + party_obj._table + '" p ' \
-                            'LEFT JOIN "' + address_obj._table + '" a ' \
-                            'ON p.id = a.party ' \
-                            'LEFT JOIN "' + contact_mechanism_obj._table + '" c ' \
-                            'ON p.id = c.party ' \
-                        'WHERE ' + red_sql + ' ' \
-                        'GROUP BY p.id', red_ids)
+                cursor.execute('SELECT p.id, '
+                        'MAX(EXTRACT(epoch FROM '
+                            'COALESCE(p.write_date, p.create_date))), '
+                        'MAX(EXTRACT(epoch FROM '
+                            'COALESCE(a.write_date, a.create_date))), '
+                        'MAX(EXTRACT(epoch FROM '
+                            'COALESCE(c.write_date, c.create_date))) '
+                    'FROM "' + party_obj._table + '" p '
+                    'LEFT JOIN "' + address_obj._table + '" a '
+                        'ON p.id = a.party '
+                    'LEFT JOIN "' + contact_mechanism_obj._table + '" c '
+                        'ON p.id = c.party '
+                    'WHERE ' + red_sql + ' '
+                    'GROUP BY p.id', red_ids)
                 for party_id2, date_p, date_a, date_c in cursor.fetchall():
                     date = max(date_p, date_a, date_c)
                     if party_id2 == party_id:
@@ -291,7 +292,8 @@ class Collection(ModelSQL, ModelView):
             except Exception:
                 raise DAV_Forbidden
             return
-        return super(Collection, self).put(uri, data, content_type, cache=cache)
+        return super(Collection, self).put(uri, data, content_type,
+            cache=cache)
 
     def mkcol(self, uri, cache=None):
         party_id = self.vcard(uri)
commit 9b42710bf780a9cccbfd3db4f52d338615f9a1be
Author: Mathias Behrle <mathiasb at m9s.biz>
Date:   Mon Oct 31 16:20:58 2011 +0100

    Adding upstream version 2.2.0.

diff --git a/CHANGELOG b/CHANGELOG
index 154bd85..06decad 100644
--- a/CHANGELOG
+++ b/CHANGELOG
@@ -1,3 +1,6 @@
+Version 2.2.0 - 2011-10-24
+* Bug fixes (see mercurial logs for details)
+
 Version 2.0.0 - 2011-04-27
 * Bug fixes (see mercurial logs for details)
 
diff --git a/MANIFEST.in b/MANIFEST.in
index dcb2afa..32879df 100644
--- a/MANIFEST.in
+++ b/MANIFEST.in
@@ -6,5 +6,5 @@ include CHANGELOG
 include LICENSE
 include *.xml
 include *.odt
-include *.csv
+include locale/*.po
 include doc/*
diff --git a/PKG-INFO b/PKG-INFO
index d184866..c00e768 100644
--- a/PKG-INFO
+++ b/PKG-INFO
@@ -1,28 +1,31 @@
 Metadata-Version: 1.0
 Name: trytond_party_vcarddav
-Version: 2.0.0
+Version: 2.2.0
 Summary: Add CardDAV on parties
 Home-page: http://www.tryton.org/
 Author: B2CK
 Author-email: info at b2ck.com
 License: GPL-3
-Download-URL: http://downloads.tryton.org/2.0/
+Download-URL: http://downloads.tryton.org/2.2/
 Description: UNKNOWN
 Platform: UNKNOWN
 Classifier: Development Status :: 5 - Production/Stable
 Classifier: Environment :: Plugins
+Classifier: Framework :: Tryton
 Classifier: Intended Audience :: Developers
 Classifier: Intended Audience :: Financial and Insurance Industry
 Classifier: Intended Audience :: Legal Industry
 Classifier: Intended Audience :: Manufacturing
 Classifier: License :: OSI Approved :: GNU General Public License (GPL)
 Classifier: Natural Language :: Bulgarian
+Classifier: Natural Language :: Czech
+Classifier: Natural Language :: Dutch
 Classifier: Natural Language :: English
 Classifier: Natural Language :: French
 Classifier: Natural Language :: German
+Classifier: Natural Language :: Russian
 Classifier: Natural Language :: Spanish
 Classifier: Operating System :: OS Independent
-Classifier: Programming Language :: Python :: 2.5
 Classifier: Programming Language :: Python :: 2.6
 Classifier: Programming Language :: Python :: 2.7
 Classifier: Topic :: Office/Business
diff --git a/__tryton__.py b/__tryton__.py
index cdd395b..d6dce77 100644
--- a/__tryton__.py
+++ b/__tryton__.py
@@ -8,7 +8,7 @@
     'name_es_CO' : 'vCardDAV de Compañía',
     'name_es_ES' : 'Tercero - vCardDAV',
     'name_fr_FR': 'Tiers - vCardDAV',
-    'version' : '2.0.0',
+    'version' : '2.2.0',
     'author' : 'B2CK',
     'email': 'info at b2ck.com',
     'website': 'http://www.tryton.org/',
@@ -28,10 +28,13 @@
         'party.xml',
     ],
     'translation': [
-        'bg_BG.csv',
-        'de_DE.csv',
-        'es_CO.csv',
-        'es_ES.csv',
-        'fr_FR.csv',
+        'locale/bg_BG.po',
+        'locale/cs_CZ.po',
+        'locale/de_DE.po',
+        'locale/es_CO.po',
+        'locale/es_ES.po',
+        'locale/fr_FR.po',
+        'locale/nl_NL.po',
+        'locale/ru_RU.po',
     ],
 }
diff --git a/bg_BG.csv b/bg_BG.csv
deleted file mode 100644
index c815f2e..0000000
--- a/bg_BG.csv
+++ /dev/null
@@ -1,7 +0,0 @@
-type,name,res_id,src,value,fuzzy
-error,party.party,0,The UUID of the party must be unique!,UUID на партньор трабва да е уникален!,0
-field,"party.party,uuid",0,UUID,UUID,0
-field,"party.party,vcard",0,VCard,VCard,0
-help,"party.party,uuid",0,Universally Unique Identifier,Универсален иникален идентификатор,0
-model,"ir.action,name",report_party_vcard,VCard,VCard,0
-selection,"ir.action.report,extension",0,VCard file,VCard файл,0
diff --git a/de_DE.csv b/de_DE.csv
deleted file mode 100644
index 093552c..0000000
--- a/de_DE.csv
+++ /dev/null
@@ -1,7 +0,0 @@
-type,name,res_id,src,value,fuzzy
-error,party.party,0,The UUID of the party must be unique!,Die UUID für eine Partei kann nicht mehrfach vergeben werden!,0
-field,"party.party,uuid",0,UUID,UUID,0
-field,"party.party,vcard",0,VCard,VCard,0
-help,"party.party,uuid",0,Universally Unique Identifier,Universally Unique Identifier,0
-model,"ir.action,name",report_party_vcard,VCard,VCard,0
-selection,"ir.action.report,extension",0,VCard file,VCard Datei,0
diff --git a/es_CO.csv b/es_CO.csv
deleted file mode 100644
index df05f1b..0000000
--- a/es_CO.csv
+++ /dev/null
@@ -1,7 +0,0 @@
-type,name,res_id,src,value,fuzzy
-error,party.party,0,The UUID of the party must be unique!,¡La UUID del tercero debe ser única!,0
-field,"party.party,uuid",0,UUID,UUID,0
-field,"party.party,vcard",0,VCard,VCard,0
-help,"party.party,uuid",0,Universally Unique Identifier,Identificador Único Universal,0
-model,"ir.action,name",report_party_vcard,VCard,VCard,0
-selection,"ir.action.report,extension",0,VCard file,Archivo VCard,0
diff --git a/es_ES.csv b/es_ES.csv
deleted file mode 100644
index 3224334..0000000
--- a/es_ES.csv
+++ /dev/null
@@ -1,7 +0,0 @@
-type,name,res_id,src,value,fuzzy
-error,party.party,0,The UUID of the party must be unique!,El UUID del tercero debe ser único,0
-field,"party.party,uuid",0,UUID,UUID,0
-field,"party.party,vcard",0,VCard,VCard,0
-help,"party.party,uuid",0,Universally Unique Identifier,Identificador universal único,0
-model,"ir.action,name",report_party_vcard,VCard,VCard,0
-selection,"ir.action.report,extension",0,VCard file,Archivo VCard,0
diff --git a/fr_FR.csv b/fr_FR.csv
deleted file mode 100644
index 6845ba2..0000000
--- a/fr_FR.csv
+++ /dev/null
@@ -1,7 +0,0 @@
-type,name,res_id,src,value,fuzzy
-error,party.party,0,The UUID of the party must be unique!,L'UUID du tiers doit être unique !,0
-field,"party.party,uuid",0,UUID,UUID,0
-field,"party.party,vcard",0,VCard,VCard,0
-help,"party.party,uuid",0,Universally Unique Identifier,Identificateur unique universel,0
-model,"ir.action,name",report_party_vcard,VCard,VCard,0
-selection,"ir.action.report,extension",0,VCard file,Fichier VCard,0
diff --git a/locale/bg_BG.po b/locale/bg_BG.po
new file mode 100644
index 0000000..60fcc7f
--- /dev/null
+++ b/locale/bg_BG.po
@@ -0,0 +1,23 @@
+# 
+msgid ""
+msgstr "Content-Type: text/plain; charset=utf-8\n"
+
+msgctxt "error:party.party:0"
+msgid "The UUID of the party must be unique!"
+msgstr "UUID на партньор трабва да е уникален!"
+
+msgctxt "field:party.party,uuid:0"
+msgid "UUID"
+msgstr "UUID"
+
+msgctxt "field:party.party,vcard:0"
+msgid "VCard"
+msgstr "VCard"
+
+msgctxt "help:party.party,uuid:0"
+msgid "Universally Unique Identifier"
+msgstr "Универсален иникален идентификатор"
+
+msgctxt "model:ir.action,name:report_party_vcard"
+msgid "VCard"
+msgstr "VCard"
diff --git a/locale/cs_CZ.po b/locale/cs_CZ.po
new file mode 100644
index 0000000..05af030
--- /dev/null
+++ b/locale/cs_CZ.po
@@ -0,0 +1,23 @@
+# 
+msgid ""
+msgstr "Content-Type: text/plain; charset=utf-8\n"
+
+msgctxt "error:party.party:0"
+msgid "The UUID of the party must be unique!"
+msgstr ""
+
+msgctxt "field:party.party,uuid:0"
+msgid "UUID"
+msgstr ""
+
+msgctxt "field:party.party,vcard:0"
+msgid "VCard"
+msgstr ""
+
+msgctxt "help:party.party,uuid:0"
+msgid "Universally Unique Identifier"
+msgstr ""
+
+msgctxt "model:ir.action,name:report_party_vcard"
+msgid "VCard"
+msgstr ""
diff --git a/locale/de_DE.po b/locale/de_DE.po
new file mode 100644
index 0000000..6165913
--- /dev/null
+++ b/locale/de_DE.po
@@ -0,0 +1,23 @@
+# 
+msgid ""
+msgstr "Content-Type: text/plain; charset=utf-8\n"
+
+msgctxt "error:party.party:0"
+msgid "The UUID of the party must be unique!"
+msgstr "Die UUID für eine Partei kann nicht mehrfach vergeben werden!"
+
+msgctxt "field:party.party,uuid:0"
+msgid "UUID"
+msgstr "UUID"
+
+msgctxt "field:party.party,vcard:0"
+msgid "VCard"
+msgstr "VCard"
+
+msgctxt "help:party.party,uuid:0"
+msgid "Universally Unique Identifier"
+msgstr "Universally Unique Identifier"
+
+msgctxt "model:ir.action,name:report_party_vcard"
+msgid "VCard"
+msgstr "VCard"
diff --git a/locale/es_CO.po b/locale/es_CO.po
new file mode 100644
index 0000000..1ce74cf
--- /dev/null
+++ b/locale/es_CO.po
@@ -0,0 +1,23 @@
+# 
+msgid ""
+msgstr "Content-Type: text/plain; charset=utf-8\n"
+
+msgctxt "error:party.party:0"
+msgid "The UUID of the party must be unique!"
+msgstr "¡La UUID del tercero debe ser única!"
+
+msgctxt "field:party.party,uuid:0"
+msgid "UUID"
+msgstr "UUID"
+
+msgctxt "field:party.party,vcard:0"
+msgid "VCard"
+msgstr "VCard"
+
+msgctxt "help:party.party,uuid:0"
+msgid "Universally Unique Identifier"
+msgstr "Identificador Único Universal"
+
+msgctxt "model:ir.action,name:report_party_vcard"
+msgid "VCard"
+msgstr "VCard"
diff --git a/locale/es_ES.po b/locale/es_ES.po
new file mode 100644
index 0000000..77b22bc
--- /dev/null
+++ b/locale/es_ES.po
@@ -0,0 +1,23 @@
+# 
+msgid ""
+msgstr "Content-Type: text/plain; charset=utf-8\n"
+
+msgctxt "error:party.party:0"
+msgid "The UUID of the party must be unique!"
+msgstr "El UUID del tercero debe ser único"
+
+msgctxt "field:party.party,uuid:0"
+msgid "UUID"
+msgstr "UUID"
+
+msgctxt "field:party.party,vcard:0"
+msgid "VCard"
+msgstr "VCard"
+
+msgctxt "help:party.party,uuid:0"
+msgid "Universally Unique Identifier"
+msgstr "Identificador universal único"
+
+msgctxt "model:ir.action,name:report_party_vcard"
+msgid "VCard"
+msgstr "VCard"
diff --git a/locale/fr_FR.po b/locale/fr_FR.po
new file mode 100644
index 0000000..d911b1e
--- /dev/null
+++ b/locale/fr_FR.po
@@ -0,0 +1,23 @@
+# 
+msgid ""
+msgstr "Content-Type: text/plain; charset=utf-8\n"
+
+msgctxt "error:party.party:0"
+msgid "The UUID of the party must be unique!"
+msgstr "L'UUID du tiers doit être unique !"
+
+msgctxt "field:party.party,uuid:0"
+msgid "UUID"
+msgstr "UUID"
+
+msgctxt "field:party.party,vcard:0"
+msgid "VCard"
+msgstr "VCard"
+
+msgctxt "help:party.party,uuid:0"
+msgid "Universally Unique Identifier"
+msgstr "Identificateur unique universel"
+
+msgctxt "model:ir.action,name:report_party_vcard"
+msgid "VCard"
+msgstr "VCard"
diff --git a/locale/nl_NL.po b/locale/nl_NL.po
new file mode 100644
index 0000000..05af030
--- /dev/null
+++ b/locale/nl_NL.po
@@ -0,0 +1,23 @@
+# 
+msgid ""
+msgstr "Content-Type: text/plain; charset=utf-8\n"
+
+msgctxt "error:party.party:0"
+msgid "The UUID of the party must be unique!"
+msgstr ""
+
+msgctxt "field:party.party,uuid:0"
+msgid "UUID"
+msgstr ""
+
+msgctxt "field:party.party,vcard:0"
+msgid "VCard"
+msgstr ""
+
+msgctxt "help:party.party,uuid:0"
+msgid "Universally Unique Identifier"
+msgstr ""
+
+msgctxt "model:ir.action,name:report_party_vcard"
+msgid "VCard"
+msgstr ""
diff --git a/locale/ru_RU.po b/locale/ru_RU.po
new file mode 100644
index 0000000..08774e3
--- /dev/null
+++ b/locale/ru_RU.po
@@ -0,0 +1,24 @@
+# 
+msgid ""
+msgstr "Content-Type: text/plain; charset=utf-8\n"
+
+msgctxt "error:party.party:0"
+msgid "The UUID of the party must be unique!"
+msgstr ""
+
+msgctxt "field:party.party,uuid:0"
+msgid "UUID"
+msgstr ""
+
+msgctxt "field:party.party,vcard:0"
+msgid "VCard"
+msgstr ""
+
+#, fuzzy
+msgctxt "help:party.party,uuid:0"
+msgid "Universally Unique Identifier"
+msgstr "Универсальный уникальный идентификатор"
+
+msgctxt "model:ir.action,name:report_party_vcard"
+msgid "VCard"
+msgstr ""
diff --git a/party.py b/party.py
index ad90956..c3f8fd0 100644
--- a/party.py
+++ b/party.py
@@ -7,6 +7,7 @@ from trytond.model import ModelSQL, ModelView, fields
 from trytond.report import Report
 from trytond.backend import TableHandler, FIELDS
 from trytond.transaction import Transaction
+from trytond.pool import Pool
 
 
 class Party(ModelSQL, ModelView):
@@ -41,15 +42,22 @@ class Party(ModelSQL, ModelView):
         return str(uuid.uuid4())
 
     def create(self, vals):
-        collection_obj = self.pool.get('webdav.collection')
+        collection_obj = Pool().get('webdav.collection')
 
         res = super(Party, self).create(vals)
         # Restart the cache for vcard
         collection_obj.vcard.reset()
         return res
 
+    def copy(self, ids, default=None):
+        if default is None:
+            default = {}
+        default = default.copy()
+        default['uuid'] = self.default_uuid()
+        return super(Party, self).copy(ids, default=default)
+
     def write(self, ids, vals):
-        collection_obj = self.pool.get('webdav.collection')
+        collection_obj = Pool().get('webdav.collection')
 
         res = super(Party, self).write(ids, vals)
         # Restart the cache for vcard
@@ -57,7 +65,7 @@ class Party(ModelSQL, ModelView):
         return res
 
     def delete(self, ids):
-        collection_obj = self.pool.get('webdav.collection')
+        collection_obj = Pool().get('webdav.collection')
 
         res = super(Party, self).delete(ids)
         # Restart the cache for vcard
@@ -73,7 +81,7 @@ class Party(ModelSQL, ModelView):
         :return: a dictionary with values
         '''
         import vobject
-        address_obj = self.pool.get('party.address')
+        address_obj = Pool().get('party.address')
 
         res = {}
         res['name'] = vcard.fn.value
@@ -208,8 +216,8 @@ class Address(ModelSQL, ModelView):
         :param adr: a adr from vcard instance of vobject
         :return: a dictionary with values
         '''
-        country_obj = self.pool.get('country.country')
-        subdivision_obj = self.pool.get('country.subdivision')
+        country_obj = Pool().get('country.country')
+        subdivision_obj = Pool().get('country.subdivision')
 
         vals = {}
         vals['street'] = adr.value.street or ''
@@ -252,8 +260,8 @@ class VCard(Report):
     _name = 'party_vcarddav.party.vcard'
 
     def execute(self, ids, datas):
-        party_obj = self.pool.get('party.party')
-        action_report_obj = self.pool.get('ir.action.report')
+        party_obj = Pool().get('party.party')
+        action_report_obj = Pool().get('ir.action.report')
 
         action_report_ids = action_report_obj.search([
             ('report_name', '=', self._name)
diff --git a/party.xml b/party.xml
index e034ae1..bde84d1 100644
--- a/party.xml
+++ b/party.xml
@@ -13,7 +13,7 @@ this repository contains the full copyright notices and license terms. -->
 
         <record model="ir.action.keyword" id="create_vcard_keyword">
             <field name="keyword">form_print</field>
-            <field name="model">party.party,0</field>
+            <field name="model">party.party,-1</field>
             <field name="action" ref="report_party_vcard"/>
         </record>
 
diff --git a/setup.py b/setup.py
index b4f2f36..bad90eb 100644
--- a/setup.py
+++ b/setup.py
@@ -38,18 +38,21 @@ setup(name='trytond_party_vcarddav',
     classifiers=[
         'Development Status :: 5 - Production/Stable',
         'Environment :: Plugins',
+        'Framework :: Tryton',
         'Intended Audience :: Developers',
         'Intended Audience :: Financial and Insurance Industry',
         'Intended Audience :: Legal Industry',
         'Intended Audience :: Manufacturing',
         'License :: OSI Approved :: GNU General Public License (GPL)',
         'Natural Language :: Bulgarian',
+        'Natural Language :: Czech',
+        'Natural Language :: Dutch',
         'Natural Language :: English',
         'Natural Language :: French',
         'Natural Language :: German',
+        'Natural Language :: Russian',
         'Natural Language :: Spanish',
         'Operating System :: OS Independent',
-        'Programming Language :: Python :: 2.5',
         'Programming Language :: Python :: 2.6',
         'Programming Language :: Python :: 2.7',
         'Topic :: Office/Business',
@@ -61,4 +64,6 @@ setup(name='trytond_party_vcarddav',
     [trytond.modules]
     party_vcarddav = trytond.modules.party_vcarddav
     """,
+    test_suite='tests',
+    test_loader='trytond.test_loader:Loader',
 )
diff --git a/trytond_party_vcarddav.egg-info/PKG-INFO b/trytond_party_vcarddav.egg-info/PKG-INFO
index 3449cd2..23e55a0 100644
--- a/trytond_party_vcarddav.egg-info/PKG-INFO
+++ b/trytond_party_vcarddav.egg-info/PKG-INFO
@@ -1,28 +1,31 @@
 Metadata-Version: 1.0
 Name: trytond-party-vcarddav
-Version: 2.0.0
+Version: 2.2.0
 Summary: Add CardDAV on parties
 Home-page: http://www.tryton.org/
 Author: B2CK
 Author-email: info at b2ck.com
 License: GPL-3
-Download-URL: http://downloads.tryton.org/2.0/
+Download-URL: http://downloads.tryton.org/2.2/
 Description: UNKNOWN
 Platform: UNKNOWN
 Classifier: Development Status :: 5 - Production/Stable
 Classifier: Environment :: Plugins
+Classifier: Framework :: Tryton
 Classifier: Intended Audience :: Developers
 Classifier: Intended Audience :: Financial and Insurance Industry
 Classifier: Intended Audience :: Legal Industry
 Classifier: Intended Audience :: Manufacturing
 Classifier: License :: OSI Approved :: GNU General Public License (GPL)
 Classifier: Natural Language :: Bulgarian
+Classifier: Natural Language :: Czech
+Classifier: Natural Language :: Dutch
 Classifier: Natural Language :: English
 Classifier: Natural Language :: French
 Classifier: Natural Language :: German
+Classifier: Natural Language :: Russian
 Classifier: Natural Language :: Spanish
 Classifier: Operating System :: OS Independent
-Classifier: Programming Language :: Python :: 2.5
 Classifier: Programming Language :: Python :: 2.6
 Classifier: Programming Language :: Python :: 2.7
 Classifier: Topic :: Office/Business
diff --git a/trytond_party_vcarddav.egg-info/SOURCES.txt b/trytond_party_vcarddav.egg-info/SOURCES.txt
index 9679a02..66fee8f 100644
--- a/trytond_party_vcarddav.egg-info/SOURCES.txt
+++ b/trytond_party_vcarddav.egg-info/SOURCES.txt
@@ -4,11 +4,6 @@ INSTALL
 LICENSE
 MANIFEST.in
 README
-bg_BG.csv
-de_DE.csv
-es_CO.csv
-es_ES.csv
-fr_FR.csv
 party.xml
 setup.py
 ./__init__.py
@@ -17,6 +12,14 @@ setup.py
 ./party.py
 ./webdav.py
 doc/index.rst
+locale/bg_BG.po
+locale/cs_CZ.po
+locale/de_DE.po
+locale/es_CO.po
+locale/es_ES.po
+locale/fr_FR.po
+locale/nl_NL.po
+locale/ru_RU.po
 trytond_party_vcarddav.egg-info/PKG-INFO
 trytond_party_vcarddav.egg-info/SOURCES.txt
 trytond_party_vcarddav.egg-info/dependency_links.txt
diff --git a/trytond_party_vcarddav.egg-info/requires.txt b/trytond_party_vcarddav.egg-info/requires.txt
index b5d5b03..8e9bc42 100644
--- a/trytond_party_vcarddav.egg-info/requires.txt
+++ b/trytond_party_vcarddav.egg-info/requires.txt
@@ -1,4 +1,4 @@
 vobject >= 0.8.0
 PyWebDAV >= 0.9.3
-trytond_party >= 2.0, < 2.1
-trytond >= 2.0, < 2.1
\ No newline at end of file
+trytond_party >= 2.2, < 2.3
+trytond >= 2.2, < 2.3
\ No newline at end of file
diff --git a/webdav.py b/webdav.py
index ec35b22..68a6156 100644
--- a/webdav.py
+++ b/webdav.py
@@ -7,6 +7,7 @@ from trytond.model import ModelView, ModelSQL
 from trytond.tools import reduce_ids
 from trytond.transaction import Transaction
 from trytond.cache import Cache
+from trytond.pool import Pool
 
 
 CARDDAV_NS = 'urn:ietf:params:xml:ns:carddav'
@@ -26,7 +27,7 @@ class Collection(ModelSQL, ModelView):
             or None if there is no party
             or False if not in Contacts
         '''
-        party_obj = self.pool.get('party.party')
+        party_obj = Pool().get('party.party')
 
         if uri and uri.startswith('Contacts/'):
             uuid = uri[9:-4]
@@ -47,8 +48,8 @@ class Collection(ModelSQL, ModelView):
         :param filter: the DOM Element of filter
         :return: a list for domain
         '''
-        address_obj = self.pool.get('party.address')
-        contact_mechanism_obj = self.pool.get('party.contact_mechanism')
+        address_obj = Pool().get('party.address')
+        contact_mechanism_obj = Pool().get('party.contact_mechanism')
 
         res = []
         if not filter:
@@ -126,7 +127,7 @@ class Collection(ModelSQL, ModelView):
         return res
 
     def get_childs(self, uri, filter=None, cache=None):
-        party_obj = self.pool.get('party.party')
+        party_obj = Pool().get('party.party')
 
         if uri in ('Contacts', 'Contacts/'):
             domain = self._carddav_filter_domain(filter)
@@ -161,7 +162,7 @@ class Collection(ModelSQL, ModelView):
         return super(Collection, self).get_contenttype(uri, cache=cache)
 
     def get_creationdate(self, uri, cache=None):
-        party_obj = self.pool.get('party.party')
+        party_obj = Pool().get('party.party')
         party_id = self.vcard(uri)
 
         cursor = Transaction().cursor
@@ -197,9 +198,10 @@ class Collection(ModelSQL, ModelView):
         return super(Collection, self).get_creationdate(uri, cache=cache)
 
     def get_lastmodified(self, uri, cache=None):
-        party_obj = self.pool.get('party.party')
-        address_obj = self.pool.get('party.address')
-        contact_mechanism_obj = self.pool.get('party.contact_mechanism')
+        pool = Pool()
+        party_obj = pool.get('party.party')
+        address_obj = pool.get('party.address')
+        contact_mechanism_obj = pool.get('party.contact_mechanism')
 
         cursor = Transaction().cursor
 
@@ -244,7 +246,7 @@ class Collection(ModelSQL, ModelView):
         return super(Collection, self).get_lastmodified(uri, cache=cache)
 
     def get_data(self, uri, cache=None):
-        vcard_obj = self.pool.get('party_vcarddav.party.vcard', type='report')
+        vcard_obj = Pool().get('party_vcarddav.party.vcard', type='report')
         party_id = self.vcard(uri)
         if party_id is None:
             raise DAV_NotFound
@@ -256,7 +258,7 @@ class Collection(ModelSQL, ModelView):
         return super(Collection, self).get_data(uri, cache=cache)
 
     def get_address_data(self, uri, cache=None):
-        vcard_obj = self.pool.get('party_vcarddav.party.vcard', type='report')
+        vcard_obj = Pool().get('party_vcarddav.party.vcard', type='report')
         party_id = self.vcard(uri)
         if not party_id:
             raise DAV_NotFound
@@ -268,7 +270,7 @@ class Collection(ModelSQL, ModelView):
 
     def put(self, uri, data, content_type, cache=None):
         import vobject
-        party_obj = self.pool.get('party.party')
+        party_obj = Pool().get('party.party')
 
         party_id = self.vcard(uri)
         if party_id is None:
@@ -308,7 +310,7 @@ class Collection(ModelSQL, ModelView):
         return super(Collection, self).rmcol(uri, cache=cache)
 
     def rm(self, uri, cache=None):
-        party_obj = self.pool.get('party.party')
+        party_obj = Pool().get('party.party')
 
         party_id = self.vcard(uri)
         if party_id is None:
commit 07115e26a373927a63f45c8db3648358c1809a23
Author: Mathias Behrle <mathiasb at m9s.biz>
Date:   Tue May 24 19:12:52 2011 +0200

    Adding upstream version 2.0.0.

diff --git a/CHANGELOG b/CHANGELOG
index 3ecb7d4..154bd85 100644
--- a/CHANGELOG
+++ b/CHANGELOG
@@ -1,3 +1,6 @@
+Version 2.0.0 - 2011-04-27
+* Bug fixes (see mercurial logs for details)
+
 Version 1.8.0 - 2010-11-01
 * Bug fixes (see mercurial logs for details)
 
diff --git a/COPYRIGHT b/COPYRIGHT
index 6e6811c..1245dd5 100644
--- a/COPYRIGHT
+++ b/COPYRIGHT
@@ -1,6 +1,6 @@
-Copyright (C) 2009-2010 Cédric Krier.
-Copyright (C) 2009-2010 Bertrand Chenal.
-Copyright (C) 2009-2010 B2CK SPRL.
+Copyright (C) 2009-2011 Cédric Krier.
+Copyright (C) 2009-2011 Bertrand Chenal.
+Copyright (C) 2009-2011 B2CK SPRL.
 
 This program is free software: you can redistribute it and/or modify
 it under the terms of the GNU General Public License as published by
diff --git a/PKG-INFO b/PKG-INFO
index 46f18bc..d184866 100644
--- a/PKG-INFO
+++ b/PKG-INFO
@@ -1,12 +1,12 @@
 Metadata-Version: 1.0
 Name: trytond_party_vcarddav
-Version: 1.8.0
+Version: 2.0.0
 Summary: Add CardDAV on parties
 Home-page: http://www.tryton.org/
 Author: B2CK
 Author-email: info at b2ck.com
 License: GPL-3
-Download-URL: http://downloads.tryton.org/1.8/
+Download-URL: http://downloads.tryton.org/2.0/
 Description: UNKNOWN
 Platform: UNKNOWN
 Classifier: Development Status :: 5 - Production/Stable
@@ -16,10 +16,13 @@ Classifier: Intended Audience :: Financial and Insurance Industry
 Classifier: Intended Audience :: Legal Industry
 Classifier: Intended Audience :: Manufacturing
 Classifier: License :: OSI Approved :: GNU General Public License (GPL)
+Classifier: Natural Language :: Bulgarian
 Classifier: Natural Language :: English
 Classifier: Natural Language :: French
 Classifier: Natural Language :: German
 Classifier: Natural Language :: Spanish
 Classifier: Operating System :: OS Independent
-Classifier: Programming Language :: Python
+Classifier: Programming Language :: Python :: 2.5
+Classifier: Programming Language :: Python :: 2.6
+Classifier: Programming Language :: Python :: 2.7
 Classifier: Topic :: Office/Business
diff --git a/__tryton__.py b/__tryton__.py
index 1133f31..cdd395b 100644
--- a/__tryton__.py
+++ b/__tryton__.py
@@ -3,15 +3,17 @@
 #this repository contains the full copyright notices and license terms.
 {
     'name' : 'Party - vCardDAV',
+    'name_bg_BG': 'Фирма - vCardDAV',
     'name_de_DE': 'Parteien vCardDAV',
     'name_es_CO' : 'vCardDAV de Compañía',
     'name_es_ES' : 'Tercero - vCardDAV',
     'name_fr_FR': 'Tiers - vCardDAV',
-    'version' : '1.8.0',
+    'version' : '2.0.0',
     'author' : 'B2CK',
     'email': 'info at b2ck.com',
     'website': 'http://www.tryton.org/',
     'description': 'Add CardDAV on parties',
+    'description_bg_BG': 'Добавя към партньор поддръжка на CardDAV',
     'description_de_DE': 'Ermöglicht CardDAV für Parteien',
     'description_es_CO': 'Soporte de CardDAV para terceros',
     'description_es_ES': 'Añade soporte de CardDAV para terceros',
@@ -26,6 +28,7 @@
         'party.xml',
     ],
     'translation': [
+        'bg_BG.csv',
         'de_DE.csv',
         'es_CO.csv',
         'es_ES.csv',
diff --git a/bg_BG.csv b/bg_BG.csv
new file mode 100644
index 0000000..c815f2e
--- /dev/null
+++ b/bg_BG.csv
@@ -0,0 +1,7 @@
+type,name,res_id,src,value,fuzzy
+error,party.party,0,The UUID of the party must be unique!,UUID на партньор трабва да е уникален!,0
+field,"party.party,uuid",0,UUID,UUID,0
+field,"party.party,vcard",0,VCard,VCard,0
+help,"party.party,uuid",0,Universally Unique Identifier,Универсален иникален идентификатор,0
+model,"ir.action,name",report_party_vcard,VCard,VCard,0
+selection,"ir.action.report,extension",0,VCard file,VCard файл,0
diff --git a/setup.py b/setup.py
index e783a55..b4f2f36 100644
--- a/setup.py
+++ b/setup.py
@@ -43,12 +43,15 @@ setup(name='trytond_party_vcarddav',
         'Intended Audience :: Legal Industry',
         'Intended Audience :: Manufacturing',
         'License :: OSI Approved :: GNU General Public License (GPL)',
+        'Natural Language :: Bulgarian',
         'Natural Language :: English',
         'Natural Language :: French',
         'Natural Language :: German',
         'Natural Language :: Spanish',
         'Operating System :: OS Independent',
-        'Programming Language :: Python',
+        'Programming Language :: Python :: 2.5',
+        'Programming Language :: Python :: 2.6',
+        'Programming Language :: Python :: 2.7',
         'Topic :: Office/Business',
     ],
     license='GPL-3',
diff --git a/trytond_party_vcarddav.egg-info/PKG-INFO b/trytond_party_vcarddav.egg-info/PKG-INFO
index bae6d25..3449cd2 100644
--- a/trytond_party_vcarddav.egg-info/PKG-INFO
+++ b/trytond_party_vcarddav.egg-info/PKG-INFO
@@ -1,12 +1,12 @@
 Metadata-Version: 1.0
 Name: trytond-party-vcarddav
-Version: 1.8.0
+Version: 2.0.0
 Summary: Add CardDAV on parties
 Home-page: http://www.tryton.org/
 Author: B2CK
 Author-email: info at b2ck.com
 License: GPL-3
-Download-URL: http://downloads.tryton.org/1.8/
+Download-URL: http://downloads.tryton.org/2.0/
 Description: UNKNOWN
 Platform: UNKNOWN
 Classifier: Development Status :: 5 - Production/Stable
@@ -16,10 +16,13 @@ Classifier: Intended Audience :: Financial and Insurance Industry
 Classifier: Intended Audience :: Legal Industry
 Classifier: Intended Audience :: Manufacturing
 Classifier: License :: OSI Approved :: GNU General Public License (GPL)
+Classifier: Natural Language :: Bulgarian
 Classifier: Natural Language :: English
 Classifier: Natural Language :: French
 Classifier: Natural Language :: German
 Classifier: Natural Language :: Spanish
 Classifier: Operating System :: OS Independent
-Classifier: Programming Language :: Python
+Classifier: Programming Language :: Python :: 2.5
+Classifier: Programming Language :: Python :: 2.6
+Classifier: Programming Language :: Python :: 2.7
 Classifier: Topic :: Office/Business
diff --git a/trytond_party_vcarddav.egg-info/SOURCES.txt b/trytond_party_vcarddav.egg-info/SOURCES.txt
index 8b4f60c..9679a02 100644
--- a/trytond_party_vcarddav.egg-info/SOURCES.txt
+++ b/trytond_party_vcarddav.egg-info/SOURCES.txt
@@ -4,6 +4,7 @@ INSTALL
 LICENSE
 MANIFEST.in
 README
+bg_BG.csv
 de_DE.csv
 es_CO.csv
 es_ES.csv
diff --git a/trytond_party_vcarddav.egg-info/requires.txt b/trytond_party_vcarddav.egg-info/requires.txt
index 9bd5493..b5d5b03 100644
--- a/trytond_party_vcarddav.egg-info/requires.txt
+++ b/trytond_party_vcarddav.egg-info/requires.txt
@@ -1,4 +1,4 @@
 vobject >= 0.8.0
 PyWebDAV >= 0.9.3
-trytond_party >= 1.8, < 1.9
-trytond >= 1.8, < 1.9
\ No newline at end of file
+trytond_party >= 2.0, < 2.1
+trytond >= 2.0, < 2.1
\ No newline at end of file
commit 12e65210f2f9a7d82ea9adbe9699a4f48a058346
Author: Daniel Baumann <daniel at debian.org>
Date:   Thu Nov 4 20:12:24 2010 +0100

    Adding upstream version 1.8.0.

diff --git a/CHANGELOG b/CHANGELOG
index 71af51b..3ecb7d4 100644
--- a/CHANGELOG
+++ b/CHANGELOG
@@ -1,3 +1,6 @@
+Version 1.8.0 - 2010-11-01
+* Bug fixes (see mercurial logs for details)
+
 Version 1.6.0 - 2010-05-11
 * Bug fixes (see mercurial logs for details)
 
diff --git a/PKG-INFO b/PKG-INFO
index 6ff6e0e..46f18bc 100644
--- a/PKG-INFO
+++ b/PKG-INFO
@@ -1,12 +1,12 @@
 Metadata-Version: 1.0
 Name: trytond_party_vcarddav
-Version: 1.6.0
+Version: 1.8.0
 Summary: Add CardDAV on parties
 Home-page: http://www.tryton.org/
 Author: B2CK
 Author-email: info at b2ck.com
 License: GPL-3
-Download-URL: http://downloads.tryton.org/1.6/
+Download-URL: http://downloads.tryton.org/1.8/
 Description: UNKNOWN
 Platform: UNKNOWN
 Classifier: Development Status :: 5 - Production/Stable
diff --git a/__tryton__.py b/__tryton__.py
index 430cfab..1133f31 100644
--- a/__tryton__.py
+++ b/__tryton__.py
@@ -7,7 +7,7 @@
     'name_es_CO' : 'vCardDAV de Compañía',
     'name_es_ES' : 'Tercero - vCardDAV',
     'name_fr_FR': 'Tiers - vCardDAV',
-    'version' : '1.6.0',
+    'version' : '1.8.0',
     'author' : 'B2CK',
     'email': 'info at b2ck.com',
     'website': 'http://www.tryton.org/',
diff --git a/carddav.py b/carddav.py
index 9cf5c18..5bd9863 100644
--- a/carddav.py
+++ b/carddav.py
@@ -1,11 +1,12 @@
 #This file is part of Tryton.  The COPYRIGHT file at the top level of
 #this repository contains the full copyright notices and license terms.
-_TRYTON_RELOAD = False
-
 from DAV import propfind
 from DAV.errors import *
-from trytond.protocols.webdav import TrytonDAVInterface, USER_ID, CACHE, DATABASE
+from trytond.protocols.webdav import TrytonDAVInterface, CACHE
 from trytond.pool import Pool
+from trytond.transaction import Transaction
+
+_TRYTON_RELOAD = False
 
 TrytonDAVInterface.PROPS['urn:ietf:params:xml:ns:carddav'] = (
         'address-data',
@@ -35,18 +36,16 @@ def _get_carddav_address_data(self, uri):
     dbname, dburi = self._get_dburi(uri)
     if not dbname:
         raise DAV_NotFound
-    cursor = DATABASE['cursor']
-    pool = Pool(DATABASE['dbname'])
+    pool = Pool(Transaction().cursor.database_name)
     try:
         collection_obj = pool.get('webdav.collection')
     except KeyError:
         raise DAV_NotFound
     try:
-        res = collection_obj.get_address_data(cursor, int(USER_ID), dburi,
-                cache=CACHE)
+        res = collection_obj.get_address_data(dburi, cache=CACHE)
     except (DAV_Error, DAV_NotFound, DAV_Secret, DAV_Forbidden):
         raise
-    except:
+    except Exception:
         raise DAV_Error(500)
     return res
 
diff --git a/party.py b/party.py
index f01170b..ad90956 100644
--- a/party.py
+++ b/party.py
@@ -1,11 +1,12 @@
 #This file is part of Tryton.  The COPYRIGHT file at the top level of
 #this repository contains the full copyright notices and license terms.
-from trytond.model import ModelSQL, ModelView, fields
-from trytond.report import Report
-from trytond.backend import TableHandler, FIELDS
 import base64
 import copy
 import uuid
+from trytond.model import ModelSQL, ModelView, fields
+from trytond.report import Report
+from trytond.backend import TableHandler, FIELDS
+from trytond.transaction import Transaction
 
 
 class Party(ModelSQL, ModelView):
@@ -16,12 +17,13 @@ class Party(ModelSQL, ModelView):
 
     def __init__(self):
         super(Party, self).__init__()
-        self._sql_constraints = [
+        self._sql_constraints += [
                 ('uuid_uniq', 'UNIQUE(uuid)',
                     'The UUID of the party must be unique!'),
         ]
 
-    def init(self, cursor, module_name):
+    def init(self, module_name):
+        cursor = Transaction().cursor
         table = TableHandler(cursor, self, module_name)
 
         if not table.column_exist('uuid'):
@@ -32,45 +34,42 @@ class Party(ModelSQL, ModelView):
             for id, in cursor.fetchall():
                 cursor.execute('UPDATE "' + self._table + '" ' \
                         'SET "uuid" = %s WHERE id = %s',
-                        (self.default_uuid(cursor, 0), id))
-        super(Party, self).init(cursor, module_name)
+                        (self.default_uuid(), id))
+        super(Party, self).init(module_name)
 
-    def default_uuid(self, cursor, user, context=None):
+    def default_uuid(self):
         return str(uuid.uuid4())
 
-    def create(self, cursor, user, vals, context=None):
+    def create(self, vals):
         collection_obj = self.pool.get('webdav.collection')
 
-        res = super(Party, self).create(cursor, user, vals, context=context)
+        res = super(Party, self).create(vals)
         # Restart the cache for vcard
-        collection_obj.vcard(cursor.dbname)
+        collection_obj.vcard.reset()
         return res
 
-    def write(self, cursor, user, ids, vals, context=None):
+    def write(self, ids, vals):
         collection_obj = self.pool.get('webdav.collection')
 
-        res = super(Party, self).write(cursor, user, ids, vals, context=context)
+        res = super(Party, self).write(ids, vals)
         # Restart the cache for vcard
-        collection_obj.vcard(cursor.dbname)
+        collection_obj.vcard.reset()
         return res
 
-    def delete(self, cursor, user, ids, context=None):
+    def delete(self, ids):
         collection_obj = self.pool.get('webdav.collection')
 
-        res = super(Party, self).delete(cursor, user, ids, context=context)
+        res = super(Party, self).delete(ids)
         # Restart the cache for vcard
-        collection_obj.vcard(cursor.dbname)
+        collection_obj.vcard.reset()
         return res
 
-    def vcard2values(self, cursor, user, party_id, vcard, context=None):
+    def vcard2values(self, party_id, vcard):
         '''
         Convert vcard to values for create or write
 
-        :param cursor: the database cursor
-        :param user: the user id
         :param party_id: the party id for write or None for create
         :param vcard: a vcard instance of vobject
-        :param context: the context
         :return: a dictionary with values
         '''
         import vobject
@@ -87,8 +86,7 @@ class Party(ModelSQL, ModelView):
                 res['uuid'] = vcard.uid.value
             res['addresses'] = []
             for adr in vcard.contents.get('adr', []):
-                vals = address_obj.vcard2values(cursor, user, adr,
-                        context=context)
+                vals = address_obj.vcard2values(adr)
                 res['addresses'].append(('create', vals))
             res['contact_mechanisms'] = []
             for email in vcard.contents.get('email', []):
@@ -105,7 +103,7 @@ class Party(ModelSQL, ModelView):
                 vals['value'] = tel.value
                 res['contact_mechanisms'].append(('create', vals))
         else:
-            party = self.browse(cursor, user, party_id, context=context)
+            party = self.browse(party_id)
             i = 0
             res['addresses'] = []
             addresses_todelete = []
@@ -120,8 +118,7 @@ class Party(ModelSQL, ModelView):
                     addresses_todelete.append(address.id)
                     i += 1
                     continue
-                vals = address_obj.vcard2values(cursor, user, adr,
-                        context=context)
+                vals = address_obj.vcard2values(adr)
                 res['addresses'].append(('write', address.id, vals))
                 i += 1
             if addresses_todelete:
@@ -133,8 +130,7 @@ class Party(ModelSQL, ModelView):
             for adr in new_addresses:
                 if not hasattr(adr, 'value'):
                     continue
-                vals = address_obj.vcard2values(cursor, user, adr,
-                        context=context)
+                vals = address_obj.vcard2values(adr)
                 res['addresses'].append(('create', vals))
 
             i = 0
@@ -205,14 +201,11 @@ Party()
 class Address(ModelSQL, ModelView):
     _name = 'party.address'
 
-    def vcard2values(self, cursor, user, adr, context=None):
+    def vcard2values(self, adr):
         '''
         Convert adr from vcard to values for create or write
 
-        :param cursor: the database cursor
-        :param user: the user id
         :param adr: a adr from vcard instance of vobject
-        :param context: the context
         :return: a dictionary with values
         '''
         country_obj = self.pool.get('country.country')
@@ -223,17 +216,16 @@ class Address(ModelSQL, ModelView):
         vals['city'] = adr.value.city or ''
         vals['zip'] = adr.value.code or ''
         if adr.value.country:
-            country_ids = country_obj.search(cursor, user, [
+            country_ids = country_obj.search([
                 ('rec_name', '=', adr.value.country),
-                ], limit=1, context=context)
+                ], limit=1)
             if country_ids:
                 vals['country'] = country_ids[0]
                 if adr.value.region:
-                    subdivision_ids = subdivision_obj.search(cursor,
-                            user, [
-                                ('rec_name', '=', adr.value.region),
-                                ('country', '=', country_ids[0]),
-                                ], limit=1, context=context)
+                    subdivision_ids = subdivision_obj.search([
+                            ('rec_name', '=', adr.value.region),
+                            ('country', '=', country_ids[0]),
+                            ], limit=1)
                     if subdivision_ids:
                         vals['subdivision'] = subdivision_ids[0]
         return vals
@@ -259,22 +251,18 @@ ActionReport()
 class VCard(Report):
     _name = 'party_vcarddav.party.vcard'
 
-    def execute(self, cursor, user, ids, datas, context=None):
+    def execute(self, ids, datas):
         party_obj = self.pool.get('party.party')
         action_report_obj = self.pool.get('ir.action.report')
 
-        if context is None:
-            context = {}
-
-        action_report_ids = action_report_obj.search(cursor, user, [
+        action_report_ids = action_report_obj.search([
             ('report_name', '=', self._name)
-            ], context=context)
+            ])
         if not action_report_ids:
             raise Exception('Error', 'Report (%s) not find!' % self._name)
-        action_report = action_report_obj.browse(cursor, user,
-                action_report_ids[0], context=context)
+        action_report = action_report_obj.browse(action_report_ids[0])
 
-        parties = party_obj.browse(cursor, user, ids, context=context)
+        parties = party_obj.browse(ids)
 
         data = ''.join(self.create_vcard(party).serialize() for party in parties)
 
diff --git a/trytond_party_vcarddav.egg-info/PKG-INFO b/trytond_party_vcarddav.egg-info/PKG-INFO
index 50416be..bae6d25 100644
--- a/trytond_party_vcarddav.egg-info/PKG-INFO
+++ b/trytond_party_vcarddav.egg-info/PKG-INFO
@@ -1,12 +1,12 @@
 Metadata-Version: 1.0
 Name: trytond-party-vcarddav
-Version: 1.6.0
+Version: 1.8.0
 Summary: Add CardDAV on parties
 Home-page: http://www.tryton.org/
 Author: B2CK
 Author-email: info at b2ck.com
 License: GPL-3
-Download-URL: http://downloads.tryton.org/1.6/
+Download-URL: http://downloads.tryton.org/1.8/
 Description: UNKNOWN
 Platform: UNKNOWN
 Classifier: Development Status :: 5 - Production/Stable
diff --git a/trytond_party_vcarddav.egg-info/requires.txt b/trytond_party_vcarddav.egg-info/requires.txt
index c8729a4..9bd5493 100644
--- a/trytond_party_vcarddav.egg-info/requires.txt
+++ b/trytond_party_vcarddav.egg-info/requires.txt
@@ -1,4 +1,4 @@
 vobject >= 0.8.0
 PyWebDAV >= 0.9.3
-trytond_party >= 1.6, < 1.7
-trytond >= 1.6, < 1.7
\ No newline at end of file
+trytond_party >= 1.8, < 1.9
+trytond >= 1.8, < 1.9
\ No newline at end of file
diff --git a/webdav.py b/webdav.py
index e16f5ec..ec35b22 100644
--- a/webdav.py
+++ b/webdav.py
@@ -1,10 +1,13 @@
 #This file is part of Tryton.  The COPYRIGHT file at the top level of
 #this repository contains the full copyright notices and license terms.
-from trytond.model import ModelView, ModelSQL
-from trytond.tools import Cache, reduce_ids
-from DAV.errors import DAV_NotFound, DAV_Forbidden
 import base64
 import urlparse
+from DAV.errors import DAV_NotFound, DAV_Forbidden
+from trytond.model import ModelView, ModelSQL
+from trytond.tools import reduce_ids
+from trytond.transaction import Transaction
+from trytond.cache import Cache
+
 
 CARDDAV_NS = 'urn:ietf:params:xml:ns:carddav'
 
@@ -14,14 +17,11 @@ class Collection(ModelSQL, ModelView):
     _name = "webdav.collection"
 
     @Cache('webdav_collection.vcard')
-    def vcard(self, cursor, user, uri, context=None):
+    def vcard(self, uri):
         '''
         Return party ids of the vcard in uri or False
 
-        :param cursor: the database cursor
-        :param user: the user id
         :param uri: the uri
-        :param context: the context
         :return: party id
             or None if there is no party
             or False if not in Contacts
@@ -30,9 +30,9 @@ class Collection(ModelSQL, ModelView):
 
         if uri and uri.startswith('Contacts/'):
             uuid = uri[9:-4]
-            party_ids = party_obj.search(cursor, user, [
+            party_ids = party_obj.search([
                 ('uuid', '=', uuid),
-                ], limit=1, context=context)
+                ], limit=1)
             if party_ids:
                 return party_ids[0]
             return None
@@ -40,15 +40,11 @@ class Collection(ModelSQL, ModelView):
             return None
         return False
 
-    def _carddav_filter_domain(self, cursor, user, filter, context=None):
+    def _carddav_filter_domain(self, filter):
         '''
         Return a domain for the carddav filter
 
-        :param cursor: the database cursor
-        :param user: the user id
         :param filter: the DOM Element of filter
-        :param context: the context
-
         :return: a list for domain
         '''
         address_obj = self.pool.get('party.address')
@@ -113,8 +109,7 @@ class Collection(ModelSQL, ModelView):
                             res2.append((field, 'not ilike', value))
                     if name == 'adr':
                         domain = res2
-                        address_ids = address_obj.search(cursor, user, domain,
-                                context=context)
+                        address_ids = address_obj.search(domain)
                         res = [('addresses', 'in', address_ids)]
                     elif name in ('mail', 'tel'):
                         if name == 'mail':
@@ -122,54 +117,55 @@ class Collection(ModelSQL, ModelView):
                         else:
                             type = ['phone', 'mobile']
                         domain = [('type', 'in', type), res2]
-                        contact_mechanism_ids = contact_mechanism_obj.search(cursor,
-                                user, domain, context=context)
-                        res2 = [('contact_mechanisms', 'in', contact_mechanism_ids)]
+                        contact_mechanism_ids = contact_mechanism_obj.search(
+                                domain)
+                        res2 = [
+                            ('contact_mechanisms', 'in', contact_mechanism_ids)
+                            ]
                     res.append(res2)
         return res
 
-    def get_childs(self, cursor, user, uri, filter=None, context=None,
-            cache=None):
+    def get_childs(self, uri, filter=None, cache=None):
         party_obj = self.pool.get('party.party')
 
         if uri in ('Contacts', 'Contacts/'):
-            domain = self._carddav_filter_domain(cursor, user, filter,
-                    context=context)
-            party_ids = party_obj.search(cursor, user, domain, context=context)
-            parties = party_obj.browse(cursor, user, party_ids, context=context)
+            domain = self._carddav_filter_domain(filter)
+            party_ids = party_obj.search(domain)
+            parties = party_obj.browse(party_ids)
             if cache is not None:
                 cache.setdefault('_contact', {})
                 for party_id in party_ids:
                     cache['_contact'][party_id] = {}
             return [x.uuid + '.vcf' for x in parties]
-        party_id = self.vcard(cursor, user, uri, context=context)
+        party_id = self.vcard(uri)
         if party_id or party_id is None:
             return []
-        res = super(Collection, self).get_childs(cursor, user, uri,
-                filter=filter, context=context, cache=cache)
+        res = super(Collection, self).get_childs(uri, filter=filter,
+                cache=cache)
         if not uri and not filter:
             res.append('Contacts')
         return res
 
-    def get_resourcetype(self, cursor, user, uri, context=None, cache=None):
+    def get_resourcetype(self, uri, cache=None):
         from DAV.constants import COLLECTION, OBJECT
-        party_id = self.vcard(cursor, user, uri, context=context)
+        party_id = self.vcard(uri)
         if party_id:
             return OBJECT
         elif party_id is None:
             return COLLECTION
-        return super(Collection, self).get_resourcetype(cursor, user, uri,
-                context=context, cache=cache)
+        return super(Collection, self).get_resourcetype(uri, cache=cache)
 
-    def get_contenttype(self, cursor, user, uri, context=None, cache=None):
-        if self.vcard(cursor, user, uri, context=context):
+    def get_contenttype(self, uri, cache=None):
+        if self.vcard(uri):
             return 'text/x-vcard'
-        return super(Collection, self).get_contenttype(cursor, user, uri,
-                context=context, cache=cache)
+        return super(Collection, self).get_contenttype(uri, cache=cache)
 
-    def get_creationdate(self, cursor, user, uri, context=None, cache=None):
+    def get_creationdate(self, uri, cache=None):
         party_obj = self.pool.get('party.party')
-        party_id = self.vcard(cursor, user, uri, context=context)
+        party_id = self.vcard(uri)
+
+        cursor = Transaction().cursor
+
         if party_id is None:
             raise DAV_NotFound
         if party_id:
@@ -198,15 +194,16 @@ class Collection(ModelSQL, ModelView):
                         cache['_contact'][party_id2]['creationdate'] = date
             if res is not None:
                 return res
-        return super(Collection, self).get_creationdate(cursor, user, uri,
-                context=context, cache=cache)
+        return super(Collection, self).get_creationdate(uri, cache=cache)
 
-    def get_lastmodified(self, cursor, user, uri, context=None, cache=None):
+    def get_lastmodified(self, uri, cache=None):
         party_obj = self.pool.get('party.party')
         address_obj = self.pool.get('party.address')
         contact_mechanism_obj = self.pool.get('party.contact_mechanism')
 
-        party_id = self.vcard(cursor, user, uri, context=context)
+        cursor = Transaction().cursor
+
+        party_id = self.vcard(uri)
         if party_id:
             if cache is not None:
                 cache.setdefault('_contact', {})
@@ -244,104 +241,94 @@ class Collection(ModelSQL, ModelView):
                         cache['_contact'][party_id2]['lastmodified'] = date
             if res is not None:
                 return res
-        return super(Collection, self).get_lastmodified(cursor, user, uri,
-                context=context, cache=cache)
+        return super(Collection, self).get_lastmodified(uri, cache=cache)
 
-    def get_data(self, cursor, user, uri, context=None, cache=None):
+    def get_data(self, uri, cache=None):
         vcard_obj = self.pool.get('party_vcarddav.party.vcard', type='report')
-        party_id = self.vcard(cursor, user, uri, context=context)
+        party_id = self.vcard(uri)
         if party_id is None:
             raise DAV_NotFound
         if party_id:
-            val = vcard_obj.execute(cursor, user, [party_id],
+            val = vcard_obj.execute([party_id],
                     {'id': party_id, 'ids': [party_id]},
-                    context=context)
+                    )
             return base64.decodestring(val[1])
-        return super(Collection, self).get_data(cursor, user, uri,
-                context=context, cache=cache)
+        return super(Collection, self).get_data(uri, cache=cache)
 
-    def get_address_data(self, cursor, user, uri, context=None, cache=None):
+    def get_address_data(self, uri, cache=None):
         vcard_obj = self.pool.get('party_vcarddav.party.vcard', type='report')
-        party_id = self.vcard(cursor, user, uri, context=context)
+        party_id = self.vcard(uri)
         if not party_id:
             raise DAV_NotFound
-        val = vcard_obj.execute(cursor, user, [party_id],
+        val = vcard_obj.execute([party_id],
                 {'id': party_id, 'ids': [party_id]},
-                context=context)
+                )
         res = base64.decodestring(val[1])
         return res.decode('utf-8')
 
-    def put(self, cursor, user, uri, data, content_type, context=None,
-            cache=None):
+    def put(self, uri, data, content_type, cache=None):
         import vobject
         party_obj = self.pool.get('party.party')
 
-        party_id = self.vcard(cursor, user, uri, context=context)
+        party_id = self.vcard(uri)
         if party_id is None:
             vcard = vobject.readOne(data)
-            values = party_obj.vcard2values(cursor, user, None, vcard,
-                    context=context)
+            values = party_obj.vcard2values(None, vcard)
             try:
-                party_id = party_obj.create(cursor, user, values,
-                        context=context)
-            except:
+                party_id = party_obj.create(values)
+            except Exception:
                 raise DAV_Forbidden
-            party = party_obj.browse(cursor, user, party_id, context=context)
-            return cursor.database_name + '/Contacts/' + party.uuid + '.vcf'
+            party = party_obj.browse(party_id)
+            return (Transaction().cursor.database_name + '/Contacts/' +
+                    party.uuid + '.vcf')
         if party_id:
             vcard = vobject.readOne(data)
-            values = party_obj.vcard2values(cursor, user, party_id, vcard,
-                    context=context)
+            values = party_obj.vcard2values(party_id, vcard)
             try:
-                party_obj.write(cursor, user, party_id, values, context=context)
-            except:
+                party_obj.write(party_id, values)
+            except Exception:
                 raise DAV_Forbidden
             return
-        return super(Collection, self).put(cursor, user, uri, data,
-                content_type, context=context)
+        return super(Collection, self).put(uri, data, content_type, cache=cache)
 
-    def mkcol(self, cursor, user, uri, context=None, cache=None):
-        party_id = self.vcard(cursor, user, uri, context=context)
+    def mkcol(self, uri, cache=None):
+        party_id = self.vcard(uri)
         if party_id is None:
             raise DAV_Forbidden
         if party_id:
             raise DAV_Forbidden
-        return super(Collection, self).mkcol(cursor, user, uri, context=context,
-                cache=cache)
+        return super(Collection, self).mkcol(uri, cache=cache)
 
-    def rmcol(self, cursor, user, uri, context=None, cache=None):
-        party_id = self.vcard(cursor, user, uri, context=context)
+    def rmcol(self, uri, cache=None):
+        party_id = self.vcard(uri)
         if party_id is None:
             raise DAV_Forbidden
         if party_id:
             raise DAV_Forbidden
-        return super(Collection, self).rmcol(cursor, user, uri, context=context,
-                cache=cache)
+        return super(Collection, self).rmcol(uri, cache=cache)
 
-    def rm(self, cursor, user, uri, context=None, cache=None):
+    def rm(self, uri, cache=None):
         party_obj = self.pool.get('party.party')
 
-        party_id = self.vcard(cursor, user, uri, context=context)
+        party_id = self.vcard(uri)
         if party_id is None:
             raise DAV_Forbidden
         if party_id:
             try:
-                party_obj.delete(cursor, user, party_id, context=context)
-            except:
+                party_obj.delete(party_id)
+            except Exception:
                 raise DAV_Forbidden
             return 200
-        return super(Collection, self).rm(cursor, user, uri, context=context,
-                cache=cache)
+        return super(Collection, self).rm(uri, cache=cache)
 
-    def exists(self, cursor, user, uri, context=None, cache=None):
-        party_id = self.vcard(cursor, user, uri, context=context)
+    def exists(self, uri, cache=None):
+        party_id = self.vcard(uri)
         if party_id is None or party_id:
             if party_id:
                 return 1
             if uri in ('Contacts', 'Contacts/'):
                 return 1
             return 0
-        return super(Collection, self).exists(cursor, user, uri, context=context,
-                cache=cache)
+        return super(Collection, self).exists(uri, cache=cache)
 
 Collection()
commit 2af0dd06a0b47c4346c89c05a2eb2ee821a5d266
Author: Mathias Behrle <mathiasb at mbsolutions.selfip.biz>
Date:   Wed May 12 17:04:05 2010 +0200

    Adding upstream version 1.6.0.

diff --git a/CHANGELOG b/CHANGELOG
index 8ce1b25..71af51b 100644
--- a/CHANGELOG
+++ b/CHANGELOG
@@ -1,5 +1,5 @@
-Version 1.4.1 - 2009-11-24
-* Some bug fixes (see mercurial logs for details)
+Version 1.6.0 - 2010-05-11
+* Bug fixes (see mercurial logs for details)
 
 Version 1.4.0 - 2009-10-19
 * Initial release
diff --git a/COPYRIGHT b/COPYRIGHT
index 3a474ed..6e6811c 100644
--- a/COPYRIGHT
+++ b/COPYRIGHT
@@ -1,6 +1,6 @@
-Copyright (C) 2009 Cédric Krier.
-Copyright (C) 2009 Bertrand Chenal.
-Copyright (C) 2009 B2CK SPRL.
+Copyright (C) 2009-2010 Cédric Krier.
+Copyright (C) 2009-2010 Bertrand Chenal.
+Copyright (C) 2009-2010 B2CK SPRL.
 
 This program is free software: you can redistribute it and/or modify
 it under the terms of the GNU General Public License as published by
diff --git a/MANIFEST.in b/MANIFEST.in
index b2a140e..dcb2afa 100644
--- a/MANIFEST.in
+++ b/MANIFEST.in
@@ -7,5 +7,4 @@ include LICENSE
 include *.xml
 include *.odt
 include *.csv
-include tests/*
 include doc/*
diff --git a/PKG-INFO b/PKG-INFO
index 1055a36..6ff6e0e 100644
--- a/PKG-INFO
+++ b/PKG-INFO
@@ -1,12 +1,12 @@
 Metadata-Version: 1.0
 Name: trytond_party_vcarddav
-Version: 1.4.1
+Version: 1.6.0
 Summary: Add CardDAV on parties
 Home-page: http://www.tryton.org/
 Author: B2CK
 Author-email: info at b2ck.com
 License: GPL-3
-Download-URL: http://downloads.tryton.org/1.4/
+Download-URL: http://downloads.tryton.org/1.6/
 Description: UNKNOWN
 Platform: UNKNOWN
 Classifier: Development Status :: 5 - Production/Stable
diff --git a/__tryton__.py b/__tryton__.py
index 0e3e43f..430cfab 100644
--- a/__tryton__.py
+++ b/__tryton__.py
@@ -7,7 +7,7 @@
     'name_es_CO' : 'vCardDAV de Compañía',
     'name_es_ES' : 'Tercero - vCardDAV',
     'name_fr_FR': 'Tiers - vCardDAV',
-    'version' : '1.4.1',
+    'version' : '1.6.0',
     'author' : 'B2CK',
     'email': 'info at b2ck.com',
     'website': 'http://www.tryton.org/',
diff --git a/fr_FR.csv b/fr_FR.csv
index 466babf..6845ba2 100644
--- a/fr_FR.csv
+++ b/fr_FR.csv
@@ -1,7 +1,7 @@
 type,name,res_id,src,value,fuzzy
-error,party.party,0,The UUID of the party must be unique!,,0
-field,"party.party,uuid",0,UUID,,0
-field,"party.party,vcard",0,VCard,,0
-help,"party.party,uuid",0,Universally Unique Identifier,,0
-model,"ir.action,name",report_party_vcard,VCard,,0
-selection,"ir.action.report,extension",0,VCard file,,0
+error,party.party,0,The UUID of the party must be unique!,L'UUID du tiers doit être unique !,0
+field,"party.party,uuid",0,UUID,UUID,0
+field,"party.party,vcard",0,VCard,VCard,0
+help,"party.party,uuid",0,Universally Unique Identifier,Identificateur unique universel,0
+model,"ir.action,name",report_party_vcard,VCard,VCard,0
+selection,"ir.action.report,extension",0,VCard file,Fichier VCard,0
diff --git a/setup.py b/setup.py
index 784f6e9..e783a55 100644
--- a/setup.py
+++ b/setup.py
@@ -2,19 +2,22 @@
 #This file is part of Tryton.  The COPYRIGHT file at the top level of
 #this repository contains the full copyright notices and license terms.
 
-from setuptools import setup, find_packages
+from setuptools import setup
 import re
 
-info = eval(file('__tryton__.py').read())
+info = eval(open('__tryton__.py').read())
+major_version, minor_version, _ = info.get('version', '0.0.1').split('.', 2)
+major_version = int(major_version)
+minor_version = int(minor_version)
 
 requires = ['vobject >= 0.8.0', 'PyWebDAV >= 0.9.3']
 for dep in info.get('depends', []):
     if not re.match(r'(ir|res|workflow|webdav)(\W|$)', dep):
-        requires.append('trytond_' + dep)
-
-major_version, minor_version, _ = info.get('version', '0.0.1').split('.', 2)
-requires.append('trytond >= %s.%s' % (major_version, minor_version))
-requires.append('trytond < %s.%s' % (major_version, int(minor_version) + 1))
+        requires.append('trytond_%s >= %s.%s, < %s.%s' %
+                (dep, major_version, minor_version, major_version,
+                    minor_version + 1))
+requires.append('trytond >= %s.%s, < %s.%s' %
+        (major_version, minor_version, major_version, minor_version + 1))
 
 setup(name='trytond_party_vcarddav',
     version=info.get('version', '0.0.1'),
diff --git a/trytond_party_vcarddav.egg-info/PKG-INFO b/trytond_party_vcarddav.egg-info/PKG-INFO
index cb6ea2c..50416be 100644
--- a/trytond_party_vcarddav.egg-info/PKG-INFO
+++ b/trytond_party_vcarddav.egg-info/PKG-INFO
@@ -1,12 +1,12 @@
 Metadata-Version: 1.0
 Name: trytond-party-vcarddav
-Version: 1.4.1
+Version: 1.6.0
 Summary: Add CardDAV on parties
 Home-page: http://www.tryton.org/
 Author: B2CK
 Author-email: info at b2ck.com
 License: GPL-3
-Download-URL: http://downloads.tryton.org/1.4/
+Download-URL: http://downloads.tryton.org/1.6/
 Description: UNKNOWN
 Platform: UNKNOWN
 Classifier: Development Status :: 5 - Production/Stable
diff --git a/trytond_party_vcarddav.egg-info/requires.txt b/trytond_party_vcarddav.egg-info/requires.txt
index fcef744..c8729a4 100644
--- a/trytond_party_vcarddav.egg-info/requires.txt
+++ b/trytond_party_vcarddav.egg-info/requires.txt
@@ -1,5 +1,4 @@
 vobject >= 0.8.0
 PyWebDAV >= 0.9.3
-trytond_party
-trytond >= 1.4
-trytond < 1.5
\ No newline at end of file
+trytond_party >= 1.6, < 1.7
+trytond >= 1.6, < 1.7
\ No newline at end of file
diff --git a/webdav.py b/webdav.py
index df760df..e16f5ec 100644
--- a/webdav.py
+++ b/webdav.py
@@ -13,6 +13,7 @@ class Collection(ModelSQL, ModelView):
 
     _name = "webdav.collection"
 
+    @Cache('webdav_collection.vcard')
     def vcard(self, cursor, user, uri, context=None):
         '''
         Return party ids of the vcard in uri or False
@@ -39,8 +40,6 @@ class Collection(ModelSQL, ModelView):
             return None
         return False
 
-    vcard = Cache('webdav_collection.vcard')(vcard)
-
     def _carddav_filter_domain(self, cursor, user, filter, context=None):
         '''
         Return a domain for the carddav filter
commit 4ec3ff1173a4734573a5323df3d79209885d1803
Author: Daniel Baumann <daniel at debian.org>
Date:   Wed Nov 25 13:10:13 2009 +0100

    Adding upstream version 1.4.1.

diff --git a/CHANGELOG b/CHANGELOG
index 931601e..8ce1b25 100644
--- a/CHANGELOG
+++ b/CHANGELOG
@@ -1,2 +1,5 @@
+Version 1.4.1 - 2009-11-24
+* Some bug fixes (see mercurial logs for details)
+
 Version 1.4.0 - 2009-10-19
 * Initial release
diff --git a/INSTALL b/INSTALL
index 05cadc4..6db776c 100644
--- a/INSTALL
+++ b/INSTALL
@@ -7,7 +7,7 @@ Prerequisites
  * Python 2.5 or later (http://www.python.org/)
  * trytond (http://www.tryton.org/)
  * trytond_party (http://www.tryton.org/)
- * vobject (http://vobject.skyhouseconsulting.com/)
+ * vobject 0.8.0 or later (http://vobject.skyhouseconsulting.com/)
  * pywebdav 0.9.3 or later (http://sourceforge.net/projects/pywebdav/)
 
 Installation
diff --git a/PKG-INFO b/PKG-INFO
index 2e5b2d4..1055a36 100644
--- a/PKG-INFO
+++ b/PKG-INFO
@@ -1,6 +1,6 @@
 Metadata-Version: 1.0
 Name: trytond_party_vcarddav
-Version: 1.4.0
+Version: 1.4.1
 Summary: Add CardDAV on parties
 Home-page: http://www.tryton.org/
 Author: B2CK
diff --git a/__tryton__.py b/__tryton__.py
index d479a81..0e3e43f 100644
--- a/__tryton__.py
+++ b/__tryton__.py
@@ -7,7 +7,7 @@
     'name_es_CO' : 'vCardDAV de Compañía',
     'name_es_ES' : 'Tercero - vCardDAV',
     'name_fr_FR': 'Tiers - vCardDAV',
-    'version' : '1.4.0',
+    'version' : '1.4.1',
     'author' : 'B2CK',
     'email': 'info at b2ck.com',
     'website': 'http://www.tryton.org/',
diff --git a/carddav.py b/carddav.py
index bd1eb02..9cf5c18 100644
--- a/carddav.py
+++ b/carddav.py
@@ -33,9 +33,14 @@ propfind.PROPFIND.mk_prop_response = mk_prop_response
 
 def _get_carddav_address_data(self, uri):
     dbname, dburi = self._get_dburi(uri)
+    if not dbname:
+        raise DAV_NotFound
     cursor = DATABASE['cursor']
     pool = Pool(DATABASE['dbname'])
-    collection_obj = pool.get('webdav.collection')
+    try:
+        collection_obj = pool.get('webdav.collection')
+    except KeyError:
+        raise DAV_NotFound
     try:
         res = collection_obj.get_address_data(cursor, int(USER_ID), dburi,
                 cache=CACHE)
diff --git a/setup.py b/setup.py
index e4d9f50..784f6e9 100644
--- a/setup.py
+++ b/setup.py
@@ -7,7 +7,7 @@ import re
 
 info = eval(file('__tryton__.py').read())
 
-requires = ['vobject', 'PyWebDAV >= 0.9.3']
+requires = ['vobject >= 0.8.0', 'PyWebDAV >= 0.9.3']
 for dep in info.get('depends', []):
     if not re.match(r'(ir|res|workflow|webdav)(\W|$)', dep):
         requires.append('trytond_' + dep)
diff --git a/trytond_party_vcarddav.egg-info/PKG-INFO b/trytond_party_vcarddav.egg-info/PKG-INFO
index 4140bda..cb6ea2c 100644
--- a/trytond_party_vcarddav.egg-info/PKG-INFO
+++ b/trytond_party_vcarddav.egg-info/PKG-INFO
@@ -1,6 +1,6 @@
 Metadata-Version: 1.0
 Name: trytond-party-vcarddav
-Version: 1.4.0
+Version: 1.4.1
 Summary: Add CardDAV on parties
 Home-page: http://www.tryton.org/
 Author: B2CK
diff --git a/trytond_party_vcarddav.egg-info/requires.txt b/trytond_party_vcarddav.egg-info/requires.txt
index 8550012..fcef744 100644
--- a/trytond_party_vcarddav.egg-info/requires.txt
+++ b/trytond_party_vcarddav.egg-info/requires.txt
@@ -1,4 +1,4 @@
-vobject
+vobject >= 0.8.0
 PyWebDAV >= 0.9.3
 trytond_party
 trytond >= 1.4
commit 2599d44a9c732928ae740b1a6c2240f4f20f2779
Author: Daniel Baumann <daniel at debian.org>
Date:   Mon Oct 19 22:54:51 2009 +0200

    Adding upstream version 1.4.0.

diff --git a/CHANGELOG b/CHANGELOG
new file mode 100644
index 0000000..931601e
--- /dev/null
+++ b/CHANGELOG
@@ -0,0 +1,2 @@
+Version 1.4.0 - 2009-10-19
+* Initial release
diff --git a/COPYRIGHT b/COPYRIGHT
new file mode 100644
index 0000000..3a474ed
--- /dev/null
+++ b/COPYRIGHT
@@ -0,0 +1,16 @@
+Copyright (C) 2009 Cédric Krier.
+Copyright (C) 2009 Bertrand Chenal.
+Copyright (C) 2009 B2CK SPRL.
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+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 General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program.  If not, see <http://www.gnu.org/licenses/>.
diff --git a/INSTALL b/INSTALL
new file mode 100644
index 0000000..05cadc4
--- /dev/null
+++ b/INSTALL
@@ -0,0 +1,32 @@
+Installing trytond_party_vcarddav
+=================================
+
+Prerequisites
+-------------
+
+ * Python 2.5 or later (http://www.python.org/)
+ * trytond (http://www.tryton.org/)
+ * trytond_party (http://www.tryton.org/)
+ * vobject (http://vobject.skyhouseconsulting.com/)
+ * pywebdav 0.9.3 or later (http://sourceforge.net/projects/pywebdav/)
+
+Installation
+------------
+
+Once you've downloaded and unpacked the trytond_party_vcarddav source
+release, enter the directory where the archive was unpacked, and run:
+
+    python setup.py install
+
+Note that you may need administrator/root privileges for this step, as
+this command will by default attempt to install module to the Python
+site-packages directory on your system.
+
+For advanced options, please refer to the easy_install and/or the distutils
+documentation:
+
+  http://peak.telecommunity.com/DevCenter/EasyInstall
+  http://docs.python.org/inst/inst.html
+
+To use without installation, extract the archive into ``trytond/modules`` with
+the directory name party_vcarddav.
diff --git a/LICENSE b/LICENSE
new file mode 100644
index 0000000..94a9ed0
--- /dev/null
+++ b/LICENSE
@@ -0,0 +1,674 @@
+                    GNU GENERAL PUBLIC LICENSE
+                       Version 3, 29 June 2007
+
+ Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/>
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+                            Preamble
+
+  The GNU General Public License is a free, copyleft license for
+software and other kinds of works.
+
+  The licenses for most software and other practical works are designed
+to take away your freedom to share and change the works.  By contrast,
+the GNU General Public License is intended to guarantee your freedom to
+share and change all versions of a program--to make sure it remains free
+software for all its users.  We, the Free Software Foundation, use the
+GNU General Public License for most of our software; it applies also to
+any other work released this way by its authors.  You can apply it to
+your programs, too.
+
+  When we speak of free software, we are referring to freedom, not
+price.  Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+them if you wish), that you receive source code or can get it if you
+want it, that you can change the software or use pieces of it in new
+free programs, and that you know you can do these things.
+
+  To protect your rights, we need to prevent others from denying you
+these rights or asking you to surrender the rights.  Therefore, you have
+certain responsibilities if you distribute copies of the software, or if
+you modify it: responsibilities to respect the freedom of others.
+
+  For example, if you distribute copies of such a program, whether
+gratis or for a fee, you must pass on to the recipients the same
+freedoms that you received.  You must make sure that they, too, receive
+or can get the source code.  And you must show them these terms so they
+know their rights.
+
+  Developers that use the GNU GPL protect your rights with two steps:
+(1) assert copyright on the software, and (2) offer you this License
+giving you legal permission to copy, distribute and/or modify it.
+
+  For the developers' and authors' protection, the GPL clearly explains
+that there is no warranty for this free software.  For both users' and
+authors' sake, the GPL requires that modified versions be marked as
+changed, so that their problems will not be attributed erroneously to
+authors of previous versions.
+
+  Some devices are designed to deny users access to install or run
+modified versions of the software inside them, although the manufacturer
+can do so.  This is fundamentally incompatible with the aim of
+protecting users' freedom to change the software.  The systematic
+pattern of such abuse occurs in the area of products for individuals to
+use, which is precisely where it is most unacceptable.  Therefore, we
+have designed this version of the GPL to prohibit the practice for those
+products.  If such problems arise substantially in other domains, we
+stand ready to extend this provision to those domains in future versions
+of the GPL, as needed to protect the freedom of users.
+
+  Finally, every program is threatened constantly by software patents.
+States should not allow patents to restrict development and use of
+software on general-purpose computers, but in those that do, we wish to
+avoid the special danger that patents applied to a free program could
+make it effectively proprietary.  To prevent this, the GPL assures that
+patents cannot be used to render the program non-free.
+
+  The precise terms and conditions for copying, distribution and
+modification follow.
+
+                       TERMS AND CONDITIONS
+
+  0. Definitions.
+
+  "This License" refers to version 3 of the GNU General Public License.
+
+  "Copyright" also means copyright-like laws that apply to other kinds of
+works, such as semiconductor masks.
+
+  "The Program" refers to any copyrightable work licensed under this
+License.  Each licensee is addressed as "you".  "Licensees" and
+"recipients" may be individuals or organizations.
+
+  To "modify" a work means to copy from or adapt all or part of the work
+in a fashion requiring copyright permission, other than the making of an
+exact copy.  The resulting work is called a "modified version" of the
+earlier work or a work "based on" the earlier work.
+
+  A "covered work" means either the unmodified Program or a work based
+on the Program.
+
+  To "propagate" a work means to do anything with it that, without
+permission, would make you directly or secondarily liable for
+infringement under applicable copyright law, except executing it on a
+computer or modifying a private copy.  Propagation includes copying,
+distribution (with or without modification), making available to the
+public, and in some countries other activities as well.
+
+  To "convey" a work means any kind of propagation that enables other
+parties to make or receive copies.  Mere interaction with a user through
+a computer network, with no transfer of a copy, is not conveying.
+
+  An interactive user interface displays "Appropriate Legal Notices"
+to the extent that it includes a convenient and prominently visible
+feature that (1) displays an appropriate copyright notice, and (2)
+tells the user that there is no warranty for the work (except to the
+extent that warranties are provided), that licensees may convey the
+work under this License, and how to view a copy of this License.  If
+the interface presents a list of user commands or options, such as a
+menu, a prominent item in the list meets this criterion.
+
+  1. Source Code.
+
+  The "source code" for a work means the preferred form of the work
+for making modifications to it.  "Object code" means any non-source
+form of a work.
+
+  A "Standard Interface" means an interface that either is an official
+standard defined by a recognized standards body, or, in the case of
+interfaces specified for a particular programming language, one that
+is widely used among developers working in that language.
+
+  The "System Libraries" of an executable work include anything, other
+than the work as a whole, that (a) is included in the normal form of
+packaging a Major Component, but which is not part of that Major
+Component, and (b) serves only to enable use of the work with that
+Major Component, or to implement a Standard Interface for which an
+implementation is available to the public in source code form.  A
+"Major Component", in this context, means a major essential component
+(kernel, window system, and so on) of the specific operating system
+(if any) on which the executable work runs, or a compiler used to
+produce the work, or an object code interpreter used to run it.
+
+  The "Corresponding Source" for a work in object code form means all
+the source code needed to generate, install, and (for an executable
+work) run the object code and to modify the work, including scripts to
+control those activities.  However, it does not include the work's
+System Libraries, or general-purpose tools or generally available free
+programs which are used unmodified in performing those activities but
+which are not part of the work.  For example, Corresponding Source
+includes interface definition files associated with source files for
+the work, and the source code for shared libraries and dynamically
+linked subprograms that the work is specifically designed to require,
+such as by intimate data communication or control flow between those
+subprograms and other parts of the work.
+
+  The Corresponding Source need not include anything that users
+can regenerate automatically from other parts of the Corresponding
+Source.
+
+  The Corresponding Source for a work in source code form is that
+same work.
+
+  2. Basic Permissions.
+
+  All rights granted under this License are granted for the term of
+copyright on the Program, and are irrevocable provided the stated
+conditions are met.  This License explicitly affirms your unlimited
+permission to run the unmodified Program.  The output from running a
+covered work is covered by this License only if the output, given its
+content, constitutes a covered work.  This License acknowledges your
+rights of fair use or other equivalent, as provided by copyright law.
+
+  You may make, run and propagate covered works that you do not
+convey, without conditions so long as your license otherwise remains
+in force.  You may convey covered works to others for the sole purpose
+of having them make modifications exclusively for you, or provide you
+with facilities for running those works, provided that you comply with
+the terms of this License in conveying all material for which you do
+not control copyright.  Those thus making or running the covered works
+for you must do so exclusively on your behalf, under your direction
+and control, on terms that prohibit them from making any copies of
+your copyrighted material outside their relationship with you.
+
+  Conveying under any other circumstances is permitted solely under
+the conditions stated below.  Sublicensing is not allowed; section 10
+makes it unnecessary.
+
+  3. Protecting Users' Legal Rights From Anti-Circumvention Law.
+
+  No covered work shall be deemed part of an effective technological
+measure under any applicable law fulfilling obligations under article
+11 of the WIPO copyright treaty adopted on 20 December 1996, or
+similar laws prohibiting or restricting circumvention of such
+measures.
+
+  When you convey a covered work, you waive any legal power to forbid
+circumvention of technological measures to the extent such circumvention
+is effected by exercising rights under this License with respect to
+the covered work, and you disclaim any intention to limit operation or
+modification of the work as a means of enforcing, against the work's
+users, your or third parties' legal rights to forbid circumvention of
+technological measures.
+
+  4. Conveying Verbatim Copies.
+
+  You may convey verbatim copies of the Program's source code as you
+receive it, in any medium, provided that you conspicuously and
+appropriately publish on each copy an appropriate copyright notice;
+keep intact all notices stating that this License and any
+non-permissive terms added in accord with section 7 apply to the code;
+keep intact all notices of the absence of any warranty; and give all
+recipients a copy of this License along with the Program.
+
+  You may charge any price or no price for each copy that you convey,
+and you may offer support or warranty protection for a fee.
+
+  5. Conveying Modified Source Versions.
+
+  You may convey a work based on the Program, or the modifications to
+produce it from the Program, in the form of source code under the
+terms of section 4, provided that you also meet all of these conditions:
+
+    a) The work must carry prominent notices stating that you modified
+    it, and giving a relevant date.
+
+    b) The work must carry prominent notices stating that it is
+    released under this License and any conditions added under section
+    7.  This requirement modifies the requirement in section 4 to
+    "keep intact all notices".
+
+    c) You must license the entire work, as a whole, under this
+    License to anyone who comes into possession of a copy.  This
+    License will therefore apply, along with any applicable section 7
+    additional terms, to the whole of the work, and all its parts,
+    regardless of how they are packaged.  This License gives no
+    permission to license the work in any other way, but it does not
+    invalidate such permission if you have separately received it.
+
+    d) If the work has interactive user interfaces, each must display
+    Appropriate Legal Notices; however, if the Program has interactive
+    interfaces that do not display Appropriate Legal Notices, your
+    work need not make them do so.
+
+  A compilation of a covered work with other separate and independent
+works, which are not by their nature extensions of the covered work,
+and which are not combined with it such as to form a larger program,
+in or on a volume of a storage or distribution medium, is called an
+"aggregate" if the compilation and its resulting copyright are not
+used to limit the access or legal rights of the compilation's users
+beyond what the individual works permit.  Inclusion of a covered work
+in an aggregate does not cause this License to apply to the other
+parts of the aggregate.
+
+  6. Conveying Non-Source Forms.
+
+  You may convey a covered work in object code form under the terms
+of sections 4 and 5, provided that you also convey the
+machine-readable Corresponding Source under the terms of this License,
+in one of these ways:
+
+    a) Convey the object code in, or embodied in, a physical product
+    (including a physical distribution medium), accompanied by the
+    Corresponding Source fixed on a durable physical medium
+    customarily used for software interchange.
+
+    b) Convey the object code in, or embodied in, a physical product
+    (including a physical distribution medium), accompanied by a
+    written offer, valid for at least three years and valid for as
+    long as you offer spare parts or customer support for that product
+    model, to give anyone who possesses the object code either (1) a
+    copy of the Corresponding Source for all the software in the
+    product that is covered by this License, on a durable physical
+    medium customarily used for software interchange, for a price no
+    more than your reasonable cost of physically performing this
+    conveying of source, or (2) access to copy the
+    Corresponding Source from a network server at no charge.
+
+    c) Convey individual copies of the object code with a copy of the
+    written offer to provide the Corresponding Source.  This
+    alternative is allowed only occasionally and noncommercially, and
+    only if you received the object code with such an offer, in accord
+    with subsection 6b.
+
+    d) Convey the object code by offering access from a designated
+    place (gratis or for a charge), and offer equivalent access to the
+    Corresponding Source in the same way through the same place at no
+    further charge.  You need not require recipients to copy the
+    Corresponding Source along with the object code.  If the place to
+    copy the object code is a network server, the Corresponding Source
+    may be on a different server (operated by you or a third party)
+    that supports equivalent copying facilities, provided you maintain
+    clear directions next to the object code saying where to find the
+    Corresponding Source.  Regardless of what server hosts the
+    Corresponding Source, you remain obligated to ensure that it is
+    available for as long as needed to satisfy these requirements.
+
+    e) Convey the object code using peer-to-peer transmission, provided
+    you inform other peers where the object code and Corresponding
+    Source of the work are being offered to the general public at no
+    charge under subsection 6d.
+
+  A separable portion of the object code, whose source code is excluded
+from the Corresponding Source as a System Library, need not be
+included in conveying the object code work.
+
+  A "User Product" is either (1) a "consumer product", which means any
+tangible personal property which is normally used for personal, family,
+or household purposes, or (2) anything designed or sold for incorporation
+into a dwelling.  In determining whether a product is a consumer product,
+doubtful cases shall be resolved in favor of coverage.  For a particular
+product received by a particular user, "normally used" refers to a
+typical or common use of that class of product, regardless of the status
+of the particular user or of the way in which the particular user
+actually uses, or expects or is expected to use, the product.  A product
+is a consumer product regardless of whether the product has substantial
+commercial, industrial or non-consumer uses, unless such uses represent
+the only significant mode of use of the product.
+
+  "Installation Information" for a User Product means any methods,
+procedures, authorization keys, or other information required to install
+and execute modified versions of a covered work in that User Product from
+a modified version of its Corresponding Source.  The information must
+suffice to ensure that the continued functioning of the modified object
+code is in no case prevented or interfered with solely because
+modification has been made.
+
+  If you convey an object code work under this section in, or with, or
+specifically for use in, a User Product, and the conveying occurs as
+part of a transaction in which the right of possession and use of the
+User Product is transferred to the recipient in perpetuity or for a
+fixed term (regardless of how the transaction is characterized), the
+Corresponding Source conveyed under this section must be accompanied
+by the Installation Information.  But this requirement does not apply
+if neither you nor any third party retains the ability to install
+modified object code on the User Product (for example, the work has
+been installed in ROM).
+
+  The requirement to provide Installation Information does not include a
+requirement to continue to provide support service, warranty, or updates
+for a work that has been modified or installed by the recipient, or for
+the User Product in which it has been modified or installed.  Access to a
+network may be denied when the modification itself materially and
+adversely affects the operation of the network or violates the rules and
+protocols for communication across the network.
+
+  Corresponding Source conveyed, and Installation Information provided,
+in accord with this section must be in a format that is publicly
+documented (and with an implementation available to the public in
+source code form), and must require no special password or key for
+unpacking, reading or copying.
+
+  7. Additional Terms.
+
+  "Additional permissions" are terms that supplement the terms of this
+License by making exceptions from one or more of its conditions.
+Additional permissions that are applicable to the entire Program shall
+be treated as though they were included in this License, to the extent
+that they are valid under applicable law.  If additional permissions
+apply only to part of the Program, that part may be used separately
+under those permissions, but the entire Program remains governed by
+this License without regard to the additional permissions.
+
+  When you convey a copy of a covered work, you may at your option
+remove any additional permissions from that copy, or from any part of
+it.  (Additional permissions may be written to require their own
+removal in certain cases when you modify the work.)  You may place
+additional permissions on material, added by you to a covered work,
+for which you have or can give appropriate copyright permission.
+
+  Notwithstanding any other provision of this License, for material you
+add to a covered work, you may (if authorized by the copyright holders of
+that material) supplement the terms of this License with terms:
+
+    a) Disclaiming warranty or limiting liability differently from the
+    terms of sections 15 and 16 of this License; or
+
+    b) Requiring preservation of specified reasonable legal notices or
+    author attributions in that material or in the Appropriate Legal
+    Notices displayed by works containing it; or
+
+    c) Prohibiting misrepresentation of the origin of that material, or
+    requiring that modified versions of such material be marked in
+    reasonable ways as different from the original version; or
+
+    d) Limiting the use for publicity purposes of names of licensors or
+    authors of the material; or
+
+    e) Declining to grant rights under trademark law for use of some
+    trade names, trademarks, or service marks; or
+
+    f) Requiring indemnification of licensors and authors of that
+    material by anyone who conveys the material (or modified versions of
+    it) with contractual assumptions of liability to the recipient, for
+    any liability that these contractual assumptions directly impose on
+    those licensors and authors.
+
+  All other non-permissive additional terms are considered "further
+restrictions" within the meaning of section 10.  If the Program as you
+received it, or any part of it, contains a notice stating that it is
+governed by this License along with a term that is a further
+restriction, you may remove that term.  If a license document contains
+a further restriction but permits relicensing or conveying under this
+License, you may add to a covered work material governed by the terms
+of that license document, provided that the further restriction does
+not survive such relicensing or conveying.
+
+  If you add terms to a covered work in accord with this section, you
+must place, in the relevant source files, a statement of the
+additional terms that apply to those files, or a notice indicating
+where to find the applicable terms.
+
+  Additional terms, permissive or non-permissive, may be stated in the
+form of a separately written license, or stated as exceptions;
+the above requirements apply either way.
+
+  8. Termination.
+
+  You may not propagate or modify a covered work except as expressly
+provided under this License.  Any attempt otherwise to propagate or
+modify it is void, and will automatically terminate your rights under
+this License (including any patent licenses granted under the third
+paragraph of section 11).
+
+  However, if you cease all violation of this License, then your
+license from a particular copyright holder is reinstated (a)
+provisionally, unless and until the copyright holder explicitly and
+finally terminates your license, and (b) permanently, if the copyright
+holder fails to notify you of the violation by some reasonable means
+prior to 60 days after the cessation.
+
+  Moreover, your license from a particular copyright holder is
+reinstated permanently if the copyright holder notifies you of the
+violation by some reasonable means, this is the first time you have
+received notice of violation of this License (for any work) from that
+copyright holder, and you cure the violation prior to 30 days after
+your receipt of the notice.
+
+  Termination of your rights under this section does not terminate the
+licenses of parties who have received copies or rights from you under
+this License.  If your rights have been terminated and not permanently
+reinstated, you do not qualify to receive new licenses for the same
+material under section 10.
+
+  9. Acceptance Not Required for Having Copies.
+
+  You are not required to accept this License in order to receive or
+run a copy of the Program.  Ancillary propagation of a covered work
+occurring solely as a consequence of using peer-to-peer transmission
+to receive a copy likewise does not require acceptance.  However,
+nothing other than this License grants you permission to propagate or
+modify any covered work.  These actions infringe copyright if you do
+not accept this License.  Therefore, by modifying or propagating a
+covered work, you indicate your acceptance of this License to do so.
+
+  10. Automatic Licensing of Downstream Recipients.
+
+  Each time you convey a covered work, the recipient automatically
+receives a license from the original licensors, to run, modify and
+propagate that work, subject to this License.  You are not responsible
+for enforcing compliance by third parties with this License.
+
+  An "entity transaction" is a transaction transferring control of an
+organization, or substantially all assets of one, or subdividing an
+organization, or merging organizations.  If propagation of a covered
+work results from an entity transaction, each party to that
+transaction who receives a copy of the work also receives whatever
+licenses to the work the party's predecessor in interest had or could
+give under the previous paragraph, plus a right to possession of the
+Corresponding Source of the work from the predecessor in interest, if
+the predecessor has it or can get it with reasonable efforts.
+
+  You may not impose any further restrictions on the exercise of the
+rights granted or affirmed under this License.  For example, you may
+not impose a license fee, royalty, or other charge for exercise of
+rights granted under this License, and you may not initiate litigation
+(including a cross-claim or counterclaim in a lawsuit) alleging that
+any patent claim is infringed by making, using, selling, offering for
+sale, or importing the Program or any portion of it.
+
+  11. Patents.
+
+  A "contributor" is a copyright holder who authorizes use under this
+License of the Program or a work on which the Program is based.  The
+work thus licensed is called the contributor's "contributor version".
+
+  A contributor's "essential patent claims" are all patent claims
+owned or controlled by the contributor, whether already acquired or
+hereafter acquired, that would be infringed by some manner, permitted
+by this License, of making, using, or selling its contributor version,
+but do not include claims that would be infringed only as a
+consequence of further modification of the contributor version.  For
+purposes of this definition, "control" includes the right to grant
+patent sublicenses in a manner consistent with the requirements of
+this License.
+
+  Each contributor grants you a non-exclusive, worldwide, royalty-free
+patent license under the contributor's essential patent claims, to
+make, use, sell, offer for sale, import and otherwise run, modify and
+propagate the contents of its contributor version.
+
+  In the following three paragraphs, a "patent license" is any express
+agreement or commitment, however denominated, not to enforce a patent
+(such as an express permission to practice a patent or covenant not to
+sue for patent infringement).  To "grant" such a patent license to a
+party means to make such an agreement or commitment not to enforce a
+patent against the party.
+
+  If you convey a covered work, knowingly relying on a patent license,
+and the Corresponding Source of the work is not available for anyone
+to copy, free of charge and under the terms of this License, through a
+publicly available network server or other readily accessible means,
+then you must either (1) cause the Corresponding Source to be so
+available, or (2) arrange to deprive yourself of the benefit of the
+patent license for this particular work, or (3) arrange, in a manner
+consistent with the requirements of this License, to extend the patent
+license to downstream recipients.  "Knowingly relying" means you have
+actual knowledge that, but for the patent license, your conveying the
+covered work in a country, or your recipient's use of the covered work
+in a country, would infringe one or more identifiable patents in that
+country that you have reason to believe are valid.
+
+  If, pursuant to or in connection with a single transaction or
+arrangement, you convey, or propagate by procuring conveyance of, a
+covered work, and grant a patent license to some of the parties
+receiving the covered work authorizing them to use, propagate, modify
+or convey a specific copy of the covered work, then the patent license
+you grant is automatically extended to all recipients of the covered
+work and works based on it.
+
+  A patent license is "discriminatory" if it does not include within
+the scope of its coverage, prohibits the exercise of, or is
+conditioned on the non-exercise of one or more of the rights that are
+specifically granted under this License.  You may not convey a covered
+work if you are a party to an arrangement with a third party that is
+in the business of distributing software, under which you make payment
+to the third party based on the extent of your activity of conveying
+the work, and under which the third party grants, to any of the
+parties who would receive the covered work from you, a discriminatory
+patent license (a) in connection with copies of the covered work
+conveyed by you (or copies made from those copies), or (b) primarily
+for and in connection with specific products or compilations that
+contain the covered work, unless you entered into that arrangement,
+or that patent license was granted, prior to 28 March 2007.
+
+  Nothing in this License shall be construed as excluding or limiting
+any implied license or other defenses to infringement that may
+otherwise be available to you under applicable patent law.
+
+  12. No Surrender of Others' Freedom.
+
+  If conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License.  If you cannot convey a
+covered work so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you may
+not convey it at all.  For example, if you agree to terms that obligate you
+to collect a royalty for further conveying from those to whom you convey
+the Program, the only way you could satisfy both those terms and this
+License would be to refrain entirely from conveying the Program.
+
+  13. Use with the GNU Affero General Public License.
+
+  Notwithstanding any other provision of this License, you have
+permission to link or combine any covered work with a work licensed
+under version 3 of the GNU Affero General Public License into a single
+combined work, and to convey the resulting work.  The terms of this
+License will continue to apply to the part which is the covered work,
+but the special requirements of the GNU Affero General Public License,
+section 13, concerning interaction through a network will apply to the
+combination as such.
+
+  14. Revised Versions of this License.
+
+  The Free Software Foundation may publish revised and/or new versions of
+the GNU General Public License from time to time.  Such new versions will
+be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+  Each version is given a distinguishing version number.  If the
+Program specifies that a certain numbered version of the GNU General
+Public License "or any later version" applies to it, you have the
+option of following the terms and conditions either of that numbered
+version or of any later version published by the Free Software
+Foundation.  If the Program does not specify a version number of the
+GNU General Public License, you may choose any version ever published
+by the Free Software Foundation.
+
+  If the Program specifies that a proxy can decide which future
+versions of the GNU General Public License can be used, that proxy's
+public statement of acceptance of a version permanently authorizes you
+to choose that version for the Program.
+
+  Later license versions may give you additional or different
+permissions.  However, no additional obligations are imposed on any
+author or copyright holder as a result of your choosing to follow a
+later version.
+
+  15. Disclaimer of Warranty.
+
+  THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
+APPLICABLE LAW.  EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
+HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
+OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
+THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+PURPOSE.  THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
+IS WITH YOU.  SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
+ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
+
+  16. Limitation of Liability.
+
+  IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
+THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
+GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
+USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
+DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
+PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
+EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
+SUCH DAMAGES.
+
+  17. Interpretation of Sections 15 and 16.
+
+  If the disclaimer of warranty and limitation of liability provided
+above cannot be given local legal effect according to their terms,
+reviewing courts shall apply local law that most closely approximates
+an absolute waiver of all civil liability in connection with the
+Program, unless a warranty or assumption of liability accompanies a
+copy of the Program in return for a fee.
+
+                     END OF TERMS AND CONDITIONS
+
+            How to Apply These Terms to Your New Programs
+
+  If you develop a new program, and you want it to be of the greatest
+possible use to the public, the best way to achieve this is to make it
+free software which everyone can redistribute and change under these terms.
+
+  To do so, attach the following notices to the program.  It is safest
+to attach them to the start of each source file to most effectively
+state the exclusion of warranty; and each file should have at least
+the "copyright" line and a pointer to where the full notice is found.
+
+    <one line to give the program's name and a brief idea of what it does.>
+    Copyright (C) <year>  <name of author>
+
+    This program is free software: you can redistribute it and/or modify
+    it under the terms of the GNU General Public License as published by
+    the Free Software Foundation, either version 3 of the License, or
+    (at your option) any later version.
+
+    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 General Public License for more details.
+
+    You should have received a copy of the GNU General Public License
+    along with this program.  If not, see <http://www.gnu.org/licenses/>.
+
+Also add information on how to contact you by electronic and paper mail.
+
+  If the program does terminal interaction, make it output a short
+notice like this when it starts in an interactive mode:
+
+    <program>  Copyright (C) <year>  <name of author>
+    This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
+    This is free software, and you are welcome to redistribute it
+    under certain conditions; type `show c' for details.
+
+The hypothetical commands `show w' and `show c' should show the appropriate
+parts of the General Public License.  Of course, your program's commands
+might be different; for a GUI interface, you would use an "about box".
+
+  You should also get your employer (if you work as a programmer) or school,
+if any, to sign a "copyright disclaimer" for the program, if necessary.
+For more information on this, and how to apply and follow the GNU GPL, see
+<http://www.gnu.org/licenses/>.
+
+  The GNU General Public License does not permit incorporating your program
+into proprietary programs.  If your program is a subroutine library, you
+may consider it more useful to permit linking proprietary applications with
+the library.  If this is what you want to do, use the GNU Lesser General
+Public License instead of this License.  But first, please read
+<http://www.gnu.org/philosophy/why-not-lgpl.html>.
diff --git a/MANIFEST.in b/MANIFEST.in
new file mode 100644
index 0000000..b2a140e
--- /dev/null
+++ b/MANIFEST.in
@@ -0,0 +1,11 @@
+include INSTALL
+include README
+include TODO
+include COPYRIGHT
+include CHANGELOG
+include LICENSE
+include *.xml
+include *.odt
+include *.csv
+include tests/*
+include doc/*
diff --git a/PKG-INFO b/PKG-INFO
new file mode 100644
index 0000000..2e5b2d4
--- /dev/null
+++ b/PKG-INFO
@@ -0,0 +1,25 @@
+Metadata-Version: 1.0
+Name: trytond_party_vcarddav
+Version: 1.4.0
+Summary: Add CardDAV on parties
+Home-page: http://www.tryton.org/
+Author: B2CK
+Author-email: info at b2ck.com
+License: GPL-3
+Download-URL: http://downloads.tryton.org/1.4/
+Description: UNKNOWN
+Platform: UNKNOWN
+Classifier: Development Status :: 5 - Production/Stable
+Classifier: Environment :: Plugins
+Classifier: Intended Audience :: Developers
+Classifier: Intended Audience :: Financial and Insurance Industry
+Classifier: Intended Audience :: Legal Industry
+Classifier: Intended Audience :: Manufacturing
+Classifier: License :: OSI Approved :: GNU General Public License (GPL)
+Classifier: Natural Language :: English
+Classifier: Natural Language :: French
+Classifier: Natural Language :: German
+Classifier: Natural Language :: Spanish
+Classifier: Operating System :: OS Independent
+Classifier: Programming Language :: Python
+Classifier: Topic :: Office/Business
diff --git a/README b/README
new file mode 100644
index 0000000..bb295c0
--- /dev/null
+++ b/README
@@ -0,0 +1,36 @@
+trytond_party_vcarddav
+======================
+
+The party_vcarddav module of the Tryton application platform.  See
+__tryton__.py
+
+Installing
+----------
+
+See INSTALL
+
+Support
+-------
+
+If you encounter any problems with Tryton, please don't hesitate to ask
+questions on the Tryton bug tracker, mailing list, wiki or IRC channel:
+
+  http://bugs.tryton.org/
+  http://groups.tryton.org/
+  http://wiki.tryton.org/
+  irc://irc.freenode.net/tryton
+
+License
+-------
+
+See LICENSE
+
+Copyright
+---------
+
+See COPYRIGHT
+
+
+For more information please visit the Tryton web site:
+
+  http://www.tryton.org/
diff --git a/__init__.py b/__init__.py
new file mode 100644
index 0000000..a021c2f
--- /dev/null
+++ b/__init__.py
@@ -0,0 +1,6 @@
+#This file is part of Tryton.  The COPYRIGHT file at the top level of
+#this repository contains the full copyright notices and license terms.
+
+import carddav
+from webdav import *
+from party import *
diff --git a/__tryton__.py b/__tryton__.py
new file mode 100644
index 0000000..d479a81
--- /dev/null
+++ b/__tryton__.py
@@ -0,0 +1,34 @@
+# -*- coding: utf-8 -*-
+#This file is part of Tryton.  The COPYRIGHT file at the top level of
+#this repository contains the full copyright notices and license terms.
+{
+    'name' : 'Party - vCardDAV',
+    'name_de_DE': 'Parteien vCardDAV',
+    'name_es_CO' : 'vCardDAV de Compañía',
+    'name_es_ES' : 'Tercero - vCardDAV',
+    'name_fr_FR': 'Tiers - vCardDAV',
+    'version' : '1.4.0',
+    'author' : 'B2CK',
+    'email': 'info at b2ck.com',
+    'website': 'http://www.tryton.org/',
+    'description': 'Add CardDAV on parties',
+    'description_de_DE': 'Ermöglicht CardDAV für Parteien',
+    'description_es_CO': 'Soporte de CardDAV para terceros',
+    'description_es_ES': 'Añade soporte de CardDAV para terceros',
+    'description_fr_FR': 'Ajoute le support CardDAV pour les tiers',
+    'depends' : [
+        'ir',
+        'res',
+        'party',
+        'webdav',
+    ],
+    'xml' : [
+        'party.xml',
+    ],
+    'translation': [
+        'de_DE.csv',
+        'es_CO.csv',
+        'es_ES.csv',
+        'fr_FR.csv',
+    ],
+}
diff --git a/carddav.py b/carddav.py
new file mode 100644
index 0000000..bd1eb02
--- /dev/null
+++ b/carddav.py
@@ -0,0 +1,49 @@
+#This file is part of Tryton.  The COPYRIGHT file at the top level of
+#this repository contains the full copyright notices and license terms.
+_TRYTON_RELOAD = False
+
+from DAV import propfind
+from DAV.errors import *
+from trytond.protocols.webdav import TrytonDAVInterface, USER_ID, CACHE, DATABASE
+from trytond.pool import Pool
+
+TrytonDAVInterface.PROPS['urn:ietf:params:xml:ns:carddav'] = (
+        'address-data',
+        'addressbook-data',
+    )
+TrytonDAVInterface.M_NS['urn:ietf:params:xml:ns:carddav'] = '_get_carddav'
+
+_mk_prop_response = propfind.PROPFIND.mk_prop_response
+
+def mk_prop_response(self, uri, good_props, bad_props, doc):
+    res = _mk_prop_response(self, uri, good_props, bad_props, doc)
+    dbname, uri = TrytonDAVInterface.get_dburi(uri)
+    if uri in ('Contacts', 'Contacts/'):
+        ad = doc.createElement('addressbook')
+        ad.setAttribute('xmlns', 'urn:ietf:params:xml:ns:carddav')
+        vc = doc.createElement('vcard-collection')
+        vc.setAttribute('xmlns', 'http://groupdav.org/')
+        cols = res.getElementsByTagName('D:collection')
+        if cols:
+            cols[0].parentNode.appendChild(ad)
+            cols[0].parentNode.appendChild(vc)
+    return res
+
+propfind.PROPFIND.mk_prop_response = mk_prop_response
+
+def _get_carddav_address_data(self, uri):
+    dbname, dburi = self._get_dburi(uri)
+    cursor = DATABASE['cursor']
+    pool = Pool(DATABASE['dbname'])
+    collection_obj = pool.get('webdav.collection')
+    try:
+        res = collection_obj.get_address_data(cursor, int(USER_ID), dburi,
+                cache=CACHE)
+    except (DAV_Error, DAV_NotFound, DAV_Secret, DAV_Forbidden):
+        raise
+    except:
+        raise DAV_Error(500)
+    return res
+
+TrytonDAVInterface._get_carddav_address_data = _get_carddav_address_data
+TrytonDAVInterface._get_carddav_addressbook_data = _get_carddav_address_data
diff --git a/de_DE.csv b/de_DE.csv
new file mode 100644
index 0000000..093552c
--- /dev/null
+++ b/de_DE.csv
@@ -0,0 +1,7 @@
+type,name,res_id,src,value,fuzzy
+error,party.party,0,The UUID of the party must be unique!,Die UUID für eine Partei kann nicht mehrfach vergeben werden!,0
+field,"party.party,uuid",0,UUID,UUID,0
+field,"party.party,vcard",0,VCard,VCard,0
+help,"party.party,uuid",0,Universally Unique Identifier,Universally Unique Identifier,0
+model,"ir.action,name",report_party_vcard,VCard,VCard,0
+selection,"ir.action.report,extension",0,VCard file,VCard Datei,0
diff --git a/doc/index.rst b/doc/index.rst
new file mode 100644
index 0000000..9bedc54
--- /dev/null
+++ b/doc/index.rst
@@ -0,0 +1,18 @@
+Party - vCardDAV Module
+#######################
+
+The Party - vCardDAV module provide a new report on parties and
+provide a CardDAV interface over WebDAV.
+
+The *VCard* report on parties allow to generate a vCard_ file
+containing contact informations that can be re-imported in other
+programs like mail clients.
+
+The CardDAV interface is available at
+``http://server_url:8080/dbname/Contacts/`` for all softwares that
+support the `CardDAV protocol`_, this brings a two-way synchronisation of
+contacts.
+
+
+.. _vCard: http://en.wikipedia.org/wiki/Vcard
+.. _CardDAV protocol: http://www.vcarddav.org/wiki
diff --git a/es_CO.csv b/es_CO.csv
new file mode 100644
index 0000000..df05f1b
--- /dev/null
+++ b/es_CO.csv
@@ -0,0 +1,7 @@
+type,name,res_id,src,value,fuzzy
+error,party.party,0,The UUID of the party must be unique!,¡La UUID del tercero debe ser única!,0
+field,"party.party,uuid",0,UUID,UUID,0
+field,"party.party,vcard",0,VCard,VCard,0
+help,"party.party,uuid",0,Universally Unique Identifier,Identificador Único Universal,0
+model,"ir.action,name",report_party_vcard,VCard,VCard,0
+selection,"ir.action.report,extension",0,VCard file,Archivo VCard,0
diff --git a/es_ES.csv b/es_ES.csv
new file mode 100644
index 0000000..3224334
--- /dev/null
+++ b/es_ES.csv
@@ -0,0 +1,7 @@
+type,name,res_id,src,value,fuzzy
+error,party.party,0,The UUID of the party must be unique!,El UUID del tercero debe ser único,0
+field,"party.party,uuid",0,UUID,UUID,0
+field,"party.party,vcard",0,VCard,VCard,0
+help,"party.party,uuid",0,Universally Unique Identifier,Identificador universal único,0
+model,"ir.action,name",report_party_vcard,VCard,VCard,0
+selection,"ir.action.report,extension",0,VCard file,Archivo VCard,0
diff --git a/fr_FR.csv b/fr_FR.csv
new file mode 100644
index 0000000..466babf
--- /dev/null
+++ b/fr_FR.csv
@@ -0,0 +1,7 @@
+type,name,res_id,src,value,fuzzy
+error,party.party,0,The UUID of the party must be unique!,,0
+field,"party.party,uuid",0,UUID,,0
+field,"party.party,vcard",0,VCard,,0
+help,"party.party,uuid",0,Universally Unique Identifier,,0
+model,"ir.action,name",report_party_vcard,VCard,,0
+selection,"ir.action.report,extension",0,VCard file,,0
diff --git a/party.py b/party.py
new file mode 100644
index 0000000..f01170b
--- /dev/null
+++ b/party.py
@@ -0,0 +1,383 @@
+#This file is part of Tryton.  The COPYRIGHT file at the top level of
+#this repository contains the full copyright notices and license terms.
+from trytond.model import ModelSQL, ModelView, fields
+from trytond.report import Report
+from trytond.backend import TableHandler, FIELDS
+import base64
+import copy
+import uuid
+
+
+class Party(ModelSQL, ModelView):
+    _name = 'party.party'
+    uuid = fields.Char('UUID', required=True,
+            help='Universally Unique Identifier')
+    vcard = fields.Binary('VCard')
+
+    def __init__(self):
+        super(Party, self).__init__()
+        self._sql_constraints = [
+                ('uuid_uniq', 'UNIQUE(uuid)',
+                    'The UUID of the party must be unique!'),
+        ]
+
+    def init(self, cursor, module_name):
+        table = TableHandler(cursor, self, module_name)
+
+        if not table.column_exist('uuid'):
+            table.add_raw_column('uuid',
+                    FIELDS[self.uuid._type].sql_type(self.uuid),
+                    FIELDS[self.uuid._type].sql_format, None, None)
+            cursor.execute('SELECT id FROM "' + self._table + '"')
+            for id, in cursor.fetchall():
+                cursor.execute('UPDATE "' + self._table + '" ' \
+                        'SET "uuid" = %s WHERE id = %s',
+                        (self.default_uuid(cursor, 0), id))
+        super(Party, self).init(cursor, module_name)
+
+    def default_uuid(self, cursor, user, context=None):
+        return str(uuid.uuid4())
+
+    def create(self, cursor, user, vals, context=None):
+        collection_obj = self.pool.get('webdav.collection')
+
+        res = super(Party, self).create(cursor, user, vals, context=context)
+        # Restart the cache for vcard
+        collection_obj.vcard(cursor.dbname)
+        return res
+
+    def write(self, cursor, user, ids, vals, context=None):
+        collection_obj = self.pool.get('webdav.collection')
+
+        res = super(Party, self).write(cursor, user, ids, vals, context=context)
+        # Restart the cache for vcard
+        collection_obj.vcard(cursor.dbname)
+        return res
+
+    def delete(self, cursor, user, ids, context=None):
+        collection_obj = self.pool.get('webdav.collection')
+
+        res = super(Party, self).delete(cursor, user, ids, context=context)
+        # Restart the cache for vcard
+        collection_obj.vcard(cursor.dbname)
+        return res
+
+    def vcard2values(self, cursor, user, party_id, vcard, context=None):
+        '''
+        Convert vcard to values for create or write
+
+        :param cursor: the database cursor
+        :param user: the user id
+        :param party_id: the party id for write or None for create
+        :param vcard: a vcard instance of vobject
+        :param context: the context
+        :return: a dictionary with values
+        '''
+        import vobject
+        address_obj = self.pool.get('party.address')
+
+        res = {}
+        res['name'] = vcard.fn.value
+        if not hasattr(vcard, 'n'):
+            vcard.add('n')
+            vcard.n.value = vobject.vcard.Name(vcard.fn.value)
+        res['vcard'] = vcard.serialize()
+        if not party_id:
+            if hasattr(vcard, 'uid'):
+                res['uuid'] = vcard.uid.value
+            res['addresses'] = []
+            for adr in vcard.contents.get('adr', []):
+                vals = address_obj.vcard2values(cursor, user, adr,
+                        context=context)
+                res['addresses'].append(('create', vals))
+            res['contact_mechanisms'] = []
+            for email in vcard.contents.get('email', []):
+                vals = {}
+                vals['type'] = 'email'
+                vals['value'] = email.value
+                res['contact_mechanisms'].append(('create', vals))
+            for tel in vcard.contents.get('tel', []):
+                vals = {}
+                vals['type'] = 'phone'
+                if hasattr(tel, 'type_param') \
+                        and 'cell' in tel.type_param.lower():
+                    vals['type'] = 'mobile'
+                vals['value'] = tel.value
+                res['contact_mechanisms'].append(('create', vals))
+        else:
+            party = self.browse(cursor, user, party_id, context=context)
+            i = 0
+            res['addresses'] = []
+            addresses_todelete = []
+            for address in party.addresses:
+                try:
+                    adr = vcard.contents.get('adr', [])[i]
+                except IndexError:
+                    addresses_todelete.append(address.id)
+                    i += 1
+                    continue
+                if not hasattr(adr, 'value'):
+                    addresses_todelete.append(address.id)
+                    i += 1
+                    continue
+                vals = address_obj.vcard2values(cursor, user, adr,
+                        context=context)
+                res['addresses'].append(('write', address.id, vals))
+                i += 1
+            if addresses_todelete:
+                res['addresses'].append(('delete', addresses_todelete))
+            try:
+                new_addresses = vcard.contents.get('adr', [])[i:]
+            except IndexError:
+                new_addresses = []
+            for adr in new_addresses:
+                if not hasattr(adr, 'value'):
+                    continue
+                vals = address_obj.vcard2values(cursor, user, adr,
+                        context=context)
+                res['addresses'].append(('create', vals))
+
+            i = 0
+            res['contact_mechanisms'] = []
+            contact_mechanisms_todelete = []
+            for cm in party.contact_mechanisms:
+                if cm.type != 'email':
+                    continue
+                try:
+                    email = vcard.contents.get('email', [])[i]
+                except IndexError:
+                    contact_mechanisms_todelete.append(cm.id)
+                    i += 1
+                    continue
+                vals = {}
+                vals['value'] = email.value
+                res['contact_mechanisms'].append(('write', cm.id, vals))
+                i += 1
+            try:
+                new_emails = vcard.contents.get('email', [])[i:]
+            except IndexError:
+                new_emails = []
+            for email in new_emails:
+                if not hasattr(email, 'value'):
+                    continue
+                vals = {}
+                vals['type'] = 'email'
+                vals['value'] = email.value
+                res['contact_mechanisms'].append(('create', vals))
+
+            i = 0
+            for cm in party.contact_mechanisms:
+                if cm.type not in ('phone', 'mobile'):
+                    continue
+                try:
+                    tel = vcard.contents.get('tel', [])[i]
+                except IndexError:
+                    contact_mechanisms_todelete.append(cm.id)
+                    i += 1
+                    continue
+                vals = {}
+                vals['value'] = tel.value
+                res['contact_mechanisms'].append(('write', cm.id, vals))
+                i += 1
+            try:
+                new_tels = vcard.contents.get('tel', [])[i:]
+            except IndexError:
+                new_tels = []
+            for tel in new_tels:
+                if not hasattr(tel, 'value'):
+                    continue
+                vals = {}
+                vals['type'] = 'phone'
+                if hasattr(tel, 'type_param') \
+                        and 'cell' in tel.type_param.lower():
+                    vals['type'] = 'mobile'
+                vals['value'] = tel.value
+                res['contact_mechanisms'].append(('create', vals))
+
+            if contact_mechanisms_todelete:
+                res['contact_mechanisms'].append(('delete',
+                    contact_mechanisms_todelete))
+        return res
+
+Party()
+
+
+class Address(ModelSQL, ModelView):
+    _name = 'party.address'
+
+    def vcard2values(self, cursor, user, adr, context=None):
+        '''
+        Convert adr from vcard to values for create or write
+
+        :param cursor: the database cursor
+        :param user: the user id
+        :param adr: a adr from vcard instance of vobject
+        :param context: the context
+        :return: a dictionary with values
+        '''
+        country_obj = self.pool.get('country.country')
+        subdivision_obj = self.pool.get('country.subdivision')
+
+        vals = {}
+        vals['street'] = adr.value.street or ''
+        vals['city'] = adr.value.city or ''
+        vals['zip'] = adr.value.code or ''
+        if adr.value.country:
+            country_ids = country_obj.search(cursor, user, [
+                ('rec_name', '=', adr.value.country),
+                ], limit=1, context=context)
+            if country_ids:
+                vals['country'] = country_ids[0]
+                if adr.value.region:
+                    subdivision_ids = subdivision_obj.search(cursor,
+                            user, [
+                                ('rec_name', '=', adr.value.region),
+                                ('country', '=', country_ids[0]),
+                                ], limit=1, context=context)
+                    if subdivision_ids:
+                        vals['subdivision'] = subdivision_ids[0]
+        return vals
+
+Address()
+
+
+class ActionReport(ModelSQL, ModelView):
+    _name = 'ir.action.report'
+
+    def __init__(self):
+        super(ActionReport, self).__init__()
+        new_ext = ('vcf', 'VCard file')
+        if new_ext not in self.extension.selection:
+            self.extension = copy.copy(self.extension)
+            self.extension.selection = copy.copy(self.extension.selection)
+            self.extension.selection.append(new_ext)
+            self._reset_columns()
+
+ActionReport()
+
+
+class VCard(Report):
+    _name = 'party_vcarddav.party.vcard'
+
+    def execute(self, cursor, user, ids, datas, context=None):
+        party_obj = self.pool.get('party.party')
+        action_report_obj = self.pool.get('ir.action.report')
+
+        if context is None:
+            context = {}
+
+        action_report_ids = action_report_obj.search(cursor, user, [
+            ('report_name', '=', self._name)
+            ], context=context)
+        if not action_report_ids:
+            raise Exception('Error', 'Report (%s) not find!' % self._name)
+        action_report = action_report_obj.browse(cursor, user,
+                action_report_ids[0], context=context)
+
+        parties = party_obj.browse(cursor, user, ids, context=context)
+
+        data = ''.join(self.create_vcard(party).serialize() for party in parties)
+
+        return ('vcf', base64.encodestring(data), False, action_report.name)
+
+    def create_vcard(self, party):
+        '''
+        Return a vcard instance of vobject for the party
+
+        :param party: a BrowseRecord of party.party
+        :return: a vcard instance of vobject
+        '''
+        import vobject
+
+        if party.vcard:
+            vcard = vobject.readOne(party.vcard)
+        else:
+            vcard = vobject.vCard()
+        if not hasattr(vcard, 'n'):
+            vcard.add('n')
+        if not vcard.n.value:
+            vcard.n.value = vobject.vcard.Name(party.name)
+        if not hasattr(vcard, 'fn'):
+            vcard.add('fn')
+        vcard.fn.value = party.full_name
+        if not hasattr(vcard, 'uid'):
+            vcard.add('uid')
+        vcard.uid.value = party.uuid
+
+        i = 0
+        for address in party.addresses:
+            try:
+                adr = vcard.contents.get('adr', [])[i]
+            except IndexError:
+                adr = None
+            if not adr:
+                adr = vcard.add('adr')
+            if not hasattr(adr, 'value'):
+                adr.value = vobject.vcard.Address()
+            adr.value.street = address.street and address.street + (
+                address.streetbis and  (" " + address.streetbis) or '') or ''
+            adr.value.city = address.city or ''
+            if address.subdivision:
+                adr.value.region = address.subdivision.name or ''
+            adr.value.code = address.zip or ''
+            if address.country:
+                adr.value.country = address.country.name or ''
+            i += 1
+        try:
+            older_addresses = vcard.contents.get('adr', [])[i:]
+        except IndexError:
+            older_addresses = []
+        for adr in older_addresses:
+            vcard.contents['adr'].remove(adr)
+
+        email_count = 0
+        tel_count = 0
+        for cm in party.contact_mechanisms:
+            if cm.type == 'email':
+                try:
+                    email = vcard.contents.get('email', [])[email_count]
+                except IndexError:
+                    email = None
+                if not email:
+                    email = vcard.add('email')
+                email.value = cm.value
+                if not hasattr(email, 'type_param'):
+                    email.type_param = 'internet'
+                elif 'internet' not in email.type_param.lower():
+                    email.type_param += ',internet'
+                email_count += 1
+            elif cm.type in ('phone', 'mobile'):
+                try:
+                    tel = vcard.contents.get('tel', [])[tel_count]
+                except IndexError:
+                    tel = None
+                if not tel:
+                    tel = vcard.add('tel')
+                tel.value = cm.value
+                if cm.type == 'mobile':
+                    if not hasattr(tel, 'type_param'):
+                        tel.type_param = 'cell'
+                    elif 'cell' not in tel.type_param.lower():
+                        tel.type_param += ',cell'
+                else:
+                    if not hasattr(tel, 'type_param'):
+                        tel.type_param = 'voice'
+                tel_count += 1
+
+        try:
+            older_emails = vcard.contents.get('email', [])[email_count:]
+        except IndexError:
+            older_emails = []
+        for email in older_emails:
+            vcard.contents['email'].remove(email)
+
+        try:
+            older_tels = vcard.contents.get('tel', [])[tel_count:]
+        except IndexError:
+            older_tels = []
+        for tel in older_tels:
+            vcard.contents['tel'].remove(tel)
+
+        return vcard
+
+VCard()
diff --git a/party.xml b/party.xml
new file mode 100644
index 0000000..e034ae1
--- /dev/null
+++ b/party.xml
@@ -0,0 +1,21 @@
+<?xml version="1.0"?>
+<!-- This file is part of Tryton.  The COPYRIGHT file at the top level of
+this repository contains the full copyright notices and license terms. -->
+<tryton>
+    <data>
+
+        <record model="ir.action.report" id="report_party_vcard">
+            <field name="name">VCard</field>
+            <field name="model">party.party</field>
+            <field name="report_name">party_vcarddav.party.vcard</field>
+            <field name="extension">vcf</field>
+        </record>
+
+        <record model="ir.action.keyword" id="create_vcard_keyword">
+            <field name="keyword">form_print</field>
+            <field name="model">party.party,0</field>
+            <field name="action" ref="report_party_vcard"/>
+        </record>
+
+    </data>
+</tryton>
diff --git a/setup.cfg b/setup.cfg
new file mode 100644
index 0000000..861a9f5
--- /dev/null
+++ b/setup.cfg
@@ -0,0 +1,5 @@
+[egg_info]
+tag_build = 
+tag_date = 0
+tag_svn_revision = 0
+
diff --git a/setup.py b/setup.py
new file mode 100644
index 0000000..e4d9f50
--- /dev/null
+++ b/setup.py
@@ -0,0 +1,58 @@
+#!/usr/bin/env python
+#This file is part of Tryton.  The COPYRIGHT file at the top level of
+#this repository contains the full copyright notices and license terms.
+
+from setuptools import setup, find_packages
+import re
+
+info = eval(file('__tryton__.py').read())
+
+requires = ['vobject', 'PyWebDAV >= 0.9.3']
+for dep in info.get('depends', []):
+    if not re.match(r'(ir|res|workflow|webdav)(\W|$)', dep):
+        requires.append('trytond_' + dep)
+
+major_version, minor_version, _ = info.get('version', '0.0.1').split('.', 2)
+requires.append('trytond >= %s.%s' % (major_version, minor_version))
+requires.append('trytond < %s.%s' % (major_version, int(minor_version) + 1))
+
+setup(name='trytond_party_vcarddav',
+    version=info.get('version', '0.0.1'),
+    description=info.get('description', ''),
+    author=info.get('author', ''),
+    author_email=info.get('email', ''),
+    url=info.get('website', ''),
+    download_url="http://downloads.tryton.org/" + \
+            info.get('version', '0.0.1').rsplit('.', 1)[0] + '/',
+    package_dir={'trytond.modules.party_vcarddav': '.'},
+    packages=[
+        'trytond.modules.party_vcarddav',
+    ],
+    package_data={
+        'trytond.modules.party_vcarddav': info.get('xml', []) \
+                + info.get('translation', []),
+    },
+    classifiers=[
+        'Development Status :: 5 - Production/Stable',
+        'Environment :: Plugins',
+        'Intended Audience :: Developers',
+        'Intended Audience :: Financial and Insurance Industry',
+        'Intended Audience :: Legal Industry',
+        'Intended Audience :: Manufacturing',
+        'License :: OSI Approved :: GNU General Public License (GPL)',
+        'Natural Language :: English',
+        'Natural Language :: French',
+        'Natural Language :: German',
+        'Natural Language :: Spanish',
+        'Operating System :: OS Independent',
+        'Programming Language :: Python',
+        'Topic :: Office/Business',
+    ],
+    license='GPL-3',
+    install_requires=requires,
+    zip_safe=False,
+    entry_points="""
+    [trytond.modules]
+    party_vcarddav = trytond.modules.party_vcarddav
+    """,
+)
diff --git a/trytond_party_vcarddav.egg-info/PKG-INFO b/trytond_party_vcarddav.egg-info/PKG-INFO
new file mode 100644
index 0000000..4140bda
--- /dev/null
+++ b/trytond_party_vcarddav.egg-info/PKG-INFO
@@ -0,0 +1,25 @@
+Metadata-Version: 1.0
+Name: trytond-party-vcarddav
+Version: 1.4.0
+Summary: Add CardDAV on parties
+Home-page: http://www.tryton.org/
+Author: B2CK
+Author-email: info at b2ck.com
+License: GPL-3
+Download-URL: http://downloads.tryton.org/1.4/
+Description: UNKNOWN
+Platform: UNKNOWN
+Classifier: Development Status :: 5 - Production/Stable
+Classifier: Environment :: Plugins
+Classifier: Intended Audience :: Developers
+Classifier: Intended Audience :: Financial and Insurance Industry
+Classifier: Intended Audience :: Legal Industry
+Classifier: Intended Audience :: Manufacturing
+Classifier: License :: OSI Approved :: GNU General Public License (GPL)
+Classifier: Natural Language :: English
+Classifier: Natural Language :: French
+Classifier: Natural Language :: German
+Classifier: Natural Language :: Spanish
+Classifier: Operating System :: OS Independent
+Classifier: Programming Language :: Python
+Classifier: Topic :: Office/Business
diff --git a/trytond_party_vcarddav.egg-info/SOURCES.txt b/trytond_party_vcarddav.egg-info/SOURCES.txt
new file mode 100644
index 0000000..8b4f60c
--- /dev/null
+++ b/trytond_party_vcarddav.egg-info/SOURCES.txt
@@ -0,0 +1,25 @@
+CHANGELOG
+COPYRIGHT
+INSTALL
+LICENSE
+MANIFEST.in
+README
+de_DE.csv
+es_CO.csv
+es_ES.csv
+fr_FR.csv
+party.xml
+setup.py
+./__init__.py
+./__tryton__.py
+./carddav.py
+./party.py
+./webdav.py
+doc/index.rst
+trytond_party_vcarddav.egg-info/PKG-INFO
+trytond_party_vcarddav.egg-info/SOURCES.txt
+trytond_party_vcarddav.egg-info/dependency_links.txt
+trytond_party_vcarddav.egg-info/entry_points.txt
+trytond_party_vcarddav.egg-info/not-zip-safe
+trytond_party_vcarddav.egg-info/requires.txt
+trytond_party_vcarddav.egg-info/top_level.txt
\ No newline at end of file
diff --git a/trytond_party_vcarddav.egg-info/dependency_links.txt b/trytond_party_vcarddav.egg-info/dependency_links.txt
new file mode 100644
index 0000000..8b13789
--- /dev/null
+++ b/trytond_party_vcarddav.egg-info/dependency_links.txt
@@ -0,0 +1 @@
+
diff --git a/trytond_party_vcarddav.egg-info/entry_points.txt b/trytond_party_vcarddav.egg-info/entry_points.txt
new file mode 100644
index 0000000..5e682c5
--- /dev/null
+++ b/trytond_party_vcarddav.egg-info/entry_points.txt
@@ -0,0 +1,4 @@
+
+    [trytond.modules]
+    party_vcarddav = trytond.modules.party_vcarddav
+    
\ No newline at end of file
diff --git a/trytond_party_vcarddav.egg-info/not-zip-safe b/trytond_party_vcarddav.egg-info/not-zip-safe
new file mode 100644
index 0000000..8b13789
--- /dev/null
+++ b/trytond_party_vcarddav.egg-info/not-zip-safe
@@ -0,0 +1 @@
+
diff --git a/trytond_party_vcarddav.egg-info/requires.txt b/trytond_party_vcarddav.egg-info/requires.txt
new file mode 100644
index 0000000..8550012
--- /dev/null
+++ b/trytond_party_vcarddav.egg-info/requires.txt
@@ -0,0 +1,5 @@
+vobject
+PyWebDAV >= 0.9.3
+trytond_party
+trytond >= 1.4
+trytond < 1.5
\ No newline at end of file
diff --git a/trytond_party_vcarddav.egg-info/top_level.txt b/trytond_party_vcarddav.egg-info/top_level.txt
new file mode 100644
index 0000000..93df119
--- /dev/null
+++ b/trytond_party_vcarddav.egg-info/top_level.txt
@@ -0,0 +1 @@
+trytond
diff --git a/webdav.py b/webdav.py
new file mode 100644
index 0000000..df760df
--- /dev/null
+++ b/webdav.py
@@ -0,0 +1,348 @@
+#This file is part of Tryton.  The COPYRIGHT file at the top level of
+#this repository contains the full copyright notices and license terms.
+from trytond.model import ModelView, ModelSQL
+from trytond.tools import Cache, reduce_ids
+from DAV.errors import DAV_NotFound, DAV_Forbidden
+import base64
+import urlparse
+
+CARDDAV_NS = 'urn:ietf:params:xml:ns:carddav'
+
+
+class Collection(ModelSQL, ModelView):
+
+    _name = "webdav.collection"
+
+    def vcard(self, cursor, user, uri, context=None):
+        '''
+        Return party ids of the vcard in uri or False
+
+        :param cursor: the database cursor
+        :param user: the user id
+        :param uri: the uri
+        :param context: the context
+        :return: party id
+            or None if there is no party
+            or False if not in Contacts
+        '''
+        party_obj = self.pool.get('party.party')
+
+        if uri and uri.startswith('Contacts/'):
+            uuid = uri[9:-4]
+            party_ids = party_obj.search(cursor, user, [
+                ('uuid', '=', uuid),
+                ], limit=1, context=context)
+            if party_ids:
+                return party_ids[0]
+            return None
+        if uri == 'Contacts':
+            return None
+        return False
+
+    vcard = Cache('webdav_collection.vcard')(vcard)
+
+    def _carddav_filter_domain(self, cursor, user, filter, context=None):
+        '''
+        Return a domain for the carddav filter
+
+        :param cursor: the database cursor
+        :param user: the user id
+        :param filter: the DOM Element of filter
+        :param context: the context
+
+        :return: a list for domain
+        '''
+        address_obj = self.pool.get('party.address')
+        contact_mechanism_obj = self.pool.get('party.contact_mechanism')
+
+        res = []
+        if not filter:
+            return []
+        if filter.localName == 'addressbook-query':
+            addressbook_filter = filter.getElementsByTagNameNS(
+                    'urn:ietf:params:xml:ns:carddav', 'filter')[0]
+            if addressbook_filter.hasAttribute('test') \
+                    and addressbook_filter.getAttribute('test') == 'allof':
+                res.append('AND')
+            else:
+                res.append('OR')
+
+            for prop in addressbook_filter.childNodes:
+                name = prop.getAttribute('name').lower()
+                field = None
+                if name == 'fn':
+                    field = 'rec_name'
+                if name == 'n':
+                    field = 'name'
+                if name == 'uid':
+                    field = 'uid'
+                if name == 'adr':
+                    field = 'rec_name'
+                if name in ('mail', 'tel'):
+                    field = 'value'
+                if field:
+                    res2 = []
+                    if prop.hasAttribute('test') \
+                            and prop.addressbook_filter.getAttribute('test') == 'allof':
+                        res2.append('AND')
+                    else:
+                        res2.append('OR')
+                    if prop.getElementsByTagNameNS(CARDDAV_NS, 'is-not-defined'):
+                        res2.append((field, '=', False))
+                    for text_match in prop.getElementsByTagNameNS(CARDDAV_NS,
+                            'text-match'):
+                        value = text_match.firstChild.data
+                        negate = False
+                        if text_match.hasAttribute('negate-condition') \
+                                and text_match.getAttribute(
+                                        'negate-condition') == 'yes':
+                            negate = True
+                        type = 'contains'
+                        if text_match.hasAttribute('match-type'):
+                            type = text_match.getAttribute('match-type')
+                        if type == 'equals':
+                            pass
+                        elif type in ('contains', 'substring'):
+                            value = '%' + value + '%'
+                        elif type == 'starts-with':
+                            value = value + '%'
+                        elif type == 'ends-with':
+                            value = '%' + value
+                        if not negate:
+                            res2.append((field, 'ilike', value))
+                        else:
+                            res2.append((field, 'not ilike', value))
+                    if name == 'adr':
+                        domain = res2
+                        address_ids = address_obj.search(cursor, user, domain,
+                                context=context)
+                        res = [('addresses', 'in', address_ids)]
+                    elif name in ('mail', 'tel'):
+                        if name == 'mail':
+                            type = ['email']
+                        else:
+                            type = ['phone', 'mobile']
+                        domain = [('type', 'in', type), res2]
+                        contact_mechanism_ids = contact_mechanism_obj.search(cursor,
+                                user, domain, context=context)
+                        res2 = [('contact_mechanisms', 'in', contact_mechanism_ids)]
+                    res.append(res2)
+        return res
+
+    def get_childs(self, cursor, user, uri, filter=None, context=None,
+            cache=None):
+        party_obj = self.pool.get('party.party')
+
+        if uri in ('Contacts', 'Contacts/'):
+            domain = self._carddav_filter_domain(cursor, user, filter,
+                    context=context)
+            party_ids = party_obj.search(cursor, user, domain, context=context)
+            parties = party_obj.browse(cursor, user, party_ids, context=context)
+            if cache is not None:
+                cache.setdefault('_contact', {})
+                for party_id in party_ids:
+                    cache['_contact'][party_id] = {}
+            return [x.uuid + '.vcf' for x in parties]
+        party_id = self.vcard(cursor, user, uri, context=context)
+        if party_id or party_id is None:
+            return []
+        res = super(Collection, self).get_childs(cursor, user, uri,
+                filter=filter, context=context, cache=cache)
+        if not uri and not filter:
+            res.append('Contacts')
+        return res
+
+    def get_resourcetype(self, cursor, user, uri, context=None, cache=None):
+        from DAV.constants import COLLECTION, OBJECT
+        party_id = self.vcard(cursor, user, uri, context=context)
+        if party_id:
+            return OBJECT
+        elif party_id is None:
+            return COLLECTION
+        return super(Collection, self).get_resourcetype(cursor, user, uri,
+                context=context, cache=cache)
+
+    def get_contenttype(self, cursor, user, uri, context=None, cache=None):
+        if self.vcard(cursor, user, uri, context=context):
+            return 'text/x-vcard'
+        return super(Collection, self).get_contenttype(cursor, user, uri,
+                context=context, cache=cache)
+
+    def get_creationdate(self, cursor, user, uri, context=None, cache=None):
+        party_obj = self.pool.get('party.party')
+        party_id = self.vcard(cursor, user, uri, context=context)
+        if party_id is None:
+            raise DAV_NotFound
+        if party_id:
+            if cache is not None:
+                cache.setdefault('_contact', {})
+                ids = cache['_contact'].keys()
+                if party_id not in ids:
+                    ids.append(party_id)
+                elif 'creationdate' in cache['_contact'][party_id]:
+                    return cache['_contact'][party_id]['creationdate']
+            else:
+                ids = [party_id]
+            res = None
+            for i in range(0, len(ids), cursor.IN_MAX):
+                sub_ids = ids[i:i + cursor.IN_MAX]
+                red_sql, red_ids = reduce_ids('id', sub_ids)
+                cursor.execute('SELECT id, ' \
+                            'EXTRACT(epoch FROM create_date) ' \
+                        'FROM "' + party_obj._table + '" ' \
+                        'WHERE ' + red_sql, red_ids)
+                for party_id2, date in cursor.fetchall():
+                    if party_id2 == party_id:
+                        res = date
+                    if cache is not None:
+                        cache['_contact'].setdefault(party_id2, {})
+                        cache['_contact'][party_id2]['creationdate'] = date
+            if res is not None:
+                return res
+        return super(Collection, self).get_creationdate(cursor, user, uri,
+                context=context, cache=cache)
+
+    def get_lastmodified(self, cursor, user, uri, context=None, cache=None):
+        party_obj = self.pool.get('party.party')
+        address_obj = self.pool.get('party.address')
+        contact_mechanism_obj = self.pool.get('party.contact_mechanism')
+
+        party_id = self.vcard(cursor, user, uri, context=context)
+        if party_id:
+            if cache is not None:
+                cache.setdefault('_contact', {})
+                ids = cache['_contact'].keys()
+                if party_id not in ids:
+                    ids.append(party_id)
+                elif 'lastmodified' in cache['_contact'][party_id]:
+                    return cache['_contact'][party_id]['lastmodified']
+            else:
+                ids = [party_id]
+            res = None
+            for i in range(0, len(ids), cursor.IN_MAX):
+                sub_ids = ids[i:i + cursor.IN_MAX]
+                red_sql, red_ids = reduce_ids('p.id', sub_ids)
+                cursor.execute('SELECT p.id, ' \
+                            'MAX(EXTRACT(epoch FROM ' \
+                                'COALESCE(p.write_date, p.create_date))), ' \
+                            'MAX(EXTRACT(epoch FROM ' \
+                                'COALESCE(a.write_date, a.create_date))), ' \
+                            'MAX(EXTRACT(epoch FROM ' \
+                                'COALESCE(c.write_date, c.create_date))) ' \
+                        'FROM "' + party_obj._table + '" p ' \
+                            'LEFT JOIN "' + address_obj._table + '" a ' \
+                            'ON p.id = a.party ' \
+                            'LEFT JOIN "' + contact_mechanism_obj._table + '" c ' \
+                            'ON p.id = c.party ' \
+                        'WHERE ' + red_sql + ' ' \
+                        'GROUP BY p.id', red_ids)
+                for party_id2, date_p, date_a, date_c in cursor.fetchall():
+                    date = max(date_p, date_a, date_c)
+                    if party_id2 == party_id:
+                        res = date
+                    if cache is not None:
+                        cache['_contact'].setdefault(party_id2, {})
+                        cache['_contact'][party_id2]['lastmodified'] = date
+            if res is not None:
+                return res
+        return super(Collection, self).get_lastmodified(cursor, user, uri,
+                context=context, cache=cache)
+
+    def get_data(self, cursor, user, uri, context=None, cache=None):
+        vcard_obj = self.pool.get('party_vcarddav.party.vcard', type='report')
+        party_id = self.vcard(cursor, user, uri, context=context)
+        if party_id is None:
+            raise DAV_NotFound
+        if party_id:
+            val = vcard_obj.execute(cursor, user, [party_id],
+                    {'id': party_id, 'ids': [party_id]},
+                    context=context)
+            return base64.decodestring(val[1])
+        return super(Collection, self).get_data(cursor, user, uri,
+                context=context, cache=cache)
+
+    def get_address_data(self, cursor, user, uri, context=None, cache=None):
+        vcard_obj = self.pool.get('party_vcarddav.party.vcard', type='report')
+        party_id = self.vcard(cursor, user, uri, context=context)
+        if not party_id:
+            raise DAV_NotFound
+        val = vcard_obj.execute(cursor, user, [party_id],
+                {'id': party_id, 'ids': [party_id]},
+                context=context)
+        res = base64.decodestring(val[1])
+        return res.decode('utf-8')
+
+    def put(self, cursor, user, uri, data, content_type, context=None,
+            cache=None):
+        import vobject
+        party_obj = self.pool.get('party.party')
+
+        party_id = self.vcard(cursor, user, uri, context=context)
+        if party_id is None:
+            vcard = vobject.readOne(data)
+            values = party_obj.vcard2values(cursor, user, None, vcard,
+                    context=context)
+            try:
+                party_id = party_obj.create(cursor, user, values,
+                        context=context)
+            except:
+                raise DAV_Forbidden
+            party = party_obj.browse(cursor, user, party_id, context=context)
+            return cursor.database_name + '/Contacts/' + party.uuid + '.vcf'
+        if party_id:
+            vcard = vobject.readOne(data)
+            values = party_obj.vcard2values(cursor, user, party_id, vcard,
+                    context=context)
+            try:
+                party_obj.write(cursor, user, party_id, values, context=context)
+            except:
+                raise DAV_Forbidden
+            return
+        return super(Collection, self).put(cursor, user, uri, data,
+                content_type, context=context)
+
+    def mkcol(self, cursor, user, uri, context=None, cache=None):
+        party_id = self.vcard(cursor, user, uri, context=context)
+        if party_id is None:
+            raise DAV_Forbidden
+        if party_id:
+            raise DAV_Forbidden
+        return super(Collection, self).mkcol(cursor, user, uri, context=context,
+                cache=cache)
+
+    def rmcol(self, cursor, user, uri, context=None, cache=None):
+        party_id = self.vcard(cursor, user, uri, context=context)
+        if party_id is None:
+            raise DAV_Forbidden
+        if party_id:
+            raise DAV_Forbidden
+        return super(Collection, self).rmcol(cursor, user, uri, context=context,
+                cache=cache)
+
+    def rm(self, cursor, user, uri, context=None, cache=None):
+        party_obj = self.pool.get('party.party')
+
+        party_id = self.vcard(cursor, user, uri, context=context)
+        if party_id is None:
+            raise DAV_Forbidden
+        if party_id:
+            try:
+                party_obj.delete(cursor, user, party_id, context=context)
+            except:
+                raise DAV_Forbidden
+            return 200
+        return super(Collection, self).rm(cursor, user, uri, context=context,
+                cache=cache)
+
+    def exists(self, cursor, user, uri, context=None, cache=None):
+        party_id = self.vcard(cursor, user, uri, context=context)
+        if party_id is None or party_id:
+            if party_id:
+                return 1
+            if uri in ('Contacts', 'Contacts/'):
+                return 1
+            return 0
+        return super(Collection, self).exists(cursor, user, uri, context=context,
+                cache=cache)
+
+Collection()
-- 
tryton-modules-party-vcarddav



More information about the tryton-debian-vcs mailing list