[CalendarServer-changes] [6192] CalendarServer/trunk
source_changes at macosforge.org
source_changes at macosforge.org
Thu Aug 26 09:47:29 PDT 2010
Revision: 6192
http://trac.macosforge.org/projects/calendarserver/changeset/6192
Author: cdaboo at apple.com
Date: 2010-08-26 09:47:28 -0700 (Thu, 26 Aug 2010)
Log Message:
-----------
Merge big re-factor of posgres.py and module re-organization.
Modified Paths:
--------------
CalendarServer/trunk/calendarserver/tap/caldav.py
CalendarServer/trunk/calendarserver/tap/util.py
CalendarServer/trunk/pyflakes
CalendarServer/trunk/setup.py
CalendarServer/trunk/support/Makefile.Apple
CalendarServer/trunk/test
CalendarServer/trunk/twistedcaldav/resource.py
CalendarServer/trunk/twistedcaldav/schedule.py
CalendarServer/trunk/twistedcaldav/storebridge.py
CalendarServer/trunk/twistedcaldav/test/test_calendarquery.py
CalendarServer/trunk/twistedcaldav/test/test_sharing.py
CalendarServer/trunk/twistedcaldav/test/test_wrapping.py
CalendarServer/trunk/txdav/common/datastore/file.py
CalendarServer/trunk/txdav/common/inotifications.py
Added Paths:
-----------
CalendarServer/trunk/txdav/base/
CalendarServer/trunk/txdav/base/__init__.py
CalendarServer/trunk/txdav/base/datastore/
CalendarServer/trunk/txdav/base/datastore/__init__.py
CalendarServer/trunk/txdav/base/datastore/file.py
CalendarServer/trunk/txdav/base/datastore/sql.py
CalendarServer/trunk/txdav/base/datastore/subpostgres.py
CalendarServer/trunk/txdav/base/datastore/test/
CalendarServer/trunk/txdav/base/datastore/test/__init__.py
CalendarServer/trunk/txdav/base/datastore/test/test_subpostgres.py
CalendarServer/trunk/txdav/base/datastore/util.py
CalendarServer/trunk/txdav/base/propertystore/
CalendarServer/trunk/txdav/base/propertystore/__init__.py
CalendarServer/trunk/txdav/base/propertystore/base.py
CalendarServer/trunk/txdav/base/propertystore/none.py
CalendarServer/trunk/txdav/base/propertystore/sql.py
CalendarServer/trunk/txdav/base/propertystore/test/
CalendarServer/trunk/txdav/base/propertystore/test/__init__.py
CalendarServer/trunk/txdav/base/propertystore/test/base.py
CalendarServer/trunk/txdav/base/propertystore/test/test_base.py
CalendarServer/trunk/txdav/base/propertystore/test/test_none.py
CalendarServer/trunk/txdav/base/propertystore/test/test_sql.py
CalendarServer/trunk/txdav/base/propertystore/test/test_xattr.py
CalendarServer/trunk/txdav/base/propertystore/xattr.py
CalendarServer/trunk/txdav/caldav/
CalendarServer/trunk/txdav/caldav/__init__.py
CalendarServer/trunk/txdav/caldav/datastore/
CalendarServer/trunk/txdav/caldav/datastore/__init__.py
CalendarServer/trunk/txdav/caldav/datastore/file.py
CalendarServer/trunk/txdav/caldav/datastore/scheduling.py
CalendarServer/trunk/txdav/caldav/datastore/sql.py
CalendarServer/trunk/txdav/caldav/datastore/test/
CalendarServer/trunk/txdav/caldav/datastore/test/__init__.py
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/1.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/2.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/3.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/24204e8682b99527cbda64d7423acda7.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/61038c41bd02ae5daf9f7fe9d54199fd.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/84be58ced1f1bb34057e1bd7e602c9c8.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/acc1015b7dc300c1b5665f6833960994.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b0d5785f275c064117ffd1fc20f4ed40.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b495c5dd5aa53392078eb43b1f906a80.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b88dd50941e4a31520ee396fd7894c96.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_empty/
CalendarServer/trunk/txdav/caldav/datastore/test/common.py
CalendarServer/trunk/txdav/caldav/datastore/test/test_file.py
CalendarServer/trunk/txdav/caldav/datastore/test/test_scheduling.py
CalendarServer/trunk/txdav/caldav/datastore/test/test_sql.py
CalendarServer/trunk/txdav/caldav/datastore/util.py
CalendarServer/trunk/txdav/caldav/icalendarstore.py
CalendarServer/trunk/txdav/caldav/resource.py
CalendarServer/trunk/txdav/carddav/
CalendarServer/trunk/txdav/carddav/__init__.py
CalendarServer/trunk/txdav/carddav/datastore/
CalendarServer/trunk/txdav/carddav/datastore/__init__.py
CalendarServer/trunk/txdav/carddav/datastore/file.py
CalendarServer/trunk/txdav/carddav/datastore/sql.py
CalendarServer/trunk/txdav/carddav/datastore/test/
CalendarServer/trunk/txdav/carddav/datastore/test/__init__.py
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/1.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/2.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/3.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/3765A955-1B96-41EA-994D-335192BEDCCD.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44745975-AE6D-4FB0-80A6-A298427E047A.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44EE78BF-8814-4471-899C-92280CEFB098.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/8424B7F0-C878-4722-B522-EBB07CF48AD7.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/934731C6-1C95-4C40-BE1F-FA4215B2307B.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/AFBB77B8-0438-4825-A1DB-A75D76B6C3A8.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E1.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E2.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/F0A6918D-8E09-43FA-9684-226810B8A96F.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/FCBA0FA3-00B2-4C95-B4EC-4CCC4843F8B1.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_empty/
CalendarServer/trunk/txdav/carddav/datastore/test/common.py
CalendarServer/trunk/txdav/carddav/datastore/test/test_file.py
CalendarServer/trunk/txdav/carddav/datastore/test/test_sql.py
CalendarServer/trunk/txdav/carddav/datastore/util.py
CalendarServer/trunk/txdav/carddav/iaddressbookstore.py
CalendarServer/trunk/txdav/carddav/resource.py
CalendarServer/trunk/txdav/common/datastore/sql.py
CalendarServer/trunk/txdav/common/datastore/sql_legacy.py
CalendarServer/trunk/txdav/common/datastore/sql_schema_v1.sql
CalendarServer/trunk/txdav/common/datastore/sql_tables.py
CalendarServer/trunk/txdav/common/datastore/test/
CalendarServer/trunk/txdav/common/datastore/test/__init__.py
CalendarServer/trunk/txdav/common/datastore/test/util.py
Removed Paths:
-------------
CalendarServer/trunk/txcaldav/
CalendarServer/trunk/txcarddav/
CalendarServer/trunk/txdav/base/__init__.py
CalendarServer/trunk/txdav/base/datastore/
CalendarServer/trunk/txdav/base/datastore/__init__.py
CalendarServer/trunk/txdav/base/datastore/file.py
CalendarServer/trunk/txdav/base/datastore/sql.py
CalendarServer/trunk/txdav/base/datastore/subpostgres.py
CalendarServer/trunk/txdav/base/datastore/test/
CalendarServer/trunk/txdav/base/datastore/test/__init__.py
CalendarServer/trunk/txdav/base/datastore/test/test_subpostgres.py
CalendarServer/trunk/txdav/base/datastore/util.py
CalendarServer/trunk/txdav/base/propertystore/
CalendarServer/trunk/txdav/base/propertystore/__init__.py
CalendarServer/trunk/txdav/base/propertystore/base.py
CalendarServer/trunk/txdav/base/propertystore/none.py
CalendarServer/trunk/txdav/base/propertystore/sql.py
CalendarServer/trunk/txdav/base/propertystore/test/
CalendarServer/trunk/txdav/base/propertystore/test/__init__.py
CalendarServer/trunk/txdav/base/propertystore/test/base.py
CalendarServer/trunk/txdav/base/propertystore/test/test_base.py
CalendarServer/trunk/txdav/base/propertystore/test/test_none.py
CalendarServer/trunk/txdav/base/propertystore/test/test_sql.py
CalendarServer/trunk/txdav/base/propertystore/test/test_xattr.py
CalendarServer/trunk/txdav/base/propertystore/xattr.py
CalendarServer/trunk/txdav/caldav/__init__.py
CalendarServer/trunk/txdav/caldav/datastore/
CalendarServer/trunk/txdav/caldav/datastore/__init__.py
CalendarServer/trunk/txdav/caldav/datastore/file.py
CalendarServer/trunk/txdav/caldav/datastore/scheduling.py
CalendarServer/trunk/txdav/caldav/datastore/sql.py
CalendarServer/trunk/txdav/caldav/datastore/test/
CalendarServer/trunk/txdav/caldav/datastore/test/__init__.py
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/1.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/2.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/3.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/24204e8682b99527cbda64d7423acda7.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/61038c41bd02ae5daf9f7fe9d54199fd.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/84be58ced1f1bb34057e1bd7e602c9c8.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/acc1015b7dc300c1b5665f6833960994.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b0d5785f275c064117ffd1fc20f4ed40.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b495c5dd5aa53392078eb43b1f906a80.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b88dd50941e4a31520ee396fd7894c96.ics
CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_empty/
CalendarServer/trunk/txdav/caldav/datastore/test/common.py
CalendarServer/trunk/txdav/caldav/datastore/test/test_file.py
CalendarServer/trunk/txdav/caldav/datastore/test/test_scheduling.py
CalendarServer/trunk/txdav/caldav/datastore/test/test_sql.py
CalendarServer/trunk/txdav/caldav/datastore/util.py
CalendarServer/trunk/txdav/caldav/icalendarstore.py
CalendarServer/trunk/txdav/caldav/resource.py
CalendarServer/trunk/txdav/carddav/__init__.py
CalendarServer/trunk/txdav/carddav/datastore/
CalendarServer/trunk/txdav/carddav/datastore/__init__.py
CalendarServer/trunk/txdav/carddav/datastore/file.py
CalendarServer/trunk/txdav/carddav/datastore/sql.py
CalendarServer/trunk/txdav/carddav/datastore/test/
CalendarServer/trunk/txdav/carddav/datastore/test/__init__.py
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/1.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/2.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/3.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/3765A955-1B96-41EA-994D-335192BEDCCD.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44745975-AE6D-4FB0-80A6-A298427E047A.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44EE78BF-8814-4471-899C-92280CEFB098.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/8424B7F0-C878-4722-B522-EBB07CF48AD7.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/934731C6-1C95-4C40-BE1F-FA4215B2307B.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/AFBB77B8-0438-4825-A1DB-A75D76B6C3A8.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E1.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E2.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/F0A6918D-8E09-43FA-9684-226810B8A96F.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/FCBA0FA3-00B2-4C95-B4EC-4CCC4843F8B1.vcf
CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_empty/
CalendarServer/trunk/txdav/carddav/datastore/test/common.py
CalendarServer/trunk/txdav/carddav/datastore/test/test_file.py
CalendarServer/trunk/txdav/carddav/datastore/test/test_sql.py
CalendarServer/trunk/txdav/carddav/datastore/util.py
CalendarServer/trunk/txdav/carddav/iaddressbookstore.py
CalendarServer/trunk/txdav/carddav/resource.py
CalendarServer/trunk/txdav/common/datastore/test/__init__.py
CalendarServer/trunk/txdav/common/datastore/test/util.py
CalendarServer/trunk/txdav/datastore/
CalendarServer/trunk/txdav/propertystore/
Property Changed:
----------------
CalendarServer/trunk/
Property changes on: CalendarServer/trunk
___________________________________________________________________
Modified: svn:mergeinfo
- /CalendarServer/branches/config-separation:4379-4443
/CalendarServer/branches/egg-info-351:4589-4625
/CalendarServer/branches/new-store:5594-5934
/CalendarServer/branches/new-store-no-caldavfile:5911-5935
/CalendarServer/branches/new-store-no-caldavfile-2:5936-5981
/CalendarServer/branches/users/cdaboo/cached-subscription-calendars-5692:5693-5702
/CalendarServer/branches/users/cdaboo/directory-cache-on-demand-3627:3628-3644
/CalendarServer/branches/users/cdaboo/more-sharing-5591:5592-5601
/CalendarServer/branches/users/cdaboo/partition-4464:4465-4957
/CalendarServer/branches/users/cdaboo/relative-config-paths-5070:5071-5105
/CalendarServer/branches/users/cdaboo/shared-calendars-5187:5188-5440
/CalendarServer/branches/users/glyph/contacts-server-merge:4971-5080
/CalendarServer/branches/users/glyph/sendfdport:5388-5424
/CalendarServer/branches/users/glyph/sql-store:5929-6073
/CalendarServer/branches/users/glyph/use-system-twisted:5084-5149
/CalendarServer/branches/users/sagen/locations-resources:5032-5051
/CalendarServer/branches/users/sagen/locations-resources-2:5052-5061
/CalendarServer/branches/users/sagen/resource-delegates-4038:4040-4067
/CalendarServer/branches/users/sagen/resource-delegates-4066:4068-4075
/CalendarServer/branches/users/sagen/resources-2:5084-5093
/CalendarServer/branches/users/wsanchez/transations:5515-5593
+ /CalendarServer/branches/config-separation:4379-4443
/CalendarServer/branches/egg-info-351:4589-4625
/CalendarServer/branches/generic-sqlstore:6167-6191
/CalendarServer/branches/new-store:5594-5934
/CalendarServer/branches/new-store-no-caldavfile:5911-5935
/CalendarServer/branches/new-store-no-caldavfile-2:5936-5981
/CalendarServer/branches/users/cdaboo/cached-subscription-calendars-5692:5693-5702
/CalendarServer/branches/users/cdaboo/directory-cache-on-demand-3627:3628-3644
/CalendarServer/branches/users/cdaboo/more-sharing-5591:5592-5601
/CalendarServer/branches/users/cdaboo/partition-4464:4465-4957
/CalendarServer/branches/users/cdaboo/relative-config-paths-5070:5071-5105
/CalendarServer/branches/users/cdaboo/shared-calendars-5187:5188-5440
/CalendarServer/branches/users/glyph/contacts-server-merge:4971-5080
/CalendarServer/branches/users/glyph/sendfdport:5388-5424
/CalendarServer/branches/users/glyph/sql-store:5929-6073
/CalendarServer/branches/users/glyph/use-system-twisted:5084-5149
/CalendarServer/branches/users/sagen/locations-resources:5032-5051
/CalendarServer/branches/users/sagen/locations-resources-2:5052-5061
/CalendarServer/branches/users/sagen/resource-delegates-4038:4040-4067
/CalendarServer/branches/users/sagen/resource-delegates-4066:4068-4075
/CalendarServer/branches/users/sagen/resources-2:5084-5093
/CalendarServer/branches/users/wsanchez/transations:5515-5593
Modified: CalendarServer/trunk/calendarserver/tap/caldav.py
===================================================================
--- CalendarServer/trunk/calendarserver/tap/caldav.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/calendarserver/tap/caldav.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -88,8 +88,8 @@
from calendarserver.tap.util import getRootResource, computeProcessCount
from calendarserver.tools.util import checkDirectory
-from txcaldav.calendarstore.postgres import v1_schema
-from txdav.datastore.subpostgres import PostgresService
+from txdav.common.datastore.sql import v1_schema
+from txdav.base.datastore.subpostgres import PostgresService
from twext.python.filepath import CachingFilePath
log = Logger()
Modified: CalendarServer/trunk/calendarserver/tap/util.py
===================================================================
--- CalendarServer/trunk/calendarserver/tap/util.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/calendarserver/tap/util.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -67,9 +67,10 @@
from calendarserver.webadmin.resource import WebAdminResource
from calendarserver.webcal.resource import WebCalendarResource
-from txdav.common.datastore.file import CommonDataStore
-from txcaldav.calendarstore.postgres import PostgresStore, v1_schema
-from txdav.datastore.subpostgres import PostgresService
+from txdav.common.datastore.sql import CommonDataStore as CommonSQLDataStore
+from txdav.common.datastore.file import CommonDataStore as CommonFileDataStore
+from txdav.common.datastore.sql import v1_schema
+from txdav.base.datastore.subpostgres import PostgresService
from twext.python.filepath import CachingFilePath
@@ -305,11 +306,10 @@
_dbRoot = CachingFilePath(config.DatabaseRoot)
_postgresService = PostgresService(_dbRoot, None, v1_schema, "caldav",
logFile=config.PostgresLogFile)
- _newStore = PostgresStore(_postgresService.produceConnection,
- notifierFactory, # config.EnableCalDAV, config.EnableCardDAV)
- _dbRoot.child("attachments"))
+ _newStore = CommonSQLDataStore(_postgresService.produceConnection,
+ notifierFactory, _dbRoot.child("attachments"), config.EnableCalDAV, config.EnableCardDAV)
else:
- _newStore = CommonDataStore(FilePath(config.DocumentRoot),
+ _newStore = CommonFileDataStore(FilePath(config.DocumentRoot),
notifierFactory, config.EnableCalDAV, config.EnableCardDAV)
if config.EnableCalDAV:
Modified: CalendarServer/trunk/pyflakes
===================================================================
--- CalendarServer/trunk/pyflakes 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/pyflakes 2010-08-26 16:47:28 UTC (rev 6192)
@@ -9,7 +9,7 @@
export PYTHONPATH="${flakes}:${PYTHONPATH:-}";
if [ $# -eq 0 ]; then
- set - calendarserver twistedcaldav twext txdav txcaldav txcarddav;
+ set - calendarserver twistedcaldav twext txdav;
fi;
cd "${wd}" && "${flakes}/bin/pyflakes" "$@" | sed \
Modified: CalendarServer/trunk/setup.py
===================================================================
--- CalendarServer/trunk/setup.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/setup.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -108,7 +108,7 @@
"zoneinfo/*/*/*.ics",
"images/*/*.jpg",
],
- "txcaldav.calendarstore": [
+ "txdav.common.datastore": [
"*.sql",
],
},
Modified: CalendarServer/trunk/support/Makefile.Apple
===================================================================
--- CalendarServer/trunk/support/Makefile.Apple 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/support/Makefile.Apple 2010-08-26 16:47:28 UTC (rev 6192)
@@ -127,7 +127,7 @@
$(BuildDirectory)/$(Project):
@echo "Copying source for $(Project)..."
$(_v) $(MKDIR) -p "$@"
- $(_v) pax -rw bin conf Makefile lib-patches setup.py calendarserver twistedcaldav twext txdav txcaldav txcarddav twisted support "$@/"
+ $(_v) pax -rw bin conf Makefile lib-patches setup.py calendarserver twistedcaldav twext txdav twisted support "$@/"
$(BuildDirectory)/%: %.tgz
@echo "Extracting source for $(notdir $<)..."
Modified: CalendarServer/trunk/test
===================================================================
--- CalendarServer/trunk/test 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/test 2010-08-26 16:47:28 UTC (rev 6192)
@@ -78,7 +78,7 @@
if [ $# -gt 0 ]; then
test_modules="$@";
else
- test_modules="calendarserver twistedcaldav twext txdav txcaldav txcarddav ${m_twisted}";
+ test_modules="calendarserver twistedcaldav twext txdav ${m_twisted}";
fi;
cd "${wd}" && "${python}" "${trial}" --rterrors ${random} ${until_fail} ${no_colour} ${coverage} ${test_modules};
Modified: CalendarServer/trunk/twistedcaldav/resource.py
===================================================================
--- CalendarServer/trunk/twistedcaldav/resource.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/twistedcaldav/resource.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -240,14 +240,14 @@
def associateWithTransaction(self, transaction):
"""
- Associate this resource with a L{txcaldav.idav.ITransaction}; when this
+ Associate this resource with a L{txdav.caldav.idav.ITransaction}; when this
resource (or any of its children) are rendered successfully, commit the
transaction. Otherwise, abort the transaction.
@param transaction: the transaction to associate this resource and its
children with.
- @type transaction: L{txcaldav.idav.ITransaction}
+ @type transaction: L{txdav.caldav.idav.ITransaction}
"""
# FIXME: needs to reject association with transaction if it's already
# got one (resources associated with a transaction are not reusable)
Modified: CalendarServer/trunk/twistedcaldav/schedule.py
===================================================================
--- CalendarServer/trunk/twistedcaldav/schedule.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/twistedcaldav/schedule.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -50,7 +50,7 @@
from twistedcaldav.resource import isCalendarCollectionResource
from twistedcaldav.scheduling.scheduler import CalDAVScheduler, IScheduleScheduler
-from txdav.propertystore.base import PropertyName
+from txdav.base.propertystore.base import PropertyName
def _schedulePrivilegeSet(deliver):
edited = False
Modified: CalendarServer/trunk/twistedcaldav/storebridge.py
===================================================================
--- CalendarServer/trunk/twistedcaldav/storebridge.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/twistedcaldav/storebridge.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -16,8 +16,8 @@
##
"""
-Wrappers to translate between the APIs in L{txcaldav.icalendarstore} and
-L{txcarddav.iaddressbookstore} and those in L{twistedcaldav}.
+Wrappers to translate between the APIs in L{txdav.caldav.icalendarstore} and
+L{txdav.carddav.iaddressbookstore} and those in L{twistedcaldav}.
"""
import hashlib
@@ -56,7 +56,7 @@
from twistedcaldav.vcard import Component as VCard
from txdav.common.icommondatastore import NoSuchObjectResourceError
-from txdav.propertystore.base import PropertyName
+from txdav.base.propertystore.base import PropertyName
log = Logger()
@@ -206,10 +206,10 @@
Initialize with a calendar.
@param calendar: the wrapped calendar.
- @type calendar: L{txcaldav.icalendarstore.ICalendar}
+ @type calendar: L{txdav.caldav.icalendarstore.ICalendar}
@param home: the home through which the given calendar was accessed.
- @type home: L{txcaldav.icalendarstore.ICalendarHome}
+ @type home: L{txdav.caldav.icalendarstore.ICalendarHome}
"""
self._newStoreCalendar = calendar
self._newStoreParentHome = home
@@ -664,12 +664,12 @@
class CalendarCollectionResource(_CalendarChildHelper, CalDAVResource):
"""
- Wrapper around a L{txcaldav.icalendar.ICalendar}.
+ Wrapper around a L{txdav.caldav.icalendar.ICalendar}.
"""
def __init__(self, calendar, home, *args, **kw):
"""
- Create a CalendarCollectionResource from a L{txcaldav.icalendar.ICalendar}
+ Create a CalendarCollectionResource from a L{txdav.caldav.icalendar.ICalendar}
and the arguments required for L{CalDAVResource}.
"""
super(CalendarCollectionResource, self).__init__(*args, **kw)
@@ -882,7 +882,7 @@
@param home: The calendar home which will be this resource's parent,
when it exists.
- @type home: L{txcaldav.icalendarstore.ICalendarHome}
+ @type home: L{txdav.caldav.icalendarstore.ICalendarHome}
"""
super(ProtoCalendarCollectionResource, self).__init__(*args, **kw)
self._newStoreParentHome = home
@@ -957,7 +957,7 @@
Construct a L{CalendarObjectResource} from an L{ICalendarObject}.
@param calendarObject: The storage for the calendar object.
- @type calendarObject: L{txcaldav.icalendarstore.ICalendarObject}
+ @type calendarObject: L{txdav.caldav.icalendarstore.ICalendarObject}
"""
super(CalendarObjectResource, self).__init__(*args, **kw)
self._initializeWithObject(calendarObject)
@@ -1221,10 +1221,10 @@
Initialize with a addressbook.
@param addressbook: the wrapped addressbook.
- @type addressbook: L{txcarddav.iaddressbookstore.IAddressBook}
+ @type addressbook: L{txdav.carddav.iaddressbookstore.IAddressBook}
@param home: the home through which the given addressbook was accessed.
- @type home: L{txcarddav.iaddressbookstore.IAddressBookHome}
+ @type home: L{txdav.carddav.iaddressbookstore.IAddressBookHome}
"""
self._newStoreAddressBook = addressbook
self._newStoreParentHome = home
@@ -1307,12 +1307,12 @@
class AddressBookCollectionResource(_AddressBookChildHelper, CalDAVResource):
"""
- Wrapper around a L{txcarddav.iaddressbook.IAddressBook}.
+ Wrapper around a L{txdav.carddav.iaddressbook.IAddressBook}.
"""
def __init__(self, addressbook, home, *args, **kw):
"""
- Create a AddressBookCollectionResource from a L{txcarddav.iaddressbook.IAddressBook}
+ Create a AddressBookCollectionResource from a L{txdav.carddav.iaddressbook.IAddressBook}
and the arguments required for L{CalDAVResource}.
"""
super(AddressBookCollectionResource, self).__init__(*args, **kw)
@@ -1486,7 +1486,7 @@
@param home: The addressbook home which will be this resource's parent,
when it exists.
- @type home: L{txcarddav.iaddressbookstore.IAddressBookHome}
+ @type home: L{txdav.carddav.iaddressbookstore.IAddressBookHome}
"""
super(ProtoAddressBookCollectionResource, self).__init__(*args, **kw)
self._newStoreParentHome = home
@@ -1552,7 +1552,7 @@
class GlobalAddressBookCollectionResource(GlobalAddressBookResource, AddressBookCollectionResource):
"""
- Wrapper around a L{txcarddav.iaddressbook.IAddressBook}.
+ Wrapper around a L{txdav.carddav.iaddressbook.IAddressBook}.
"""
pass
@@ -1575,7 +1575,7 @@
Construct a L{AddressBookObjectResource} from an L{IAddressBookObject}.
@param Object: The storage for the addressbook object.
- @type Object: L{txcarddav.iaddressbookstore.IAddressBookObject}
+ @type Object: L{txdav.carddav.iaddressbookstore.IAddressBookObject}
"""
super(AddressBookObjectResource, self).__init__(*args, **kw)
self._initializeWithObject(Object)
@@ -1812,12 +1812,12 @@
class StoreNotificationCollectionResource(_NotificationChildHelper,
NotificationCollectionResource):
"""
- Wrapper around a L{txcaldav.icalendar.ICalendar}.
+ Wrapper around a L{txdav.caldav.icalendar.ICalendar}.
"""
def __init__(self, notifications, home, *args, **kw):
"""
- Create a CalendarCollectionResource from a L{txcaldav.icalendar.ICalendar}
+ Create a CalendarCollectionResource from a L{txdav.caldav.icalendar.ICalendar}
and the arguments required for L{CalDAVResource}.
"""
super(StoreNotificationCollectionResource, self).__init__(*args, **kw)
@@ -1855,7 +1855,7 @@
@param home: The calendar home which will be this resource's parent,
when it exists.
- @type home: L{txcaldav.icalendarstore.ICalendarHome}
+ @type home: L{txdav.caldav.icalendarstore.ICalendarHome}
"""
self._newStoreParentHome = home
super(StoreProtoNotificationCollectionResource, self).__init__(*args, **kw)
@@ -1925,7 +1925,7 @@
Construct a L{CalendarObjectResource} from an L{ICalendarObject}.
@param calendarObject: The storage for the calendar object.
- @type calendarObject: L{txcaldav.icalendarstore.ICalendarObject}
+ @type calendarObject: L{txdav.caldav.icalendarstore.ICalendarObject}
"""
super(StoreNotificationObjectFile, self).__init__(*args, **kw)
self._initializeWithObject(notificationObject)
Modified: CalendarServer/trunk/twistedcaldav/test/test_calendarquery.py
===================================================================
--- CalendarServer/trunk/twistedcaldav/test/test_calendarquery.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/twistedcaldav/test/test_calendarquery.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -34,8 +34,8 @@
from twistedcaldav.config import config
from twistedcaldav.test.util import HomeTestCase
from twisted.internet.defer import inlineCallbacks, returnValue
-from txcaldav.calendarstore.test.test_postgres import buildStore
-from txcaldav.calendarstore.test.common import StubNotifierFactory
+from txdav.caldav.datastore.test.test_sql import buildStore
+from txdav.caldav.datastore.test.common import StubNotifierFactory
@inlineCallbacks
Modified: CalendarServer/trunk/twistedcaldav/test/test_sharing.py
===================================================================
--- CalendarServer/trunk/twistedcaldav/test/test_sharing.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/twistedcaldav/test/test_sharing.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -25,8 +25,8 @@
from twistedcaldav.config import config
from twistedcaldav.test.util import HomeTestCase, norequest
from twistedcaldav.resource import CalDAVResource
-from txcaldav.calendarstore.test.test_postgres import buildStore
-from txcaldav.calendarstore.test.common import StubNotifierFactory
+from txdav.caldav.datastore.test.test_sql import buildStore
+from txdav.caldav.datastore.test.common import StubNotifierFactory
class SharingTests(HomeTestCase):
Modified: CalendarServer/trunk/twistedcaldav/test/test_wrapping.py
===================================================================
--- CalendarServer/trunk/twistedcaldav/test/test_wrapping.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/twistedcaldav/test/test_wrapping.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -37,15 +37,15 @@
from twistedcaldav.test.util import TestCase
-from txcaldav.calendarstore.test.test_file import event4_text
+from txdav.caldav.datastore.test.test_file import event4_text
-from txcarddav.addressbookstore.test.test_file import vcard4_text
+from txdav.carddav.datastore.test.test_file import vcard4_text
-from txcaldav.calendarstore.test.test_postgres import buildStore
-from txcaldav.calendarstore.test.common import StubNotifierFactory, \
+from txdav.caldav.datastore.test.test_sql import buildStore
+from txdav.caldav.datastore.test.common import StubNotifierFactory, \
assertProvides
-from txcaldav.icalendarstore import ICalendarHome
-from txcarddav.iaddressbookstore import IAddressBookHome
+from txdav.caldav.icalendarstore import ICalendarHome
+from txdav.carddav.iaddressbookstore import IAddressBookHome
@@ -73,7 +73,7 @@
class WrappingTests(TestCase):
"""
Tests for L{twistedcaldav.static.CalDAVResource} creating the appropriate type
- of txcaldav.calendarstore.file underlying object when it can determine what
+ of txdav.caldav.datastore.file underlying object when it can determine what
type it really represents.
"""
Deleted: CalendarServer/trunk/txdav/base/__init__.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/__init__.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,19 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Base DAV store.
-"""
Copied: CalendarServer/trunk/txdav/base/__init__.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/__init__.py)
===================================================================
--- CalendarServer/trunk/txdav/base/__init__.py (rev 0)
+++ CalendarServer/trunk/txdav/base/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,19 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Base DAV store.
+"""
Deleted: CalendarServer/trunk/txdav/base/datastore/__init__.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/datastore/__init__.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/datastore/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,19 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-WebDAV data store for Twisted.
-"""
Copied: CalendarServer/trunk/txdav/base/datastore/__init__.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/datastore/__init__.py)
===================================================================
--- CalendarServer/trunk/txdav/base/datastore/__init__.py (rev 0)
+++ CalendarServer/trunk/txdav/base/datastore/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,19 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+WebDAV data store for Twisted.
+"""
Deleted: CalendarServer/trunk/txdav/base/datastore/file.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/datastore/file.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/datastore/file.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,259 +0,0 @@
-# -*- test-case-name: txdav -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-
-"""
-Common utility functions for a file based datastore.
-"""
-
-from twext.python.log import LoggingMixIn
-from txdav.idav import IDataStoreResource
-from txdav.idav import AlreadyFinishedError
-from txdav.base.propertystore.base import PropertyName
-
-from twext.web2.dav.element.rfc2518 import GETContentType
-from twext.web2.dav.resource import TwistedGETContentMD5
-
-
-
-from zope.interface.declarations import implements
-
-def isValidName(name):
- """
- Determine if the given string is a valid name. i.e. does it conflict with
- any of the other entities which may be on the filesystem?
-
- @param name: a name which might be given to a calendar.
- """
- return not name.startswith(".")
-
-
-def hidden(path):
- return path.sibling('.' + path.basename())
-
-
-def writeOperation(thunk):
- # FIXME: tests
- def inner(self, *a, **kw):
- if self._transaction._termination is not None:
- raise RuntimeError(
- "%s.%s is a write operation, but transaction already %s"
- % (self, thunk.__name__, self._transaction._termination))
- return thunk(self, *a, **kw)
- return inner
-
-
-
-class DataStore(LoggingMixIn):
- """
- Generic data store.
- """
-
- _transactionClass = None # Derived class must set this
-
- def __init__(self, path):
- """
- Create a calendar store.
-
- @param path: a L{FilePath} pointing at a directory on disk.
- """
- self._path = path
-
-# if not path.isdir():
- # FIXME: Add DataStoreNotFoundError?
-# raise NotFoundError("No such data store")
-
- def __repr__(self):
- return "<%s: %s>" % (self.__class__.__name__, self._path.path)
-
- def newTransaction(self, name='no name'):
- """
- Create a new transaction.
-
- @see Transaction
- """
- return self._transactionClass(self)
-
-
-
-class _CommitTracker(object):
- """
- Diagnostic tool to find transactions that were never committed.
- """
-
- def __init__(self, name):
- self.name = name
- self.done = False
- self.info = []
-
- def __del__(self):
- if not self.done and self.info:
- print '**** UNCOMMITTED TRANSACTION (%s) BEING GARBAGE COLLECTED ****' % (
- self.name,
- )
- for info in self.info:
- print ' ', info
- print '---- END OF OPERATIONS'
-
-
-
-class DataStoreTransaction(LoggingMixIn):
- """
- In-memory implementation of a data store transaction.
- """
-
- def __init__(self, dataStore, name):
- """
- Initialize a transaction; do not call this directly, instead call
- L{CalendarStore.newTransaction}.
-
- @param calendarStore: The store that created this transaction.
-
- @type calendarStore: L{CalendarStore}
- """
- self._dataStore = dataStore
- self._termination = None
- self._operations = []
- self._postCommitOperations = []
- self._tracker = _CommitTracker(name)
-
-
- def store(self):
- return self._dataStore
-
- def addOperation(self, operation, name):
- self._operations.append(operation)
- self._tracker.info.append(name)
-
-
- def _terminate(self, mode):
- """
- Check to see if this transaction has already been terminated somehow,
- either via committing or aborting, and if not, note that it has been
- terminated.
-
- @param mode: The manner of the termination of this transaction.
-
- @type mode: C{str}
-
- @raise AlreadyFinishedError: This transaction has already been
- terminated.
- """
- if self._termination is not None:
- raise AlreadyFinishedError("already %s" % (self._termination,))
- self._termination = mode
- self._tracker.done = True
-
-
- def abort(self):
- self._terminate("aborted")
-
-
- def commit(self):
- self._terminate("committed")
-
- self.committed = True
- undos = []
-
- for operation in self._operations:
- try:
- undo = operation()
- if undo is not None:
- undos.append(undo)
- except:
- self.log_debug("Undoing DataStoreTransaction")
- for undo in undos:
- try:
- undo()
- except:
- self.log_error("Cannot undo DataStoreTransaction")
- raise
-
- for operation in self._postCommitOperations:
- operation()
-
- def postCommit(self, operation):
- self._postCommitOperations.append(operation)
-
-class FileMetaDataMixin(object):
-
- implements(IDataStoreResource)
-
- def name(self):
- """
- Identify the name of the object
-
- @return: the name of this object.
- @rtype: C{str}
- """
-
- return self._path.basename()
-
- def contentType(self):
- """
- The content type of the object's content.
-
- @rtype: L{MimeType}
- """
- try:
- return self.properties()[PropertyName.fromElement(GETContentType)].mimeType()
- except KeyError:
- return None
-
- def md5(self):
- """
- The MD5 hex digest of this object's content.
-
- @rtype: C{str}
- """
- try:
- return str(self.properties()[PropertyName.fromElement(TwistedGETContentMD5)])
- except KeyError:
- return None
-
- def size(self):
- """
- The octet-size of this object's content.
-
- @rtype: C{int}
- """
- if self._path.exists():
- return self._path.getsize()
- else:
- return 0
-
- def created(self):
- """
- The creation date-time stamp of this object.
-
- @rtype: C{int}
- """
- if self._path.exists():
- return self._path.getmtime() # No creation time on POSIX
- else:
- return None
-
- def modified(self):
- """
- The last modification date-time stamp of this object.
-
- @rtype: C{int}
- """
- if self._path.exists():
- return self._path.getmtime()
- else:
- return None
Copied: CalendarServer/trunk/txdav/base/datastore/file.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/datastore/file.py)
===================================================================
--- CalendarServer/trunk/txdav/base/datastore/file.py (rev 0)
+++ CalendarServer/trunk/txdav/base/datastore/file.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,259 @@
+# -*- test-case-name: txdav -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+
+"""
+Common utility functions for a file based datastore.
+"""
+
+from twext.python.log import LoggingMixIn
+from txdav.idav import IDataStoreResource
+from txdav.idav import AlreadyFinishedError
+from txdav.base.propertystore.base import PropertyName
+
+from twext.web2.dav.element.rfc2518 import GETContentType
+from twext.web2.dav.resource import TwistedGETContentMD5
+
+
+
+from zope.interface.declarations import implements
+
+def isValidName(name):
+ """
+ Determine if the given string is a valid name. i.e. does it conflict with
+ any of the other entities which may be on the filesystem?
+
+ @param name: a name which might be given to a calendar.
+ """
+ return not name.startswith(".")
+
+
+def hidden(path):
+ return path.sibling('.' + path.basename())
+
+
+def writeOperation(thunk):
+ # FIXME: tests
+ def inner(self, *a, **kw):
+ if self._transaction._termination is not None:
+ raise RuntimeError(
+ "%s.%s is a write operation, but transaction already %s"
+ % (self, thunk.__name__, self._transaction._termination))
+ return thunk(self, *a, **kw)
+ return inner
+
+
+
+class DataStore(LoggingMixIn):
+ """
+ Generic data store.
+ """
+
+ _transactionClass = None # Derived class must set this
+
+ def __init__(self, path):
+ """
+ Create a calendar store.
+
+ @param path: a L{FilePath} pointing at a directory on disk.
+ """
+ self._path = path
+
+# if not path.isdir():
+ # FIXME: Add DataStoreNotFoundError?
+# raise NotFoundError("No such data store")
+
+ def __repr__(self):
+ return "<%s: %s>" % (self.__class__.__name__, self._path.path)
+
+ def newTransaction(self, name='no name'):
+ """
+ Create a new transaction.
+
+ @see Transaction
+ """
+ return self._transactionClass(self)
+
+
+
+class _CommitTracker(object):
+ """
+ Diagnostic tool to find transactions that were never committed.
+ """
+
+ def __init__(self, name):
+ self.name = name
+ self.done = False
+ self.info = []
+
+ def __del__(self):
+ if not self.done and self.info:
+ print '**** UNCOMMITTED TRANSACTION (%s) BEING GARBAGE COLLECTED ****' % (
+ self.name,
+ )
+ for info in self.info:
+ print ' ', info
+ print '---- END OF OPERATIONS'
+
+
+
+class DataStoreTransaction(LoggingMixIn):
+ """
+ In-memory implementation of a data store transaction.
+ """
+
+ def __init__(self, dataStore, name):
+ """
+ Initialize a transaction; do not call this directly, instead call
+ L{CalendarStore.newTransaction}.
+
+ @param calendarStore: The store that created this transaction.
+
+ @type calendarStore: L{CalendarStore}
+ """
+ self._dataStore = dataStore
+ self._termination = None
+ self._operations = []
+ self._postCommitOperations = []
+ self._tracker = _CommitTracker(name)
+
+
+ def store(self):
+ return self._dataStore
+
+ def addOperation(self, operation, name):
+ self._operations.append(operation)
+ self._tracker.info.append(name)
+
+
+ def _terminate(self, mode):
+ """
+ Check to see if this transaction has already been terminated somehow,
+ either via committing or aborting, and if not, note that it has been
+ terminated.
+
+ @param mode: The manner of the termination of this transaction.
+
+ @type mode: C{str}
+
+ @raise AlreadyFinishedError: This transaction has already been
+ terminated.
+ """
+ if self._termination is not None:
+ raise AlreadyFinishedError("already %s" % (self._termination,))
+ self._termination = mode
+ self._tracker.done = True
+
+
+ def abort(self):
+ self._terminate("aborted")
+
+
+ def commit(self):
+ self._terminate("committed")
+
+ self.committed = True
+ undos = []
+
+ for operation in self._operations:
+ try:
+ undo = operation()
+ if undo is not None:
+ undos.append(undo)
+ except:
+ self.log_debug("Undoing DataStoreTransaction")
+ for undo in undos:
+ try:
+ undo()
+ except:
+ self.log_error("Cannot undo DataStoreTransaction")
+ raise
+
+ for operation in self._postCommitOperations:
+ operation()
+
+ def postCommit(self, operation):
+ self._postCommitOperations.append(operation)
+
+class FileMetaDataMixin(object):
+
+ implements(IDataStoreResource)
+
+ def name(self):
+ """
+ Identify the name of the object
+
+ @return: the name of this object.
+ @rtype: C{str}
+ """
+
+ return self._path.basename()
+
+ def contentType(self):
+ """
+ The content type of the object's content.
+
+ @rtype: L{MimeType}
+ """
+ try:
+ return self.properties()[PropertyName.fromElement(GETContentType)].mimeType()
+ except KeyError:
+ return None
+
+ def md5(self):
+ """
+ The MD5 hex digest of this object's content.
+
+ @rtype: C{str}
+ """
+ try:
+ return str(self.properties()[PropertyName.fromElement(TwistedGETContentMD5)])
+ except KeyError:
+ return None
+
+ def size(self):
+ """
+ The octet-size of this object's content.
+
+ @rtype: C{int}
+ """
+ if self._path.exists():
+ return self._path.getsize()
+ else:
+ return 0
+
+ def created(self):
+ """
+ The creation date-time stamp of this object.
+
+ @rtype: C{int}
+ """
+ if self._path.exists():
+ return self._path.getmtime() # No creation time on POSIX
+ else:
+ return None
+
+ def modified(self):
+ """
+ The last modification date-time stamp of this object.
+
+ @rtype: C{int}
+ """
+ if self._path.exists():
+ return self._path.getmtime()
+ else:
+ return None
Deleted: CalendarServer/trunk/txdav/base/datastore/sql.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/datastore/sql.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/datastore/sql.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,85 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Logic common to SQL implementations.
-"""
-
-from inspect import getargspec
-
-def _getarg(argname, argspec, args, kw):
- """
- Get an argument from some arguments.
-
- @param argname: The name of the argument to retrieve.
-
- @param argspec: The result of L{inspect.getargspec}.
-
- @param args: positional arguments passed to the function specified by
- argspec.
-
- @param kw: keyword arguments passed to the function specified by
- argspec.
-
- @return: The value of the argument named by 'argname'.
- """
- argnames = argspec[0]
- try:
- argpos = argnames.index(argname)
- except ValueError:
- argpos = None
- if argpos is not None:
- if len(args) > argpos:
- return args[argpos]
- if argname in kw:
- return kw[argname]
- else:
- raise TypeError("could not find key argument %r in %r/%r (%r)" %
- (argname, args, kw, argpos)
- )
-
-
-
-def memoized(keyArgument, memoAttribute):
- """
- Decorator which memoizes the result of a method on that method's instance.
-
- @param keyArgument: The name of the 'key' argument.
-
- @type keyArgument: C{str}
-
- @param memoAttribute: The name of the attribute on the instance which
- should be used for memoizing the result of this method; the attribute
- itself must be a dictionary.
-
- @type memoAttribute: C{str}
- """
- def decorate(thunk):
- spec = getargspec(thunk)
- def outer(*a, **kw):
- self = a[0]
- memo = getattr(self, memoAttribute)
- key = _getarg(keyArgument, spec, a, kw)
- if key in memo:
- return memo[key]
- result = thunk(*a, **kw)
- if result is not None:
- memo[key] = result
- return result
- return outer
- return decorate
-
-
Copied: CalendarServer/trunk/txdav/base/datastore/sql.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/datastore/sql.py)
===================================================================
--- CalendarServer/trunk/txdav/base/datastore/sql.py (rev 0)
+++ CalendarServer/trunk/txdav/base/datastore/sql.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,85 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Logic common to SQL implementations.
+"""
+
+from inspect import getargspec
+
+def _getarg(argname, argspec, args, kw):
+ """
+ Get an argument from some arguments.
+
+ @param argname: The name of the argument to retrieve.
+
+ @param argspec: The result of L{inspect.getargspec}.
+
+ @param args: positional arguments passed to the function specified by
+ argspec.
+
+ @param kw: keyword arguments passed to the function specified by
+ argspec.
+
+ @return: The value of the argument named by 'argname'.
+ """
+ argnames = argspec[0]
+ try:
+ argpos = argnames.index(argname)
+ except ValueError:
+ argpos = None
+ if argpos is not None:
+ if len(args) > argpos:
+ return args[argpos]
+ if argname in kw:
+ return kw[argname]
+ else:
+ raise TypeError("could not find key argument %r in %r/%r (%r)" %
+ (argname, args, kw, argpos)
+ )
+
+
+
+def memoized(keyArgument, memoAttribute):
+ """
+ Decorator which memoizes the result of a method on that method's instance.
+
+ @param keyArgument: The name of the 'key' argument.
+
+ @type keyArgument: C{str}
+
+ @param memoAttribute: The name of the attribute on the instance which
+ should be used for memoizing the result of this method; the attribute
+ itself must be a dictionary.
+
+ @type memoAttribute: C{str}
+ """
+ def decorate(thunk):
+ spec = getargspec(thunk)
+ def outer(*a, **kw):
+ self = a[0]
+ memo = getattr(self, memoAttribute)
+ key = _getarg(keyArgument, spec, a, kw)
+ if key in memo:
+ return memo[key]
+ result = thunk(*a, **kw)
+ if result is not None:
+ memo[key] = result
+ return result
+ return outer
+ return decorate
+
+
Deleted: CalendarServer/trunk/txdav/base/datastore/subpostgres.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/datastore/subpostgres.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/datastore/subpostgres.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,449 +0,0 @@
-# -*- test-case-name: txdav.base.datastore.test.test_subpostgres -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Run and manage PostgreSQL as a subprocess.
-"""
-import os
-import pwd
-from hashlib import md5
-
-from twisted.python.procutils import which
-from twisted.internet.protocol import ProcessProtocol
-
-from twisted.python.reflect import namedAny
-from twisted.python import log
-from twext.python.filepath import CachingFilePath
-
-
-pgdb = namedAny("pgdb")
-
-from twisted.protocols.basic import LineReceiver
-from twisted.internet import reactor
-from twisted.internet.defer import Deferred
-
-from twisted.application.service import MultiService
-
-
-# This appears in the postgres log to indicate that it is accepting
-# connections.
-_MAGIC_READY_COOKIE = "database system is ready to accept connections"
-
-
-class DiagnosticCursorWrapper(object):
- """
- Diagnostic wrapper around a DB-API 2.0 cursor for debugging connection
- status.
- """
-
- def __init__(self, realCursor, connectionWrapper):
- self.realCursor = realCursor
- self.connectionWrapper = connectionWrapper
-
-
- @property
- def rowcount(self):
- return self.realCursor.rowcount
-
-
- @property
- def description(self):
- return self.realCursor.description
-
-
- def execute(self, sql, args=()):
- self.connectionWrapper.state = 'executing %r' % (sql,)
- self.realCursor.execute(sql, args)
-
-
- def close(self):
- self.realCursor.close()
-
-
- def fetchall(self):
- return self.realCursor.fetchall()
-
-
-
-class DiagnosticConnectionWrapper(object):
- """
- Diagnostic wrapper around a DB-API 2.0 connection for debugging connection
- status.
- """
-
- def __init__(self, realConnection, label):
- self.realConnection = realConnection
- self.label = label
- self.state = 'idle (start)'
-
-
- def cursor(self):
- return DiagnosticCursorWrapper(self.realConnection.cursor(), self)
-
-
- def close(self):
- self.realConnection.close()
- self.state = 'closed'
-
-
- def commit(self):
- self.realConnection.commit()
- self.state = 'idle (after commit)'
-
-
- def rollback(self):
- self.realConnection.rollback()
- self.state = 'idle (after rollback)'
-
-
-
-class _PostgresMonitor(ProcessProtocol):
- """
- A monitoring protocol which watches the postgres subprocess.
- """
-
- def __init__(self, svc=None):
- self.lineReceiver = LineReceiver()
- self.lineReceiver.delimiter = '\n'
- self.lineReceiver.lineReceived = self.lineReceived
- self.svc = svc
- self.isReady = False
- self.completionDeferred = Deferred()
-
-
- def lineReceived(self, line):
- if self.svc is None:
- return
- if not self.isReady:
- if _MAGIC_READY_COOKIE in line:
- self.svc.ready()
-
-
- disconnecting = False
- def connectionMade(self):
- self.lineReceiver.makeConnection(self)
-
-
- def outReceived(self, out):
- log.msg("received postgres stdout %r" % (out,))
- # self.lineReceiver.dataReceived(out)
-
-
- def errReceived(self, err):
- log.msg("received postgres stderr %r" % (err,))
- self.lineReceiver.dataReceived(err)
-
-
- def processEnded(self, reason):
- log.msg("postgres process ended %r" % (reason,))
- self.lineReceiver.connectionLost(reason)
- self.completionDeferred.callback(None)
-
-
-class ErrorOutput(Exception):
- """
- The process produced some error output and exited with a non-zero exit
- code.
- """
-
-
-class CapturingProcessProtocol(ProcessProtocol):
- """
- A L{ProcessProtocol} that captures its output and error.
-
- @ivar output: a C{list} of all C{str}s received to stderr.
-
- @ivar error: a C{list} of all C{str}s received to stderr.
- """
-
- def __init__(self, deferred, inputData):
- """
- Initialize a L{CapturingProcessProtocol}.
-
- @param deferred: the L{Deferred} to fire when the process is complete.
-
- @param inputData: a C{str} to feed to the subprocess's stdin.
- """
- self.deferred = deferred
- self.input = inputData
- self.output = []
- self.error = []
-
-
- def connectionMade(self):
- """
- The process started; feed its input on stdin.
- """
- if self.input is not None:
- self.transport.write(self.input)
- self.transport.closeStdin()
-
-
- def outReceived(self, data):
- """
- Some output was received on stdout.
- """
- self.output.append(data)
-
- def errReceived(self, data):
- """
- Some output was received on stderr.
- """
- self.output.append(data)
-
-
- def processEnded(self, why):
- """
- The process is over, fire the Deferred with the output.
- """
- self.deferred.callback(''.join(self.output))
-
-
-class PostgresService(MultiService):
-
- def __init__(self, dataStoreDirectory, subServiceFactory,
- schema, databaseName='subpostgres', resetSchema=False,
- logFile="postgres.log", testMode=False,
- uid=None, gid=None):
- """
- Initialize a L{PostgresService} pointed at a data store directory.
-
- @param dataStoreDirectory: the directory to
- @type dataStoreDirectory: L{twext.python.filepath.CachingFilePath}
-
- @param subServiceFactory: a 1-arg callable that will be called with a
- 1-arg callable which returns a DB-API cursor.
- @type subServiceFactory: C{callable}
- """
- MultiService.__init__(self)
- self.subServiceFactory = subServiceFactory
- self.dataStoreDirectory = dataStoreDirectory
- self.resetSchema = resetSchema
-
- if os.getuid() == 0:
- socketRoot = "/var/run"
- else:
- socketRoot = "/tmp"
- self.socketDir = CachingFilePath("%s/ccs_postgres_%s/" %
- (socketRoot, md5(dataStoreDirectory.path).hexdigest()))
- self.databaseName = databaseName
- self.logFile = logFile
- self.uid = uid
- self.gid = gid
- self.schema = schema
- self.monitor = None
- self.openConnections = []
-
- # FIXME: By default there is very little (4MB) shared memory available,
- # so at the moment I am lowering these postgres config options to allow
- # multiple servers to run. We might want to look into raising
- # kern.sysv.shmmax.
- # See: http://www.postgresql.org/docs/8.4/static/kernel-resources.html
- if testMode:
- self.sharedBuffers = 16
- self.maxConnections = 2
- else:
- self.sharedBuffers = 30
- self.maxConnections = 20
-
-
- def produceConnection(self, label="<unlabeled>", databaseName=None):
- """
- Produce a DB-API 2.0 connection pointed at this database.
- """
- if databaseName is None:
- databaseName = self.databaseName
-
- if self.uid is not None:
- dsn = "%s:dbname=%s:%s" % (self.socketDir.path, databaseName,
- pwd.getpwuid(self.uid).pw_name)
- else:
- dsn = "%s:dbname=%s" % (self.socketDir.path, databaseName)
- connection = pgdb.connect(dsn)
-
- w = DiagnosticConnectionWrapper(connection, label)
- c = w.cursor()
- # Turn on standard conforming strings. This option is _required_ if
- # you want to get correct behavior out of parameter-passing with the
- # pgdb module. If it is not set then the server is potentially
- # vulnerable to certain types of SQL injection.
- c.execute("set standard_conforming_strings=on")
-
- # Abort any second that takes more than 30 seconds (30000ms) to
- # execute. This is necessary as a temporary workaround since it's
- # hypothetically possible that different database operations could
- # block each other, while executing SQL in the same process (in the
- # same thread, since SQL executes in the main thread now). It's
- # preferable to see some exceptions while we're in this state than to
- # have the entire worker process hang.
- c.execute("set statement_timeout=30000")
- w.commit()
- c.close()
- return w
-
-
- def ready(self):
- """
- Subprocess is ready. Time to initialize the subservice.
- """
- createDatabaseConn = self.produceConnection(
- 'schema creation', 'postgres'
- )
- createDatabaseCursor = createDatabaseConn.cursor()
- createDatabaseCursor.execute("commit")
-
- if self.resetSchema:
- try:
- createDatabaseCursor.execute(
- "drop database %s" % (self.databaseName)
- )
- except pgdb.DatabaseError:
- pass
-
- try:
- createDatabaseCursor.execute(
- "create database %s" % (self.databaseName)
- )
- except:
- execSchema = False
- else:
- execSchema = True
-
- createDatabaseCursor.close()
- createDatabaseConn.close()
-
- if execSchema:
- connection = self.produceConnection()
- cursor = connection.cursor()
- cursor.execute(self.schema)
- connection.commit()
- connection.close()
-
- connection = self.produceConnection()
- cursor = connection.cursor()
-
- self.subServiceFactory(self.produceConnection).setServiceParent(self)
-
-
- def pauseMonitor(self):
- """
- Pause monitoring. This is a testing hook for when (if) we are
- continuously monitoring output from the 'postgres' process.
- """
-# for pipe in self.monitor.transport.pipes.values():
-# pipe.stopReading()
-# pipe.stopWriting()
-
-
- def unpauseMonitor(self):
- """
- Unpause monitoring.
-
- @see: L{pauseMonitor}
- """
-# for pipe in self.monitor.transport.pipes.values():
-# pipe.startReading()
-# pipe.startWriting()
-
-
- def startDatabase(self):
- """
- Start the database and initialize the subservice.
- """
- monitor = _PostgresMonitor(self)
- pg_ctl = which("pg_ctl")[0]
- # check consistency of initdb and postgres?
- reactor.spawnProcess(
- monitor, pg_ctl,
- [
- pg_ctl,
- "start",
- "-l", self.logFile,
- "-w",
- # XXX what are the quoting rules for '-o'? do I need to repr()
- # the path here?
- "-o", "-c listen_addresses='' -k '%s' -c standard_conforming_strings=on -c shared_buffers=%d -c max_connections=%d"
- % (self.socketDir.path, self.sharedBuffers, self.maxConnections),
- ],
- self.env,
- uid=self.uid, gid=self.gid,
- )
- self.monitor = monitor
- def gotReady(result):
- self.ready()
- def reportit(f):
- log.err(f)
- self.monitor.completionDeferred.addCallback(
- gotReady).addErrback(reportit)
-
-
- def startService(self):
- MultiService.startService(self)
- clusterDir = self.dataStoreDirectory.child("cluster")
- workingDir = self.dataStoreDirectory.child("working")
- env = self.env = os.environ.copy()
- env.update(PGDATA=clusterDir.path,
- PGHOST=self.socketDir.path)
- initdb = which("initdb")[0]
- if not self.socketDir.isdir():
- self.socketDir.createDirectory()
- if self.uid and self.gid:
- os.chown(self.socketDir.path, self.uid, self.gid)
- if self.dataStoreDirectory.isdir():
- self.startDatabase()
- else:
- self.dataStoreDirectory.createDirectory()
- workingDir.createDirectory()
- if self.uid and self.gid:
- os.chown(self.dataStoreDirectory.path, self.uid, self.gid)
- os.chown(workingDir.path, self.uid, self.gid)
- dbInited = Deferred()
- reactor.spawnProcess(
- CapturingProcessProtocol(dbInited, None),
- initdb, [initdb], env, workingDir.path,
- uid=self.uid, gid=self.gid,
- )
- def doCreate(result):
- self.startDatabase()
- dbInited.addCallback(doCreate)
-
-
- def stopService(self):
- """
- Stop all child services, then stop the subprocess, if it's running.
- """
- d = MultiService.stopService(self)
- def superStopped(result):
- # Probably want to stop and wait for startup if that hasn't
- # completed yet...
- monitor = _PostgresMonitor()
- pg_ctl = which("pg_ctl")[0]
- reactor.spawnProcess(monitor, pg_ctl,
- [pg_ctl, '-l', 'logfile', 'stop'],
- self.env,
- uid=self.uid, gid=self.gid,
- )
- return monitor.completionDeferred
- return d.addCallback(superStopped)
-
-# def maybeStopSubprocess(result):
-# if self.monitor is not None:
-# self.monitor.transport.signalProcess("INT")
-# return self.monitor.completionDeferred
-# return result
-# d.addCallback(maybeStopSubprocess)
-# return d
Copied: CalendarServer/trunk/txdav/base/datastore/subpostgres.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/datastore/subpostgres.py)
===================================================================
--- CalendarServer/trunk/txdav/base/datastore/subpostgres.py (rev 0)
+++ CalendarServer/trunk/txdav/base/datastore/subpostgres.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,449 @@
+# -*- test-case-name: txdav.base.datastore.test.test_subpostgres -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Run and manage PostgreSQL as a subprocess.
+"""
+import os
+import pwd
+from hashlib import md5
+
+from twisted.python.procutils import which
+from twisted.internet.protocol import ProcessProtocol
+
+from twisted.python.reflect import namedAny
+from twisted.python import log
+from twext.python.filepath import CachingFilePath
+
+
+pgdb = namedAny("pgdb")
+
+from twisted.protocols.basic import LineReceiver
+from twisted.internet import reactor
+from twisted.internet.defer import Deferred
+
+from twisted.application.service import MultiService
+
+
+# This appears in the postgres log to indicate that it is accepting
+# connections.
+_MAGIC_READY_COOKIE = "database system is ready to accept connections"
+
+
+class DiagnosticCursorWrapper(object):
+ """
+ Diagnostic wrapper around a DB-API 2.0 cursor for debugging connection
+ status.
+ """
+
+ def __init__(self, realCursor, connectionWrapper):
+ self.realCursor = realCursor
+ self.connectionWrapper = connectionWrapper
+
+
+ @property
+ def rowcount(self):
+ return self.realCursor.rowcount
+
+
+ @property
+ def description(self):
+ return self.realCursor.description
+
+
+ def execute(self, sql, args=()):
+ self.connectionWrapper.state = 'executing %r' % (sql,)
+ self.realCursor.execute(sql, args)
+
+
+ def close(self):
+ self.realCursor.close()
+
+
+ def fetchall(self):
+ return self.realCursor.fetchall()
+
+
+
+class DiagnosticConnectionWrapper(object):
+ """
+ Diagnostic wrapper around a DB-API 2.0 connection for debugging connection
+ status.
+ """
+
+ def __init__(self, realConnection, label):
+ self.realConnection = realConnection
+ self.label = label
+ self.state = 'idle (start)'
+
+
+ def cursor(self):
+ return DiagnosticCursorWrapper(self.realConnection.cursor(), self)
+
+
+ def close(self):
+ self.realConnection.close()
+ self.state = 'closed'
+
+
+ def commit(self):
+ self.realConnection.commit()
+ self.state = 'idle (after commit)'
+
+
+ def rollback(self):
+ self.realConnection.rollback()
+ self.state = 'idle (after rollback)'
+
+
+
+class _PostgresMonitor(ProcessProtocol):
+ """
+ A monitoring protocol which watches the postgres subprocess.
+ """
+
+ def __init__(self, svc=None):
+ self.lineReceiver = LineReceiver()
+ self.lineReceiver.delimiter = '\n'
+ self.lineReceiver.lineReceived = self.lineReceived
+ self.svc = svc
+ self.isReady = False
+ self.completionDeferred = Deferred()
+
+
+ def lineReceived(self, line):
+ if self.svc is None:
+ return
+ if not self.isReady:
+ if _MAGIC_READY_COOKIE in line:
+ self.svc.ready()
+
+
+ disconnecting = False
+ def connectionMade(self):
+ self.lineReceiver.makeConnection(self)
+
+
+ def outReceived(self, out):
+ log.msg("received postgres stdout %r" % (out,))
+ # self.lineReceiver.dataReceived(out)
+
+
+ def errReceived(self, err):
+ log.msg("received postgres stderr %r" % (err,))
+ self.lineReceiver.dataReceived(err)
+
+
+ def processEnded(self, reason):
+ log.msg("postgres process ended %r" % (reason,))
+ self.lineReceiver.connectionLost(reason)
+ self.completionDeferred.callback(None)
+
+
+class ErrorOutput(Exception):
+ """
+ The process produced some error output and exited with a non-zero exit
+ code.
+ """
+
+
+class CapturingProcessProtocol(ProcessProtocol):
+ """
+ A L{ProcessProtocol} that captures its output and error.
+
+ @ivar output: a C{list} of all C{str}s received to stderr.
+
+ @ivar error: a C{list} of all C{str}s received to stderr.
+ """
+
+ def __init__(self, deferred, inputData):
+ """
+ Initialize a L{CapturingProcessProtocol}.
+
+ @param deferred: the L{Deferred} to fire when the process is complete.
+
+ @param inputData: a C{str} to feed to the subprocess's stdin.
+ """
+ self.deferred = deferred
+ self.input = inputData
+ self.output = []
+ self.error = []
+
+
+ def connectionMade(self):
+ """
+ The process started; feed its input on stdin.
+ """
+ if self.input is not None:
+ self.transport.write(self.input)
+ self.transport.closeStdin()
+
+
+ def outReceived(self, data):
+ """
+ Some output was received on stdout.
+ """
+ self.output.append(data)
+
+ def errReceived(self, data):
+ """
+ Some output was received on stderr.
+ """
+ self.output.append(data)
+
+
+ def processEnded(self, why):
+ """
+ The process is over, fire the Deferred with the output.
+ """
+ self.deferred.callback(''.join(self.output))
+
+
+class PostgresService(MultiService):
+
+ def __init__(self, dataStoreDirectory, subServiceFactory,
+ schema, databaseName='subpostgres', resetSchema=False,
+ logFile="postgres.log", testMode=False,
+ uid=None, gid=None):
+ """
+ Initialize a L{PostgresService} pointed at a data store directory.
+
+ @param dataStoreDirectory: the directory to
+ @type dataStoreDirectory: L{twext.python.filepath.CachingFilePath}
+
+ @param subServiceFactory: a 1-arg callable that will be called with a
+ 1-arg callable which returns a DB-API cursor.
+ @type subServiceFactory: C{callable}
+ """
+ MultiService.__init__(self)
+ self.subServiceFactory = subServiceFactory
+ self.dataStoreDirectory = dataStoreDirectory
+ self.resetSchema = resetSchema
+
+ if os.getuid() == 0:
+ socketRoot = "/var/run"
+ else:
+ socketRoot = "/tmp"
+ self.socketDir = CachingFilePath("%s/ccs_postgres_%s/" %
+ (socketRoot, md5(dataStoreDirectory.path).hexdigest()))
+ self.databaseName = databaseName
+ self.logFile = logFile
+ self.uid = uid
+ self.gid = gid
+ self.schema = schema
+ self.monitor = None
+ self.openConnections = []
+
+ # FIXME: By default there is very little (4MB) shared memory available,
+ # so at the moment I am lowering these postgres config options to allow
+ # multiple servers to run. We might want to look into raising
+ # kern.sysv.shmmax.
+ # See: http://www.postgresql.org/docs/8.4/static/kernel-resources.html
+ if testMode:
+ self.sharedBuffers = 16
+ self.maxConnections = 2
+ else:
+ self.sharedBuffers = 30
+ self.maxConnections = 20
+
+
+ def produceConnection(self, label="<unlabeled>", databaseName=None):
+ """
+ Produce a DB-API 2.0 connection pointed at this database.
+ """
+ if databaseName is None:
+ databaseName = self.databaseName
+
+ if self.uid is not None:
+ dsn = "%s:dbname=%s:%s" % (self.socketDir.path, databaseName,
+ pwd.getpwuid(self.uid).pw_name)
+ else:
+ dsn = "%s:dbname=%s" % (self.socketDir.path, databaseName)
+ connection = pgdb.connect(dsn)
+
+ w = DiagnosticConnectionWrapper(connection, label)
+ c = w.cursor()
+ # Turn on standard conforming strings. This option is _required_ if
+ # you want to get correct behavior out of parameter-passing with the
+ # pgdb module. If it is not set then the server is potentially
+ # vulnerable to certain types of SQL injection.
+ c.execute("set standard_conforming_strings=on")
+
+ # Abort any second that takes more than 30 seconds (30000ms) to
+ # execute. This is necessary as a temporary workaround since it's
+ # hypothetically possible that different database operations could
+ # block each other, while executing SQL in the same process (in the
+ # same thread, since SQL executes in the main thread now). It's
+ # preferable to see some exceptions while we're in this state than to
+ # have the entire worker process hang.
+ c.execute("set statement_timeout=30000")
+ w.commit()
+ c.close()
+ return w
+
+
+ def ready(self):
+ """
+ Subprocess is ready. Time to initialize the subservice.
+ """
+ createDatabaseConn = self.produceConnection(
+ 'schema creation', 'postgres'
+ )
+ createDatabaseCursor = createDatabaseConn.cursor()
+ createDatabaseCursor.execute("commit")
+
+ if self.resetSchema:
+ try:
+ createDatabaseCursor.execute(
+ "drop database %s" % (self.databaseName)
+ )
+ except pgdb.DatabaseError:
+ pass
+
+ try:
+ createDatabaseCursor.execute(
+ "create database %s" % (self.databaseName)
+ )
+ except:
+ execSchema = False
+ else:
+ execSchema = True
+
+ createDatabaseCursor.close()
+ createDatabaseConn.close()
+
+ if execSchema:
+ connection = self.produceConnection()
+ cursor = connection.cursor()
+ cursor.execute(self.schema)
+ connection.commit()
+ connection.close()
+
+ connection = self.produceConnection()
+ cursor = connection.cursor()
+
+ self.subServiceFactory(self.produceConnection).setServiceParent(self)
+
+
+ def pauseMonitor(self):
+ """
+ Pause monitoring. This is a testing hook for when (if) we are
+ continuously monitoring output from the 'postgres' process.
+ """
+# for pipe in self.monitor.transport.pipes.values():
+# pipe.stopReading()
+# pipe.stopWriting()
+
+
+ def unpauseMonitor(self):
+ """
+ Unpause monitoring.
+
+ @see: L{pauseMonitor}
+ """
+# for pipe in self.monitor.transport.pipes.values():
+# pipe.startReading()
+# pipe.startWriting()
+
+
+ def startDatabase(self):
+ """
+ Start the database and initialize the subservice.
+ """
+ monitor = _PostgresMonitor(self)
+ pg_ctl = which("pg_ctl")[0]
+ # check consistency of initdb and postgres?
+ reactor.spawnProcess(
+ monitor, pg_ctl,
+ [
+ pg_ctl,
+ "start",
+ "-l", self.logFile,
+ "-w",
+ # XXX what are the quoting rules for '-o'? do I need to repr()
+ # the path here?
+ "-o", "-c listen_addresses='' -k '%s' -c standard_conforming_strings=on -c shared_buffers=%d -c max_connections=%d"
+ % (self.socketDir.path, self.sharedBuffers, self.maxConnections),
+ ],
+ self.env,
+ uid=self.uid, gid=self.gid,
+ )
+ self.monitor = monitor
+ def gotReady(result):
+ self.ready()
+ def reportit(f):
+ log.err(f)
+ self.monitor.completionDeferred.addCallback(
+ gotReady).addErrback(reportit)
+
+
+ def startService(self):
+ MultiService.startService(self)
+ clusterDir = self.dataStoreDirectory.child("cluster")
+ workingDir = self.dataStoreDirectory.child("working")
+ env = self.env = os.environ.copy()
+ env.update(PGDATA=clusterDir.path,
+ PGHOST=self.socketDir.path)
+ initdb = which("initdb")[0]
+ if not self.socketDir.isdir():
+ self.socketDir.createDirectory()
+ if self.uid and self.gid:
+ os.chown(self.socketDir.path, self.uid, self.gid)
+ if self.dataStoreDirectory.isdir():
+ self.startDatabase()
+ else:
+ self.dataStoreDirectory.createDirectory()
+ workingDir.createDirectory()
+ if self.uid and self.gid:
+ os.chown(self.dataStoreDirectory.path, self.uid, self.gid)
+ os.chown(workingDir.path, self.uid, self.gid)
+ dbInited = Deferred()
+ reactor.spawnProcess(
+ CapturingProcessProtocol(dbInited, None),
+ initdb, [initdb], env, workingDir.path,
+ uid=self.uid, gid=self.gid,
+ )
+ def doCreate(result):
+ self.startDatabase()
+ dbInited.addCallback(doCreate)
+
+
+ def stopService(self):
+ """
+ Stop all child services, then stop the subprocess, if it's running.
+ """
+ d = MultiService.stopService(self)
+ def superStopped(result):
+ # Probably want to stop and wait for startup if that hasn't
+ # completed yet...
+ monitor = _PostgresMonitor()
+ pg_ctl = which("pg_ctl")[0]
+ reactor.spawnProcess(monitor, pg_ctl,
+ [pg_ctl, '-l', 'logfile', 'stop'],
+ self.env,
+ uid=self.uid, gid=self.gid,
+ )
+ return monitor.completionDeferred
+ return d.addCallback(superStopped)
+
+# def maybeStopSubprocess(result):
+# if self.monitor is not None:
+# self.monitor.transport.signalProcess("INT")
+# return self.monitor.completionDeferred
+# return result
+# d.addCallback(maybeStopSubprocess)
+# return d
Deleted: CalendarServer/trunk/txdav/base/datastore/test/__init__.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/datastore/test/__init__.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/datastore/test/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,16 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
Copied: CalendarServer/trunk/txdav/base/datastore/test/__init__.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/datastore/test/__init__.py)
===================================================================
--- CalendarServer/trunk/txdav/base/datastore/test/__init__.py (rev 0)
+++ CalendarServer/trunk/txdav/base/datastore/test/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,16 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
Deleted: CalendarServer/trunk/txdav/base/datastore/test/test_subpostgres.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/datastore/test/test_subpostgres.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/datastore/test/test_subpostgres.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,86 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Tests for txdav.base.datastore.subpostgres.
-"""
-
-from twisted.trial.unittest import TestCase
-
-from twext.python.filepath import CachingFilePath
-
-from txdav.base.datastore.subpostgres import PostgresService
-from twisted.internet.defer import inlineCallbacks, Deferred
-from twisted.application.service import Service
-
-class SubprocessStartup(TestCase):
- """
- Tests for starting and stopping the subprocess.
- """
-
- @inlineCallbacks
- def test_startService(self):
- """
- Assuming a properly configured environment ($PATH points at an 'initdb'
- and 'postgres', $PYTHONPATH includes pgdb), starting a
- L{PostgresService} will start the service passed to it, after executing
- the schema.
- """
-
- test = self
- class SimpleService(Service):
-
- instances = []
- rows = []
- ready = Deferred()
-
- def __init__(self, connectionFactory):
- self.connection = connectionFactory()
- test.addCleanup(self.connection.close)
- self.instances.append(self)
-
-
- def startService(self):
- cursor = self.connection.cursor()
- try:
- cursor.execute(
- "insert into test_dummy_table values ('dummy')"
- )
- except:
- self.ready.errback()
- else:
- self.ready.callback(None)
- finally:
- cursor.close()
-
-
- dbPath = "../_postgres_test_db"
- svc = PostgresService(
- CachingFilePath(dbPath),
- SimpleService,
- "create table TEST_DUMMY_TABLE (stub varchar)",
- "dummy_db",
- testMode=True
- )
-
- svc.startService()
- self.addCleanup(svc.stopService)
- yield SimpleService.ready
- connection = SimpleService.instances[0].connection
- cursor = connection.cursor()
- cursor.execute("select * from test_dummy_table")
- values = cursor.fetchall()
- self.assertEquals(values, [["dummy"]])
Copied: CalendarServer/trunk/txdav/base/datastore/test/test_subpostgres.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/datastore/test/test_subpostgres.py)
===================================================================
--- CalendarServer/trunk/txdav/base/datastore/test/test_subpostgres.py (rev 0)
+++ CalendarServer/trunk/txdav/base/datastore/test/test_subpostgres.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,86 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Tests for txdav.base.datastore.subpostgres.
+"""
+
+from twisted.trial.unittest import TestCase
+
+from twext.python.filepath import CachingFilePath
+
+from txdav.base.datastore.subpostgres import PostgresService
+from twisted.internet.defer import inlineCallbacks, Deferred
+from twisted.application.service import Service
+
+class SubprocessStartup(TestCase):
+ """
+ Tests for starting and stopping the subprocess.
+ """
+
+ @inlineCallbacks
+ def test_startService(self):
+ """
+ Assuming a properly configured environment ($PATH points at an 'initdb'
+ and 'postgres', $PYTHONPATH includes pgdb), starting a
+ L{PostgresService} will start the service passed to it, after executing
+ the schema.
+ """
+
+ test = self
+ class SimpleService(Service):
+
+ instances = []
+ rows = []
+ ready = Deferred()
+
+ def __init__(self, connectionFactory):
+ self.connection = connectionFactory()
+ test.addCleanup(self.connection.close)
+ self.instances.append(self)
+
+
+ def startService(self):
+ cursor = self.connection.cursor()
+ try:
+ cursor.execute(
+ "insert into test_dummy_table values ('dummy')"
+ )
+ except:
+ self.ready.errback()
+ else:
+ self.ready.callback(None)
+ finally:
+ cursor.close()
+
+
+ dbPath = "../_postgres_test_db"
+ svc = PostgresService(
+ CachingFilePath(dbPath),
+ SimpleService,
+ "create table TEST_DUMMY_TABLE (stub varchar)",
+ "dummy_db",
+ testMode=True
+ )
+
+ svc.startService()
+ self.addCleanup(svc.stopService)
+ yield SimpleService.ready
+ connection = SimpleService.instances[0].connection
+ cursor = connection.cursor()
+ cursor.execute("select * from test_dummy_table")
+ values = cursor.fetchall()
+ self.assertEquals(values, [["dummy"]])
Deleted: CalendarServer/trunk/txdav/base/datastore/util.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/datastore/util.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/datastore/util.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,47 +0,0 @@
-# -*- test-case-name: txdav.caldav.datastore.test.test_file -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Common utility functions for a datastores.
-"""
-
-_unset = object()
-
-class cached(object):
- """
- This object is a decorator for a 0-argument method which should be called
- only once, and its result cached so that future invocations just return the
- same result without calling the underlying method again.
-
- @ivar thunk: the function to call to generate a cached value.
- """
-
- def __init__(self, thunk):
- self.thunk = thunk
-
-
- def __get__(self, oself, owner):
- def inner():
- cacheKey = "_" + self.thunk.__name__ + "_cached"
- cached = getattr(oself, cacheKey, _unset)
- if cached is _unset:
- value = self.thunk(oself)
- setattr(oself, cacheKey, value)
- return value
- else:
- return cached
- return inner
Copied: CalendarServer/trunk/txdav/base/datastore/util.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/datastore/util.py)
===================================================================
--- CalendarServer/trunk/txdav/base/datastore/util.py (rev 0)
+++ CalendarServer/trunk/txdav/base/datastore/util.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,47 @@
+# -*- test-case-name: txdav.caldav.datastore.test.test_file -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Common utility functions for a datastores.
+"""
+
+_unset = object()
+
+class cached(object):
+ """
+ This object is a decorator for a 0-argument method which should be called
+ only once, and its result cached so that future invocations just return the
+ same result without calling the underlying method again.
+
+ @ivar thunk: the function to call to generate a cached value.
+ """
+
+ def __init__(self, thunk):
+ self.thunk = thunk
+
+
+ def __get__(self, oself, owner):
+ def inner():
+ cacheKey = "_" + self.thunk.__name__ + "_cached"
+ cached = getattr(oself, cacheKey, _unset)
+ if cached is _unset:
+ value = self.thunk(oself)
+ setattr(oself, cacheKey, value)
+ return value
+ else:
+ return cached
+ return inner
Deleted: CalendarServer/trunk/txdav/base/propertystore/__init__.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/__init__.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/propertystore/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,19 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-WebDAV property support for Twisted.
-"""
Copied: CalendarServer/trunk/txdav/base/propertystore/__init__.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/__init__.py)
===================================================================
--- CalendarServer/trunk/txdav/base/propertystore/__init__.py (rev 0)
+++ CalendarServer/trunk/txdav/base/propertystore/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,19 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+WebDAV property support for Twisted.
+"""
Deleted: CalendarServer/trunk/txdav/base/propertystore/base.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/base.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/propertystore/base.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,209 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Base property store.
-"""
-
-__all__ = [
- "AbstractPropertyStore",
- "PropertyName",
-]
-
-from twext.python.log import LoggingMixIn
-from twext.web2.dav import davxml
-from twext.web2.dav.resource import TwistedGETContentMD5,\
- TwistedQuotaRootProperty
-
-from txdav.idav import IPropertyStore, IPropertyName
-
-from UserDict import DictMixin
-
-from zope.interface import implements
-
-class PropertyName(LoggingMixIn):
- """
- Property name.
- """
- implements(IPropertyName)
-
- @staticmethod
- def fromString(sname):
- index = sname.find("}")
-
- if (index is -1 or not len(sname) > index or not sname[0] == "{"):
- raise TypeError("Invalid sname: %r" % (sname,))
-
- return PropertyName(sname[1:index], sname[index+1:])
-
- @staticmethod
- def fromElement(element):
- return PropertyName(element.namespace, element.name)
-
- def __init__(self, namespace, name):
- self.namespace = namespace
- self.name = name
-
-
- def _cmpval(self):
- """
- Return a value to use for hashing and comparisons.
- """
- return (self.namespace, self.name)
-
-
- # FIXME: need direct tests for presence-in-dictionary
- def __hash__(self):
- return hash(self._cmpval())
-
-
- def __eq__(self, other):
- if not isinstance(other, PropertyName):
- return NotImplemented
- return self._cmpval() == other._cmpval()
-
-
- def __ne__(self, other):
- if not isinstance(other, PropertyName):
- return NotImplemented
- return self._cmpval() != other._cmpval()
-
-
- def __repr__(self):
- return "<%s: %s>" % (
- self.__class__.__name__,
- self.toString(),
- )
-
- def toString(self):
- return "{%s}%s" % (self.namespace, self.name)
-
-
-class AbstractPropertyStore(LoggingMixIn, DictMixin):
- """
- Base property store.
- """
- implements(IPropertyStore)
-
- _defaultShadowableKeys = set()
- _defaultGlobalKeys = set((
- PropertyName.fromElement(davxml.ACL),
- PropertyName.fromElement(davxml.ResourceID),
- PropertyName.fromElement(davxml.ResourceType),
- PropertyName.fromElement(davxml.GETContentType),
- PropertyName.fromElement(TwistedGETContentMD5),
- PropertyName.fromElement(TwistedQuotaRootProperty),
- ))
-
- def __init__(self, defaultuser):
- """
- Instantiate the property store for a user. The default is the default user
- (owner) property to read in the case of global or shadowable properties.
-
- @param defaultuser: the default user uid
- @type defaultuser: C{str}
- """
-
- self._peruser = self._defaultuser = defaultuser
- self._shadowableKeys = set(AbstractPropertyStore._defaultShadowableKeys)
- self._globalKeys = set(AbstractPropertyStore._defaultGlobalKeys)
-
-
- def _setPerUserUID(self, uid):
- self._peruser = uid
-
-
- def setSpecialProperties(self, shadowableKeys, globalKeys):
- self._shadowableKeys.update(shadowableKeys)
- self._globalKeys.update(globalKeys)
-
- #
- # Subclasses must override these
- #
-
- def _getitem_uid(self, key, uid):
- raise NotImplementedError()
-
- def _setitem_uid(self, key, value, uid):
- raise NotImplementedError()
-
- def _delitem_uid(self, key, uid):
- raise NotImplementedError()
-
- def _keys_uid(self, uid):
- raise NotImplementedError()
-
- #
- # Required UserDict implementations
- #
-
- def __getitem__(self, key):
- # Handle per-user behavior
- if self.isShadowableProperty(key):
- try:
- result = self._getitem_uid(key, self._peruser)
- except KeyError:
- result = self._getitem_uid(key, self._defaultuser)
- return result
- elif self.isGlobalProperty(key):
- return self._getitem_uid(key, self._defaultuser)
- else:
- return self._getitem_uid(key, self._peruser)
-
- def __setitem__(self, key, value):
- # Handle per-user behavior
- if self.isGlobalProperty(key):
- return self._setitem_uid(key, value, self._defaultuser)
- else:
- return self._setitem_uid(key, value, self._peruser)
-
- def __delitem__(self, key):
- # Handle per-user behavior
- if self.isGlobalProperty(key):
- self._delitem_uid(key, self._defaultuser)
- else:
- self._delitem_uid(key, self._peruser)
-
- def keys(self):
-
- userkeys = self._keys_uid(self._peruser)
- if self._defaultuser != self._peruser:
- defaultkeys = self._keys_uid(self._defaultuser)
- for key in defaultkeys:
- if self.isShadowableProperty(key) and key not in userkeys:
- userkeys.append(key)
- return tuple(userkeys)
-
- def update(self, other):
- # FIXME: direct tests.
- # FIXME: support positional signature (although since strings aren't
- # valid, it should just raise an error.
- for key in other:
- self[key] = other[key]
-
-
- # Per-user property handling
- def isShadowableProperty(self, key):
- return key in self._shadowableKeys
-
- def isGlobalProperty(self, key):
- return key in self._globalKeys
-
-# FIXME: Actually, we should replace this with calls to IPropertyName()
-def validKey(key):
- # Used by implementations to verify that keys are valid
- if not isinstance(key, PropertyName):
- raise TypeError("Not a PropertyName: %r" % (key,))
Copied: CalendarServer/trunk/txdav/base/propertystore/base.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/base.py)
===================================================================
--- CalendarServer/trunk/txdav/base/propertystore/base.py (rev 0)
+++ CalendarServer/trunk/txdav/base/propertystore/base.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,209 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Base property store.
+"""
+
+__all__ = [
+ "AbstractPropertyStore",
+ "PropertyName",
+]
+
+from twext.python.log import LoggingMixIn
+from twext.web2.dav import davxml
+from twext.web2.dav.resource import TwistedGETContentMD5,\
+ TwistedQuotaRootProperty
+
+from txdav.idav import IPropertyStore, IPropertyName
+
+from UserDict import DictMixin
+
+from zope.interface import implements
+
+class PropertyName(LoggingMixIn):
+ """
+ Property name.
+ """
+ implements(IPropertyName)
+
+ @staticmethod
+ def fromString(sname):
+ index = sname.find("}")
+
+ if (index is -1 or not len(sname) > index or not sname[0] == "{"):
+ raise TypeError("Invalid sname: %r" % (sname,))
+
+ return PropertyName(sname[1:index], sname[index+1:])
+
+ @staticmethod
+ def fromElement(element):
+ return PropertyName(element.namespace, element.name)
+
+ def __init__(self, namespace, name):
+ self.namespace = namespace
+ self.name = name
+
+
+ def _cmpval(self):
+ """
+ Return a value to use for hashing and comparisons.
+ """
+ return (self.namespace, self.name)
+
+
+ # FIXME: need direct tests for presence-in-dictionary
+ def __hash__(self):
+ return hash(self._cmpval())
+
+
+ def __eq__(self, other):
+ if not isinstance(other, PropertyName):
+ return NotImplemented
+ return self._cmpval() == other._cmpval()
+
+
+ def __ne__(self, other):
+ if not isinstance(other, PropertyName):
+ return NotImplemented
+ return self._cmpval() != other._cmpval()
+
+
+ def __repr__(self):
+ return "<%s: %s>" % (
+ self.__class__.__name__,
+ self.toString(),
+ )
+
+ def toString(self):
+ return "{%s}%s" % (self.namespace, self.name)
+
+
+class AbstractPropertyStore(LoggingMixIn, DictMixin):
+ """
+ Base property store.
+ """
+ implements(IPropertyStore)
+
+ _defaultShadowableKeys = set()
+ _defaultGlobalKeys = set((
+ PropertyName.fromElement(davxml.ACL),
+ PropertyName.fromElement(davxml.ResourceID),
+ PropertyName.fromElement(davxml.ResourceType),
+ PropertyName.fromElement(davxml.GETContentType),
+ PropertyName.fromElement(TwistedGETContentMD5),
+ PropertyName.fromElement(TwistedQuotaRootProperty),
+ ))
+
+ def __init__(self, defaultuser):
+ """
+ Instantiate the property store for a user. The default is the default user
+ (owner) property to read in the case of global or shadowable properties.
+
+ @param defaultuser: the default user uid
+ @type defaultuser: C{str}
+ """
+
+ self._peruser = self._defaultuser = defaultuser
+ self._shadowableKeys = set(AbstractPropertyStore._defaultShadowableKeys)
+ self._globalKeys = set(AbstractPropertyStore._defaultGlobalKeys)
+
+
+ def _setPerUserUID(self, uid):
+ self._peruser = uid
+
+
+ def setSpecialProperties(self, shadowableKeys, globalKeys):
+ self._shadowableKeys.update(shadowableKeys)
+ self._globalKeys.update(globalKeys)
+
+ #
+ # Subclasses must override these
+ #
+
+ def _getitem_uid(self, key, uid):
+ raise NotImplementedError()
+
+ def _setitem_uid(self, key, value, uid):
+ raise NotImplementedError()
+
+ def _delitem_uid(self, key, uid):
+ raise NotImplementedError()
+
+ def _keys_uid(self, uid):
+ raise NotImplementedError()
+
+ #
+ # Required UserDict implementations
+ #
+
+ def __getitem__(self, key):
+ # Handle per-user behavior
+ if self.isShadowableProperty(key):
+ try:
+ result = self._getitem_uid(key, self._peruser)
+ except KeyError:
+ result = self._getitem_uid(key, self._defaultuser)
+ return result
+ elif self.isGlobalProperty(key):
+ return self._getitem_uid(key, self._defaultuser)
+ else:
+ return self._getitem_uid(key, self._peruser)
+
+ def __setitem__(self, key, value):
+ # Handle per-user behavior
+ if self.isGlobalProperty(key):
+ return self._setitem_uid(key, value, self._defaultuser)
+ else:
+ return self._setitem_uid(key, value, self._peruser)
+
+ def __delitem__(self, key):
+ # Handle per-user behavior
+ if self.isGlobalProperty(key):
+ self._delitem_uid(key, self._defaultuser)
+ else:
+ self._delitem_uid(key, self._peruser)
+
+ def keys(self):
+
+ userkeys = self._keys_uid(self._peruser)
+ if self._defaultuser != self._peruser:
+ defaultkeys = self._keys_uid(self._defaultuser)
+ for key in defaultkeys:
+ if self.isShadowableProperty(key) and key not in userkeys:
+ userkeys.append(key)
+ return tuple(userkeys)
+
+ def update(self, other):
+ # FIXME: direct tests.
+ # FIXME: support positional signature (although since strings aren't
+ # valid, it should just raise an error.
+ for key in other:
+ self[key] = other[key]
+
+
+ # Per-user property handling
+ def isShadowableProperty(self, key):
+ return key in self._shadowableKeys
+
+ def isGlobalProperty(self, key):
+ return key in self._globalKeys
+
+# FIXME: Actually, we should replace this with calls to IPropertyName()
+def validKey(key):
+ # Used by implementations to verify that keys are valid
+ if not isinstance(key, PropertyName):
+ raise TypeError("Not a PropertyName: %r" % (key,))
Deleted: CalendarServer/trunk/txdav/base/propertystore/none.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/none.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/propertystore/none.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,116 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Property store with no storage.
-"""
-
-__all__ = [
- "PropertyStore",
-]
-
-from txdav.base.propertystore.base import AbstractPropertyStore, validKey
-
-class PropertyStore(AbstractPropertyStore):
- """
- Property store with no storage.
- """
-
- properties = {}
-
- def __init__(self, defaultuser):
- super(PropertyStore, self).__init__(defaultuser)
-
- self.modified = {}
- self.removed = set()
-
- def __str__(self):
- return "<%s>" % (self.__class__.__name__,)
-
- #
- # Required implementations
- #
-
- def _getitem_uid(self, key, uid):
- validKey(key)
- effectiveKey = (key, uid)
-
- if effectiveKey in self.modified:
- return self.modified[effectiveKey]
-
- if effectiveKey in self.removed:
- raise KeyError(key)
-
- return self.properties[effectiveKey]
-
- def _setitem_uid(self, key, value, uid):
- validKey(key)
- effectiveKey = (key, uid)
-
- if effectiveKey in self.removed:
- self.removed.remove(effectiveKey)
- self.modified[effectiveKey] = value
-
- def _delitem_uid(self, key, uid):
- validKey(key)
- effectiveKey = (key, uid)
-
- if effectiveKey in self.modified:
- del self.modified[effectiveKey]
- elif effectiveKey not in self.properties:
- raise KeyError(key)
-
- self.removed.add(effectiveKey)
-
- def _keys_uid(self, uid):
- seen = set()
-
- for effectivekey in self.properties:
- if effectivekey[1] == uid and effectivekey not in self.removed:
- seen.add(effectivekey)
- yield effectivekey[0]
-
- for effectivekey in self.modified:
- if effectivekey[1] == uid and effectivekey not in seen:
- yield effectivekey[0]
-
- #
- # I/O
- #
-
- def flush(self):
- props = self.properties
- removed = self.removed
- modified = self.modified
-
- for effectivekey in removed:
- assert effectivekey not in modified
- try:
- del props[effectivekey]
- except KeyError:
- pass
-
- for effectivekey in modified:
- assert effectivekey not in removed
- value = modified[effectivekey]
- props[effectivekey] = value
-
- self.removed.clear()
- self.modified.clear()
-
- def abort(self):
- self.removed.clear()
- self.modified.clear()
Copied: CalendarServer/trunk/txdav/base/propertystore/none.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/none.py)
===================================================================
--- CalendarServer/trunk/txdav/base/propertystore/none.py (rev 0)
+++ CalendarServer/trunk/txdav/base/propertystore/none.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,116 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Property store with no storage.
+"""
+
+__all__ = [
+ "PropertyStore",
+]
+
+from txdav.base.propertystore.base import AbstractPropertyStore, validKey
+
+class PropertyStore(AbstractPropertyStore):
+ """
+ Property store with no storage.
+ """
+
+ properties = {}
+
+ def __init__(self, defaultuser):
+ super(PropertyStore, self).__init__(defaultuser)
+
+ self.modified = {}
+ self.removed = set()
+
+ def __str__(self):
+ return "<%s>" % (self.__class__.__name__,)
+
+ #
+ # Required implementations
+ #
+
+ def _getitem_uid(self, key, uid):
+ validKey(key)
+ effectiveKey = (key, uid)
+
+ if effectiveKey in self.modified:
+ return self.modified[effectiveKey]
+
+ if effectiveKey in self.removed:
+ raise KeyError(key)
+
+ return self.properties[effectiveKey]
+
+ def _setitem_uid(self, key, value, uid):
+ validKey(key)
+ effectiveKey = (key, uid)
+
+ if effectiveKey in self.removed:
+ self.removed.remove(effectiveKey)
+ self.modified[effectiveKey] = value
+
+ def _delitem_uid(self, key, uid):
+ validKey(key)
+ effectiveKey = (key, uid)
+
+ if effectiveKey in self.modified:
+ del self.modified[effectiveKey]
+ elif effectiveKey not in self.properties:
+ raise KeyError(key)
+
+ self.removed.add(effectiveKey)
+
+ def _keys_uid(self, uid):
+ seen = set()
+
+ for effectivekey in self.properties:
+ if effectivekey[1] == uid and effectivekey not in self.removed:
+ seen.add(effectivekey)
+ yield effectivekey[0]
+
+ for effectivekey in self.modified:
+ if effectivekey[1] == uid and effectivekey not in seen:
+ yield effectivekey[0]
+
+ #
+ # I/O
+ #
+
+ def flush(self):
+ props = self.properties
+ removed = self.removed
+ modified = self.modified
+
+ for effectivekey in removed:
+ assert effectivekey not in modified
+ try:
+ del props[effectivekey]
+ except KeyError:
+ pass
+
+ for effectivekey in modified:
+ assert effectivekey not in removed
+ value = modified[effectivekey]
+ props[effectivekey] = value
+
+ self.removed.clear()
+ self.modified.clear()
+
+ def abort(self):
+ self.removed.clear()
+ self.modified.clear()
Deleted: CalendarServer/trunk/txdav/base/propertystore/sql.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/sql.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/propertystore/sql.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,81 +0,0 @@
-# -*- test-case-name: txdav.base.propertystore.test.test_sql -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-PostgreSQL data store.
-"""
-
-__all__ = [
- "PropertyStore",
-]
-
-from txdav.base.propertystore.base import AbstractPropertyStore, PropertyName,\
- validKey
-
-from twext.web2.dav.davxml import WebDAVDocument
-
-class PropertyStore(AbstractPropertyStore):
-
- def __init__(self, defaultuser, txn, resourceID):
- super(PropertyStore, self).__init__(defaultuser)
- self._txn = txn
- self._resourceID = resourceID
-
-
- def _getitem_uid(self, key, uid):
- validKey(key)
- rows = self._txn.execSQL(
- "select VALUE from RESOURCE_PROPERTY where "
- "RESOURCE_ID = %s and NAME = %s and VIEWER_UID = %s",
- [self._resourceID, key.toString(), uid]
- )
- if not rows:
- raise KeyError(key)
- return WebDAVDocument.fromString(rows[0][0]).root_element
-
-
- def _setitem_uid(self, key, value, uid):
- validKey(key)
- try:
- self._delitem_uid(key, uid)
- except KeyError:
- pass
- self._txn.execSQL(
- "insert into RESOURCE_PROPERTY "
- "(RESOURCE_ID, NAME, VALUE, VIEWER_UID) values (%s, %s, %s, %s)",
- [self._resourceID, key.toString(), value.toxml(), uid]
- )
-
-
- def _delitem_uid(self, key, uid):
- validKey(key)
- self._txn.execSQL(
- "delete from RESOURCE_PROPERTY where VIEWER_UID = %s"
- "and RESOURCE_ID = %s AND NAME = %s",
- [uid, self._resourceID, key.toString()],
- raiseOnZeroRowCount=lambda:KeyError(key)
- )
-
-
- def _keys_uid(self, uid):
- rows = self._txn.execSQL(
- "select NAME from RESOURCE_PROPERTY where "
- "VIEWER_UID = %s and RESOURCE_ID = %s",
- [uid, self._resourceID]
- )
- for row in rows:
- yield PropertyName.fromString(row[0])
Copied: CalendarServer/trunk/txdav/base/propertystore/sql.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/sql.py)
===================================================================
--- CalendarServer/trunk/txdav/base/propertystore/sql.py (rev 0)
+++ CalendarServer/trunk/txdav/base/propertystore/sql.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,81 @@
+# -*- test-case-name: txdav.base.propertystore.test.test_sql -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+PostgreSQL data store.
+"""
+
+__all__ = [
+ "PropertyStore",
+]
+
+from txdav.base.propertystore.base import AbstractPropertyStore, PropertyName,\
+ validKey
+
+from twext.web2.dav.davxml import WebDAVDocument
+
+class PropertyStore(AbstractPropertyStore):
+
+ def __init__(self, defaultuser, txn, resourceID):
+ super(PropertyStore, self).__init__(defaultuser)
+ self._txn = txn
+ self._resourceID = resourceID
+
+
+ def _getitem_uid(self, key, uid):
+ validKey(key)
+ rows = self._txn.execSQL(
+ "select VALUE from RESOURCE_PROPERTY where "
+ "RESOURCE_ID = %s and NAME = %s and VIEWER_UID = %s",
+ [self._resourceID, key.toString(), uid]
+ )
+ if not rows:
+ raise KeyError(key)
+ return WebDAVDocument.fromString(rows[0][0]).root_element
+
+
+ def _setitem_uid(self, key, value, uid):
+ validKey(key)
+ try:
+ self._delitem_uid(key, uid)
+ except KeyError:
+ pass
+ self._txn.execSQL(
+ "insert into RESOURCE_PROPERTY "
+ "(RESOURCE_ID, NAME, VALUE, VIEWER_UID) values (%s, %s, %s, %s)",
+ [self._resourceID, key.toString(), value.toxml(), uid]
+ )
+
+
+ def _delitem_uid(self, key, uid):
+ validKey(key)
+ self._txn.execSQL(
+ "delete from RESOURCE_PROPERTY where VIEWER_UID = %s"
+ "and RESOURCE_ID = %s AND NAME = %s",
+ [uid, self._resourceID, key.toString()],
+ raiseOnZeroRowCount=lambda:KeyError(key)
+ )
+
+
+ def _keys_uid(self, uid):
+ rows = self._txn.execSQL(
+ "select NAME from RESOURCE_PROPERTY where "
+ "VIEWER_UID = %s and RESOURCE_ID = %s",
+ [uid, self._resourceID]
+ )
+ for row in rows:
+ yield PropertyName.fromString(row[0])
Deleted: CalendarServer/trunk/txdav/base/propertystore/test/__init__.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/test/__init__.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/propertystore/test/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,19 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Property store tests.
-"""
Copied: CalendarServer/trunk/txdav/base/propertystore/test/__init__.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/test/__init__.py)
===================================================================
--- CalendarServer/trunk/txdav/base/propertystore/test/__init__.py (rev 0)
+++ CalendarServer/trunk/txdav/base/propertystore/test/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,19 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Property store tests.
+"""
Deleted: CalendarServer/trunk/txdav/base/propertystore/test/base.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/test/base.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/propertystore/test/base.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,319 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Generic property store tests.
-"""
-
-__all__ = [
- "PropertyStoreTest",
- "propertyName",
- "propertyValue",
-]
-
-
-from zope.interface.verify import verifyObject, BrokenMethodImplementation
-
-from twisted.trial import unittest
-
-from twext.web2.dav import davxml
-
-from txdav.idav import IPropertyStore
-from txdav.base.propertystore.base import PropertyName
-
-
-class PropertyStoreTest(unittest.TestCase):
- # Subclass must define self.propertyStore in setUp().
-
- def _preTest(self):
- self.addCleanup(self._postTest)
- def _postTest(self):
- pass
- def _changed(self, store):
- store.flush()
- def _abort(self, store):
- store.abort()
-
- def test_interface(self):
- try:
- self._preTest()
- verifyObject(IPropertyStore, self.propertyStore)
- except BrokenMethodImplementation, e:
- self.fail(e)
-
- def test_set_get_contains(self):
-
- self._preTest()
- store = self.propertyStore
-
- name = propertyName("test")
- value = propertyValue("Hello, World!")
-
- # Test with commit after change
- store[name] = value
- self._changed(store)
- self.assertEquals(store.get(name, None), value)
- self.failUnless(name in store)
-
- # Test without commit after change
- value = propertyValue("Hello, Universe!")
- store[name] = value
- self.assertEquals(store.get(name, None), value)
- self.failUnless(name in store)
-
- def test_delete_get_contains(self):
-
- self._preTest()
- store = self.propertyStore
-
- # Test with commit after change
- name = propertyName("test")
- value = propertyValue("Hello, World!")
-
- store[name] = value
- self._changed(store)
-
- del store[name]
- self._changed(store)
-
- self.assertEquals(store.get(name, None), None)
- self.failIf(name in store)
-
- # Test without commit after change
- name = propertyName("test")
- value = propertyValue("Hello, Universe!")
-
- store[name] = value
- self._changed(store)
-
- del store[name]
-
- self.assertEquals(store.get(name, None), None)
- self.failIf(name in store)
-
- def test_peruser(self):
-
- self._preTest()
- store1 = self.propertyStore1
- store2 = self.propertyStore2
-
- name = propertyName("test")
- value1 = propertyValue("Hello, World1!")
- value2 = propertyValue("Hello, World2!")
-
- store1[name] = value1
- self._changed(store1)
- self.assertEquals(store1.get(name, None), value1)
- self.assertEquals(store2.get(name, None), None)
- self.failUnless(name in store1)
- self.failIf(name in store2)
-
- store2[name] = value2
- self._changed(store2)
- self.assertEquals(store1.get(name, None), value1)
- self.assertEquals(store2.get(name, None), value2)
- self.failUnless(name in store1)
- self.failUnless(name in store2)
-
- del store2[name]
- self._changed(store2)
- self.assertEquals(store1.get(name, None), value1)
- self.assertEquals(store2.get(name, None), None)
- self.failUnless(name in store1)
- self.failIf(name in store2)
-
- del store1[name]
- self._changed(store1)
- self.assertEquals(store1.get(name, None), None)
- self.assertEquals(store2.get(name, None), None)
- self.failIf(name in store1)
- self.failIf(name in store2)
-
- def test_peruser_shadow(self):
-
- self._preTest()
- store1 = self.propertyStore1
- store2 = self.propertyStore2
-
- name = propertyName("shadow")
-
- store1.setSpecialProperties((name,), ())
- store2.setSpecialProperties((name,), ())
-
- value1 = propertyValue("Hello, World1!")
- value2 = propertyValue("Hello, World2!")
-
- store1[name] = value1
- self._changed(store1)
- self.assertEquals(store1.get(name, None), value1)
- self.assertEquals(store2.get(name, None), value1)
- self.failUnless(name in store1)
- self.failUnless(name in store2)
-
- store2[name] = value2
- self._changed(store2)
- self.assertEquals(store1.get(name, None), value1)
- self.assertEquals(store2.get(name, None), value2)
- self.failUnless(name in store1)
- self.failUnless(name in store2)
-
- del store2[name]
- self._changed(store2)
- self.assertEquals(store1.get(name, None), value1)
- self.assertEquals(store2.get(name, None), value1)
- self.failUnless(name in store1)
- self.failUnless(name in store2)
-
- del store1[name]
- self._changed(store1)
- self.assertEquals(store1.get(name, None), None)
- self.assertEquals(store2.get(name, None), None)
- self.failIf(name in store1)
- self.failIf(name in store2)
-
-
- def test_peruser_global(self):
-
- self._preTest()
- store1 = self.propertyStore1
- store2 = self.propertyStore2
-
- name = propertyName("global")
-
- store1.setSpecialProperties((), (name,))
- store2.setSpecialProperties((), (name,))
-
- value1 = propertyValue("Hello, World1!")
- value2 = propertyValue("Hello, World2!")
-
- store1[name] = value1
- self._changed(store1)
- self.assertEquals(store1.get(name, None), value1)
- self.assertEquals(store2.get(name, None), value1)
- self.failUnless(name in store1)
- self.failUnless(name in store2)
-
- store2[name] = value2
- self._changed(store2)
- self.assertEquals(store1.get(name, None), value2)
- self.assertEquals(store2.get(name, None), value2)
- self.failUnless(name in store1)
- self.failUnless(name in store2)
-
- del store2[name]
- self._changed(store2)
- self.assertEquals(store1.get(name, None), None)
- self.assertEquals(store2.get(name, None), None)
- self.failIf(name in store1)
- self.failIf(name in store2)
-
-
- def test_iteration(self):
-
- self._preTest()
- store = self.propertyStore
-
- value = propertyValue("Hello, World!")
-
- names = set(propertyName(str(i)) for i in (1,2,3,4))
-
- for name in names:
- store[name] = value
-
- self.assertEquals(set(store.keys()), names)
- self.assertEquals(len(store), len(names))
-
- def test_delete_none(self):
-
- self._preTest()
- def doDelete():
- del self.propertyStore[propertyName("xyzzy")]
-
- self.assertRaises(KeyError, doDelete)
-
- def test_keyInPropertyName(self):
-
- self._preTest()
- store = self.propertyStore
-
- def doGet():
- store["xyzzy"]
-
- def doSet():
- store["xyzzy"] = propertyValue("Hello, World!")
-
- def doDelete():
- del store["xyzzy"]
-
- def doContains():
- return "xyzzy" in store
-
- self.assertRaises(TypeError, doGet)
- self.assertRaises(TypeError, doSet)
- self.assertRaises(TypeError, doDelete)
- self.assertRaises(TypeError, doContains)
-
- def test_flush(self):
-
- self._preTest()
- store = self.propertyStore
-
- name = propertyName("test")
- value = propertyValue("Hello, World!")
-
- #
- # Set value flushes correctly
- #
- store[name] = value
-
- self._changed(store)
- self._abort(store)
-
- self.assertEquals(store.get(name, None), value)
- self.assertEquals(len(store), 1)
-
- #
- # Deleted value flushes correctly
- #
- del store[name]
-
- self._changed(store)
- self._abort(store)
-
- self.assertEquals(store.get(name, None), None)
- self.assertEquals(len(store), 0)
-
- def test_abort(self):
-
- self._preTest()
- store = self.propertyStore
-
- name = propertyName("test")
- value = propertyValue("Hello, World!")
-
- store[name] = value
-
- self._abort(store)
-
- self.assertEquals(store.get(name, None), None)
- self.assertEquals(len(store), 0)
-
-
-def propertyName(name):
- return PropertyName("http://calendarserver.org/ns/test/", name)
-
-def propertyValue(value):
- return davxml.ResponseDescription(value)
Copied: CalendarServer/trunk/txdav/base/propertystore/test/base.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/test/base.py)
===================================================================
--- CalendarServer/trunk/txdav/base/propertystore/test/base.py (rev 0)
+++ CalendarServer/trunk/txdav/base/propertystore/test/base.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,319 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Generic property store tests.
+"""
+
+__all__ = [
+ "PropertyStoreTest",
+ "propertyName",
+ "propertyValue",
+]
+
+
+from zope.interface.verify import verifyObject, BrokenMethodImplementation
+
+from twisted.trial import unittest
+
+from twext.web2.dav import davxml
+
+from txdav.idav import IPropertyStore
+from txdav.base.propertystore.base import PropertyName
+
+
+class PropertyStoreTest(unittest.TestCase):
+ # Subclass must define self.propertyStore in setUp().
+
+ def _preTest(self):
+ self.addCleanup(self._postTest)
+ def _postTest(self):
+ pass
+ def _changed(self, store):
+ store.flush()
+ def _abort(self, store):
+ store.abort()
+
+ def test_interface(self):
+ try:
+ self._preTest()
+ verifyObject(IPropertyStore, self.propertyStore)
+ except BrokenMethodImplementation, e:
+ self.fail(e)
+
+ def test_set_get_contains(self):
+
+ self._preTest()
+ store = self.propertyStore
+
+ name = propertyName("test")
+ value = propertyValue("Hello, World!")
+
+ # Test with commit after change
+ store[name] = value
+ self._changed(store)
+ self.assertEquals(store.get(name, None), value)
+ self.failUnless(name in store)
+
+ # Test without commit after change
+ value = propertyValue("Hello, Universe!")
+ store[name] = value
+ self.assertEquals(store.get(name, None), value)
+ self.failUnless(name in store)
+
+ def test_delete_get_contains(self):
+
+ self._preTest()
+ store = self.propertyStore
+
+ # Test with commit after change
+ name = propertyName("test")
+ value = propertyValue("Hello, World!")
+
+ store[name] = value
+ self._changed(store)
+
+ del store[name]
+ self._changed(store)
+
+ self.assertEquals(store.get(name, None), None)
+ self.failIf(name in store)
+
+ # Test without commit after change
+ name = propertyName("test")
+ value = propertyValue("Hello, Universe!")
+
+ store[name] = value
+ self._changed(store)
+
+ del store[name]
+
+ self.assertEquals(store.get(name, None), None)
+ self.failIf(name in store)
+
+ def test_peruser(self):
+
+ self._preTest()
+ store1 = self.propertyStore1
+ store2 = self.propertyStore2
+
+ name = propertyName("test")
+ value1 = propertyValue("Hello, World1!")
+ value2 = propertyValue("Hello, World2!")
+
+ store1[name] = value1
+ self._changed(store1)
+ self.assertEquals(store1.get(name, None), value1)
+ self.assertEquals(store2.get(name, None), None)
+ self.failUnless(name in store1)
+ self.failIf(name in store2)
+
+ store2[name] = value2
+ self._changed(store2)
+ self.assertEquals(store1.get(name, None), value1)
+ self.assertEquals(store2.get(name, None), value2)
+ self.failUnless(name in store1)
+ self.failUnless(name in store2)
+
+ del store2[name]
+ self._changed(store2)
+ self.assertEquals(store1.get(name, None), value1)
+ self.assertEquals(store2.get(name, None), None)
+ self.failUnless(name in store1)
+ self.failIf(name in store2)
+
+ del store1[name]
+ self._changed(store1)
+ self.assertEquals(store1.get(name, None), None)
+ self.assertEquals(store2.get(name, None), None)
+ self.failIf(name in store1)
+ self.failIf(name in store2)
+
+ def test_peruser_shadow(self):
+
+ self._preTest()
+ store1 = self.propertyStore1
+ store2 = self.propertyStore2
+
+ name = propertyName("shadow")
+
+ store1.setSpecialProperties((name,), ())
+ store2.setSpecialProperties((name,), ())
+
+ value1 = propertyValue("Hello, World1!")
+ value2 = propertyValue("Hello, World2!")
+
+ store1[name] = value1
+ self._changed(store1)
+ self.assertEquals(store1.get(name, None), value1)
+ self.assertEquals(store2.get(name, None), value1)
+ self.failUnless(name in store1)
+ self.failUnless(name in store2)
+
+ store2[name] = value2
+ self._changed(store2)
+ self.assertEquals(store1.get(name, None), value1)
+ self.assertEquals(store2.get(name, None), value2)
+ self.failUnless(name in store1)
+ self.failUnless(name in store2)
+
+ del store2[name]
+ self._changed(store2)
+ self.assertEquals(store1.get(name, None), value1)
+ self.assertEquals(store2.get(name, None), value1)
+ self.failUnless(name in store1)
+ self.failUnless(name in store2)
+
+ del store1[name]
+ self._changed(store1)
+ self.assertEquals(store1.get(name, None), None)
+ self.assertEquals(store2.get(name, None), None)
+ self.failIf(name in store1)
+ self.failIf(name in store2)
+
+
+ def test_peruser_global(self):
+
+ self._preTest()
+ store1 = self.propertyStore1
+ store2 = self.propertyStore2
+
+ name = propertyName("global")
+
+ store1.setSpecialProperties((), (name,))
+ store2.setSpecialProperties((), (name,))
+
+ value1 = propertyValue("Hello, World1!")
+ value2 = propertyValue("Hello, World2!")
+
+ store1[name] = value1
+ self._changed(store1)
+ self.assertEquals(store1.get(name, None), value1)
+ self.assertEquals(store2.get(name, None), value1)
+ self.failUnless(name in store1)
+ self.failUnless(name in store2)
+
+ store2[name] = value2
+ self._changed(store2)
+ self.assertEquals(store1.get(name, None), value2)
+ self.assertEquals(store2.get(name, None), value2)
+ self.failUnless(name in store1)
+ self.failUnless(name in store2)
+
+ del store2[name]
+ self._changed(store2)
+ self.assertEquals(store1.get(name, None), None)
+ self.assertEquals(store2.get(name, None), None)
+ self.failIf(name in store1)
+ self.failIf(name in store2)
+
+
+ def test_iteration(self):
+
+ self._preTest()
+ store = self.propertyStore
+
+ value = propertyValue("Hello, World!")
+
+ names = set(propertyName(str(i)) for i in (1,2,3,4))
+
+ for name in names:
+ store[name] = value
+
+ self.assertEquals(set(store.keys()), names)
+ self.assertEquals(len(store), len(names))
+
+ def test_delete_none(self):
+
+ self._preTest()
+ def doDelete():
+ del self.propertyStore[propertyName("xyzzy")]
+
+ self.assertRaises(KeyError, doDelete)
+
+ def test_keyInPropertyName(self):
+
+ self._preTest()
+ store = self.propertyStore
+
+ def doGet():
+ store["xyzzy"]
+
+ def doSet():
+ store["xyzzy"] = propertyValue("Hello, World!")
+
+ def doDelete():
+ del store["xyzzy"]
+
+ def doContains():
+ return "xyzzy" in store
+
+ self.assertRaises(TypeError, doGet)
+ self.assertRaises(TypeError, doSet)
+ self.assertRaises(TypeError, doDelete)
+ self.assertRaises(TypeError, doContains)
+
+ def test_flush(self):
+
+ self._preTest()
+ store = self.propertyStore
+
+ name = propertyName("test")
+ value = propertyValue("Hello, World!")
+
+ #
+ # Set value flushes correctly
+ #
+ store[name] = value
+
+ self._changed(store)
+ self._abort(store)
+
+ self.assertEquals(store.get(name, None), value)
+ self.assertEquals(len(store), 1)
+
+ #
+ # Deleted value flushes correctly
+ #
+ del store[name]
+
+ self._changed(store)
+ self._abort(store)
+
+ self.assertEquals(store.get(name, None), None)
+ self.assertEquals(len(store), 0)
+
+ def test_abort(self):
+
+ self._preTest()
+ store = self.propertyStore
+
+ name = propertyName("test")
+ value = propertyValue("Hello, World!")
+
+ store[name] = value
+
+ self._abort(store)
+
+ self.assertEquals(store.get(name, None), None)
+ self.assertEquals(len(store), 0)
+
+
+def propertyName(name):
+ return PropertyName("http://calendarserver.org/ns/test/", name)
+
+def propertyValue(value):
+ return davxml.ResponseDescription(value)
Deleted: CalendarServer/trunk/txdav/base/propertystore/test/test_base.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/test/test_base.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/propertystore/test/test_base.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,55 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Property store tests.
-"""
-
-from zope.interface.verify import verifyObject, BrokenMethodImplementation
-
-from twisted.trial import unittest
-
-from txdav.idav import IPropertyName
-from txdav.base.propertystore.base import PropertyName
-
-
-class PropertyNameTest(unittest.TestCase):
- def test_interface(self):
- name = PropertyName("http://calendarserver.org/", "bleargh")
- try:
- verifyObject(IPropertyName, name)
- except BrokenMethodImplementation, e:
- self.fail(e)
-
- def test_init(self):
- name = PropertyName("http://calendarserver.org/", "bleargh")
-
- self.assertEquals(name.namespace, "http://calendarserver.org/")
- self.assertEquals(name.name, "bleargh")
-
- def test_fromString(self):
- name = PropertyName.fromString("{http://calendarserver.org/}bleargh")
-
- self.assertEquals(name.namespace, "http://calendarserver.org/")
- self.assertEquals(name.name, "bleargh")
-
- def test_toString(self):
- name = PropertyName("http://calendarserver.org/", "bleargh")
-
- self.assertEquals(
- name.toString(),
- "{http://calendarserver.org/}bleargh"
- )
Copied: CalendarServer/trunk/txdav/base/propertystore/test/test_base.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/test/test_base.py)
===================================================================
--- CalendarServer/trunk/txdav/base/propertystore/test/test_base.py (rev 0)
+++ CalendarServer/trunk/txdav/base/propertystore/test/test_base.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,55 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Property store tests.
+"""
+
+from zope.interface.verify import verifyObject, BrokenMethodImplementation
+
+from twisted.trial import unittest
+
+from txdav.idav import IPropertyName
+from txdav.base.propertystore.base import PropertyName
+
+
+class PropertyNameTest(unittest.TestCase):
+ def test_interface(self):
+ name = PropertyName("http://calendarserver.org/", "bleargh")
+ try:
+ verifyObject(IPropertyName, name)
+ except BrokenMethodImplementation, e:
+ self.fail(e)
+
+ def test_init(self):
+ name = PropertyName("http://calendarserver.org/", "bleargh")
+
+ self.assertEquals(name.namespace, "http://calendarserver.org/")
+ self.assertEquals(name.name, "bleargh")
+
+ def test_fromString(self):
+ name = PropertyName.fromString("{http://calendarserver.org/}bleargh")
+
+ self.assertEquals(name.namespace, "http://calendarserver.org/")
+ self.assertEquals(name.name, "bleargh")
+
+ def test_toString(self):
+ name = PropertyName("http://calendarserver.org/", "bleargh")
+
+ self.assertEquals(
+ name.toString(),
+ "{http://calendarserver.org/}bleargh"
+ )
Deleted: CalendarServer/trunk/txdav/base/propertystore/test/test_none.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/test/test_none.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/propertystore/test/test_none.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,35 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Property store tests.
-"""
-
-from txdav.base.propertystore.none import PropertyStore
-
-from txdav.base.propertystore.test import base
-
-class PropertyStoreTest(base.PropertyStoreTest):
- def setUp(self):
- self.propertyStore = self.propertyStore1 = PropertyStore("user01")
- self.propertyStore2 = PropertyStore("user01")
- self.propertyStore2._setPerUserUID("user02")
-
- def test_abort(self):
- super(PropertyStoreTest, self).test_abort()
- store = self.propertyStore
- self.assertEquals(store.removed, set())
- self.assertEquals(store.modified, {})
Copied: CalendarServer/trunk/txdav/base/propertystore/test/test_none.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/test/test_none.py)
===================================================================
--- CalendarServer/trunk/txdav/base/propertystore/test/test_none.py (rev 0)
+++ CalendarServer/trunk/txdav/base/propertystore/test/test_none.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,35 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Property store tests.
+"""
+
+from txdav.base.propertystore.none import PropertyStore
+
+from txdav.base.propertystore.test import base
+
+class PropertyStoreTest(base.PropertyStoreTest):
+ def setUp(self):
+ self.propertyStore = self.propertyStore1 = PropertyStore("user01")
+ self.propertyStore2 = PropertyStore("user01")
+ self.propertyStore2._setPerUserUID("user02")
+
+ def test_abort(self):
+ super(PropertyStoreTest, self).test_abort()
+ store = self.propertyStore
+ self.assertEquals(store.removed, set())
+ self.assertEquals(store.modified, {})
Deleted: CalendarServer/trunk/txdav/base/propertystore/test/test_sql.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/test/test_sql.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/propertystore/test/test_sql.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,88 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Tests for txdav.caldav.datastore.postgres, mostly based on
-L{txdav.caldav.datastore.test.common}.
-"""
-
-from twisted.internet.defer import inlineCallbacks
-
-from txdav.caldav.datastore.test.common import StubNotifierFactory
-
-from txdav.common.datastore.test.util import SQLStoreBuilder
-
-from txdav.base.propertystore.base import PropertyName
-from txdav.base.propertystore.test import base
-
-try:
- from txdav.base.propertystore.sql import PropertyStore
-except ImportError, e:
- PropertyStore = None
- importErrorMessage = str(e)
-
-
-
-theStoreBuilder = SQLStoreBuilder()
-buildStore = theStoreBuilder.buildStore
-
-
-class PropertyStoreTest(base.PropertyStoreTest):
-
- def _preTest(self):
- self._txn = self.store.newTransaction()
- self.propertyStore = self.propertyStore1 = PropertyStore(
- "user01", self._txn, 1
- )
- self.propertyStore2 = PropertyStore("user01", self._txn, 1)
- self.propertyStore2._setPerUserUID("user02")
-
- self.addCleanup(self._postTest)
-
- def _postTest(self):
- if hasattr(self, "_txn"):
- self._txn.commit()
- delattr(self, "_txn")
- self.propertyStore = self.propertyStore1 = self.propertyStore2 = None
-
- def _changed(self, store):
- if hasattr(self, "_txn"):
- self._txn.commit()
- delattr(self, "_txn")
- self._txn = self.store.newTransaction()
- self.propertyStore1._txn = self._txn
- self.propertyStore2._txn = self._txn
-
- def _abort(self, store):
- if hasattr(self, "_txn"):
- self._txn.abort()
- delattr(self, "_txn")
-
- self._txn = self.store.newTransaction()
- self.propertyStore1._txn = self._txn
- self.propertyStore2._txn = self._txn
-
- @inlineCallbacks
- def setUp(self):
- self.notifierFactory = StubNotifierFactory()
- self.store = yield buildStore(self, self.notifierFactory)
-
-if PropertyStore is None:
- PropertyStoreTest.skip = importErrorMessage
-
-
-def propertyName(name):
- return PropertyName("http://calendarserver.org/ns/test/", name)
Copied: CalendarServer/trunk/txdav/base/propertystore/test/test_sql.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/test/test_sql.py)
===================================================================
--- CalendarServer/trunk/txdav/base/propertystore/test/test_sql.py (rev 0)
+++ CalendarServer/trunk/txdav/base/propertystore/test/test_sql.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,88 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Tests for txdav.caldav.datastore.postgres, mostly based on
+L{txdav.caldav.datastore.test.common}.
+"""
+
+from twisted.internet.defer import inlineCallbacks
+
+from txdav.caldav.datastore.test.common import StubNotifierFactory
+
+from txdav.common.datastore.test.util import SQLStoreBuilder
+
+from txdav.base.propertystore.base import PropertyName
+from txdav.base.propertystore.test import base
+
+try:
+ from txdav.base.propertystore.sql import PropertyStore
+except ImportError, e:
+ PropertyStore = None
+ importErrorMessage = str(e)
+
+
+
+theStoreBuilder = SQLStoreBuilder()
+buildStore = theStoreBuilder.buildStore
+
+
+class PropertyStoreTest(base.PropertyStoreTest):
+
+ def _preTest(self):
+ self._txn = self.store.newTransaction()
+ self.propertyStore = self.propertyStore1 = PropertyStore(
+ "user01", self._txn, 1
+ )
+ self.propertyStore2 = PropertyStore("user01", self._txn, 1)
+ self.propertyStore2._setPerUserUID("user02")
+
+ self.addCleanup(self._postTest)
+
+ def _postTest(self):
+ if hasattr(self, "_txn"):
+ self._txn.commit()
+ delattr(self, "_txn")
+ self.propertyStore = self.propertyStore1 = self.propertyStore2 = None
+
+ def _changed(self, store):
+ if hasattr(self, "_txn"):
+ self._txn.commit()
+ delattr(self, "_txn")
+ self._txn = self.store.newTransaction()
+ self.propertyStore1._txn = self._txn
+ self.propertyStore2._txn = self._txn
+
+ def _abort(self, store):
+ if hasattr(self, "_txn"):
+ self._txn.abort()
+ delattr(self, "_txn")
+
+ self._txn = self.store.newTransaction()
+ self.propertyStore1._txn = self._txn
+ self.propertyStore2._txn = self._txn
+
+ @inlineCallbacks
+ def setUp(self):
+ self.notifierFactory = StubNotifierFactory()
+ self.store = yield buildStore(self, self.notifierFactory)
+
+if PropertyStore is None:
+ PropertyStoreTest.skip = importErrorMessage
+
+
+def propertyName(name):
+ return PropertyName("http://calendarserver.org/ns/test/", name)
Deleted: CalendarServer/trunk/txdav/base/propertystore/test/test_xattr.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/test/test_xattr.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/propertystore/test/test_xattr.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,93 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-from twext.web2.dav.element.base import WebDAVTextElement
-
-"""
-Property store tests.
-"""
-
-from twext.python.filepath import CachingFilePath as FilePath
-
-from txdav.base.propertystore.base import PropertyName
-
-from txdav.base.propertystore.test import base
-
-try:
- from txdav.base.propertystore.xattr import PropertyStore
- from xattr import xattr
-except ImportError, e:
- PropertyStore = None
- importErrorMessage = str(e)
-
-
-class PropertyStoreTest(base.PropertyStoreTest):
- def setUp(self):
- tempDir = FilePath(self.mktemp())
- tempDir.makedirs()
- tempFile = tempDir.child("test")
- tempFile.touch()
- self.propertyStore = self.propertyStore1 = PropertyStore(
- "user01", lambda : tempFile
- )
- self.propertyStore2 = PropertyStore("user01", lambda : tempFile)
- self.propertyStore2._setPerUserUID("user02")
-
- def test_init(self):
- store = self.propertyStore
- self.failUnless(isinstance(store.attrs, xattr))
- self.assertEquals(store.removed, set())
- self.assertEquals(store.modified, {})
-
- def test_abort(self):
- super(PropertyStoreTest, self).test_abort()
- store = self.propertyStore
- self.assertEquals(store.removed, set())
- self.assertEquals(store.modified, {})
-
- def test_compress(self):
-
- class DummyProperty (WebDAVTextElement):
- namespace = "http://calendarserver.org/ns/"
- name = "dummy"
-
- name = PropertyName.fromElement(DummyProperty)
- compressedKey = self.propertyStore._encodeKey((name, self.propertyStore._defaultuser))
- uncompressedKey = self.propertyStore._encodeKey((name, self.propertyStore._defaultuser), compressNamespace=False)
-
- self.propertyStore[name] = DummyProperty.fromString("data")
- self.propertyStore.flush()
- self.assertEqual(self.propertyStore[name], DummyProperty.fromString("data"))
- self.assertTrue(compressedKey in self.propertyStore.attrs)
- self.assertFalse(uncompressedKey in self.propertyStore.attrs)
-
- def test_compress_upgrade(self):
-
- class DummyProperty (WebDAVTextElement):
- namespace = "http://calendarserver.org/ns/"
- name = "dummy"
-
- name = PropertyName.fromElement(DummyProperty)
- uncompressedKey = self.propertyStore._encodeKey((name, self.propertyStore._defaultuser), compressNamespace=False)
- self.propertyStore.attrs[uncompressedKey] = DummyProperty.fromString("data").toxml()
- self.assertEqual(self.propertyStore[name], DummyProperty.fromString("data"))
- self.assertRaises(KeyError, lambda:self.propertyStore.attrs[uncompressedKey])
-
-if PropertyStore is None:
- PropertyStoreTest.skip = importErrorMessage
-
-
-def propertyName(name):
- return PropertyName("http://calendarserver.org/ns/test/", name)
Copied: CalendarServer/trunk/txdav/base/propertystore/test/test_xattr.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/test/test_xattr.py)
===================================================================
--- CalendarServer/trunk/txdav/base/propertystore/test/test_xattr.py (rev 0)
+++ CalendarServer/trunk/txdav/base/propertystore/test/test_xattr.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,93 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+from twext.web2.dav.element.base import WebDAVTextElement
+
+"""
+Property store tests.
+"""
+
+from twext.python.filepath import CachingFilePath as FilePath
+
+from txdav.base.propertystore.base import PropertyName
+
+from txdav.base.propertystore.test import base
+
+try:
+ from txdav.base.propertystore.xattr import PropertyStore
+ from xattr import xattr
+except ImportError, e:
+ PropertyStore = None
+ importErrorMessage = str(e)
+
+
+class PropertyStoreTest(base.PropertyStoreTest):
+ def setUp(self):
+ tempDir = FilePath(self.mktemp())
+ tempDir.makedirs()
+ tempFile = tempDir.child("test")
+ tempFile.touch()
+ self.propertyStore = self.propertyStore1 = PropertyStore(
+ "user01", lambda : tempFile
+ )
+ self.propertyStore2 = PropertyStore("user01", lambda : tempFile)
+ self.propertyStore2._setPerUserUID("user02")
+
+ def test_init(self):
+ store = self.propertyStore
+ self.failUnless(isinstance(store.attrs, xattr))
+ self.assertEquals(store.removed, set())
+ self.assertEquals(store.modified, {})
+
+ def test_abort(self):
+ super(PropertyStoreTest, self).test_abort()
+ store = self.propertyStore
+ self.assertEquals(store.removed, set())
+ self.assertEquals(store.modified, {})
+
+ def test_compress(self):
+
+ class DummyProperty (WebDAVTextElement):
+ namespace = "http://calendarserver.org/ns/"
+ name = "dummy"
+
+ name = PropertyName.fromElement(DummyProperty)
+ compressedKey = self.propertyStore._encodeKey((name, self.propertyStore._defaultuser))
+ uncompressedKey = self.propertyStore._encodeKey((name, self.propertyStore._defaultuser), compressNamespace=False)
+
+ self.propertyStore[name] = DummyProperty.fromString("data")
+ self.propertyStore.flush()
+ self.assertEqual(self.propertyStore[name], DummyProperty.fromString("data"))
+ self.assertTrue(compressedKey in self.propertyStore.attrs)
+ self.assertFalse(uncompressedKey in self.propertyStore.attrs)
+
+ def test_compress_upgrade(self):
+
+ class DummyProperty (WebDAVTextElement):
+ namespace = "http://calendarserver.org/ns/"
+ name = "dummy"
+
+ name = PropertyName.fromElement(DummyProperty)
+ uncompressedKey = self.propertyStore._encodeKey((name, self.propertyStore._defaultuser), compressNamespace=False)
+ self.propertyStore.attrs[uncompressedKey] = DummyProperty.fromString("data").toxml()
+ self.assertEqual(self.propertyStore[name], DummyProperty.fromString("data"))
+ self.assertRaises(KeyError, lambda:self.propertyStore.attrs[uncompressedKey])
+
+if PropertyStore is None:
+ PropertyStoreTest.skip = importErrorMessage
+
+
+def propertyName(name):
+ return PropertyName("http://calendarserver.org/ns/test/", name)
Deleted: CalendarServer/trunk/txdav/base/propertystore/xattr.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/xattr.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/base/propertystore/xattr.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,300 +0,0 @@
-# -*- test-case-name: txdav.base.propertystore.test.test_xattr,txdav.caldav.datastore,txdav.carddav.datastore -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Property store using filesystem extended attributes.
-"""
-
-from __future__ import absolute_import
-
-__all__ = [
- "PropertyStore",
-]
-
-import sys
-import errno
-import urllib
-from zlib import compress, decompress, error as ZlibError
-from cPickle import UnpicklingError, loads as unpickle
-from xattr import xattr
-
-from twext.web2.dav.davxml import WebDAVDocument
-
-from txdav.base.propertystore.base import AbstractPropertyStore, PropertyName, validKey
-from txdav.idav import PropertyStoreError
-
-
-#
-# RFC 2518 Section 12.13.1 says that removal of non-existing property
-# is not an error. python-xattr on Linux fails with ENODATA in this
-# case. On OS X, the xattr library fails with ENOATTR, which CPython
-# does not expose. Its value is 93.
-#
-if sys.platform is "darwin":
- if hasattr(errno, "ENOATTR"):
- _ERRNO_NO_ATTR = errno.ENOATTR
- else:
- _ERRNO_NO_ATTR = 93
-else:
- _ERRNO_NO_ATTR = errno.ENODATA
-
-
-class PropertyStore(AbstractPropertyStore):
- """
- Property store using filesystem extended attributes.
-
- This implementation uses Bob Ippolito's xattr package, available from::
-
- http://undefined.org/python/#xattr
- """
- #
- # Dead properties are stored as extended attributes on disk. In order to
- # avoid conflicts with other attributes, prefix dead property names.
- #
- deadPropertyXattrPrefix = "WebDAV:"
-
- # Linux seems to require that attribute names use a "user." prefix.
- if sys.platform == "linux2":
- deadPropertyXattrPrefix = "user."
-
- # There is a 127 character limit for xattr keys so we need to compress/expand
- # overly long namespaces to help stay under that limit now that GUIDs are also
- # encoded in the keys.
- _namespaceCompress = {
- "urn:ietf:params:xml:ns:caldav" :"CALDAV:",
- "urn:ietf:params:xml:ns:carddav" :"CARDDAV:",
- "http://calendarserver.org/ns/" :"CS:",
- "http://cal.me.com/_namespace/" :"ME:",
- "http://twistedmatrix.com/xml_namespace/dav/" :"TD:",
- "http://twistedmatrix.com/xml_namespace/dav/private/" :"TDP:",
- }
- _namespaceExpand = dict([ (v, k) for k, v in _namespaceCompress.iteritems() ])
-
- def __init__(self, defaultuser, pathFactory):
- """
- Initialize a L{PropertyStore}.
-
- @param pathFactory: a 0-arg callable that returns the L{CachingFilePath} to set extended attributes on.
- """
- super(PropertyStore, self).__init__(defaultuser)
-
- self._pathFactory = pathFactory
- # self.attrs = xattr(path.path)
- self.removed = set()
- self.modified = {}
-
-
- @property
- def path(self):
- return self._pathFactory()
-
- @property
- def attrs(self):
- return xattr(self.path.path)
-
- def __str__(self):
- return "<%s %s>" % (self.__class__.__name__, self.path.path)
-
- def _encodeKey(self, effective, compressNamespace=True):
-
- qname, uid = effective
- namespace = self._namespaceCompress.get(qname.namespace, qname.namespace) if compressNamespace else qname.namespace
- result = urllib.quote("{%s}%s" % (namespace, qname.name), safe="{}:")
- if uid:
- result = uid + result
- r = self.deadPropertyXattrPrefix + result
- return r
-
- def _decodeKey(self, name):
-
- name = urllib.unquote(name[len(self.deadPropertyXattrPrefix):])
-
- index1 = name.find("{")
- index2 = name.find("}")
-
- if (index1 is - 1 or index2 is - 1 or not len(name) > index2):
- raise ValueError("Invalid encoded name: %r" % (name,))
- if index1 == 0:
- uid = None
- else:
- uid = name[:index1]
- propnamespace = name[index1 + 1:index2]
- propnamespace = self._namespaceExpand.get(propnamespace, propnamespace)
- propname = name[index2 + 1:]
-
- return PropertyName(propnamespace, propname), uid
-
- #
- # Required implementations
- #
-
- def _getitem_uid(self, key, uid):
- validKey(key)
- effectiveKey = (key, uid)
-
- if effectiveKey in self.modified:
- return self.modified[effectiveKey]
-
- if effectiveKey in self.removed:
- raise KeyError(key)
-
- try:
- try:
- data = self.attrs[self._encodeKey(effectiveKey)]
- except IOError, e:
- if e.errno in [_ERRNO_NO_ATTR, errno.ENOENT]:
- raise KeyError(key)
- raise PropertyStoreError(e)
- except KeyError:
- # Check for uncompressed namespace
- if effectiveKey[0].namespace in self._namespaceCompress:
- try:
- data = self.attrs[self._encodeKey(effectiveKey, compressNamespace=False)]
- except IOError, e:
- raise KeyError(key)
-
- try:
- # Write it back using the compressed format
- self.attrs[self._encodeKey(effectiveKey)] = data
- del self.attrs[self._encodeKey(effectiveKey, compressNamespace=False)]
- except IOError, e:
- msg = "Unable to upgrade property to compressed namespace: %s" % (
- key.toString()
- )
- self.log_error(msg)
- raise PropertyStoreError(msg)
- else:
- raise
-
- #
- # Unserialize XML data from an xattr. The storage format has changed
- # over time:
- #
- # 1- Started with XML
- # 2- Started compressing the XML due to limits on xattr size
- # 3- Switched to pickle which is faster, still compressing
- # 4- Back to compressed XML for interoperability, size
- #
- # We only write the current format, but we also read the old
- # ones for compatibility.
- #
- legacy = False
-
- try:
- data = decompress(data)
- except ZlibError:
- legacy = True
-
- try:
- doc = WebDAVDocument.fromString(data)
- except ValueError:
- try:
- doc = unpickle(data)
- except UnpicklingError:
- msg = "Invalid property value stored on server: %s %s" % (
- key.toString(), data
- )
- self.log_error(msg)
- raise PropertyStoreError(msg)
- else:
- legacy = True
-
- if legacy:
- # XXX untested: CDT catches this though.
- self._setitem_uid(key, doc.root_element, uid)
-
- return doc.root_element
-
- def _setitem_uid(self, key, value, uid):
- validKey(key)
- effectiveKey = (key, uid)
-
- if effectiveKey in self.removed:
- self.removed.remove(effectiveKey)
- self.modified[effectiveKey] = value
-
- def _delitem_uid(self, key, uid):
- validKey(key)
- effectiveKey = (key, uid)
-
- if effectiveKey in self.modified:
- del self.modified[effectiveKey]
- elif self._encodeKey(effectiveKey) not in self.attrs:
- raise KeyError(key)
-
- self.removed.add(effectiveKey)
-
- def _keys_uid(self, uid):
- seen = set()
-
- try:
- iterattr = iter(self.attrs)
- except IOError, e:
- if e.errno != errno.ENOENT:
- raise
- iterattr = iter(())
-
- for key in iterattr:
- effectivekey = self._decodeKey(key)
- if effectivekey[1] == uid and effectivekey not in self.removed:
- seen.add(effectivekey)
- yield effectivekey[0]
-
- for effectivekey in self.modified:
- if effectivekey[1] == uid and effectivekey not in seen:
- yield effectivekey[0]
-
- #
- # I/O
- #
-
- def flush(self):
- # FIXME: The transaction may have deleted the file, and then obviously
- # flushing would fail. Let's try to detect that scenario. The
- # transaction should not attempt to flush properties if it is also
- # deleting the resource, though, and there are other reasons we might
- # want to know about that the file doesn't exist, so this should be
- # fixed.
- self.path.changed()
- if not self.path.exists():
- return
-
- attrs = self.attrs
- removed = self.removed
- modified = self.modified
-
- for key in removed:
- assert key not in modified
- try:
- del attrs[self._encodeKey(key)]
- except KeyError:
- pass
- except IOError, e:
- if e.errno != _ERRNO_NO_ATTR:
- raise
-
- for key in modified:
- assert key not in removed
- value = modified[key]
- attrs[self._encodeKey(key)] = compress(value.toxml())
-
- self.removed.clear()
- self.modified.clear()
-
- def abort(self):
- self.removed.clear()
- self.modified.clear()
Copied: CalendarServer/trunk/txdav/base/propertystore/xattr.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/base/propertystore/xattr.py)
===================================================================
--- CalendarServer/trunk/txdav/base/propertystore/xattr.py (rev 0)
+++ CalendarServer/trunk/txdav/base/propertystore/xattr.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,300 @@
+# -*- test-case-name: txdav.base.propertystore.test.test_xattr,txdav.caldav.datastore,txdav.carddav.datastore -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Property store using filesystem extended attributes.
+"""
+
+from __future__ import absolute_import
+
+__all__ = [
+ "PropertyStore",
+]
+
+import sys
+import errno
+import urllib
+from zlib import compress, decompress, error as ZlibError
+from cPickle import UnpicklingError, loads as unpickle
+from xattr import xattr
+
+from twext.web2.dav.davxml import WebDAVDocument
+
+from txdav.base.propertystore.base import AbstractPropertyStore, PropertyName, validKey
+from txdav.idav import PropertyStoreError
+
+
+#
+# RFC 2518 Section 12.13.1 says that removal of non-existing property
+# is not an error. python-xattr on Linux fails with ENODATA in this
+# case. On OS X, the xattr library fails with ENOATTR, which CPython
+# does not expose. Its value is 93.
+#
+if sys.platform is "darwin":
+ if hasattr(errno, "ENOATTR"):
+ _ERRNO_NO_ATTR = errno.ENOATTR
+ else:
+ _ERRNO_NO_ATTR = 93
+else:
+ _ERRNO_NO_ATTR = errno.ENODATA
+
+
+class PropertyStore(AbstractPropertyStore):
+ """
+ Property store using filesystem extended attributes.
+
+ This implementation uses Bob Ippolito's xattr package, available from::
+
+ http://undefined.org/python/#xattr
+ """
+ #
+ # Dead properties are stored as extended attributes on disk. In order to
+ # avoid conflicts with other attributes, prefix dead property names.
+ #
+ deadPropertyXattrPrefix = "WebDAV:"
+
+ # Linux seems to require that attribute names use a "user." prefix.
+ if sys.platform == "linux2":
+ deadPropertyXattrPrefix = "user."
+
+ # There is a 127 character limit for xattr keys so we need to compress/expand
+ # overly long namespaces to help stay under that limit now that GUIDs are also
+ # encoded in the keys.
+ _namespaceCompress = {
+ "urn:ietf:params:xml:ns:caldav" :"CALDAV:",
+ "urn:ietf:params:xml:ns:carddav" :"CARDDAV:",
+ "http://calendarserver.org/ns/" :"CS:",
+ "http://cal.me.com/_namespace/" :"ME:",
+ "http://twistedmatrix.com/xml_namespace/dav/" :"TD:",
+ "http://twistedmatrix.com/xml_namespace/dav/private/" :"TDP:",
+ }
+ _namespaceExpand = dict([ (v, k) for k, v in _namespaceCompress.iteritems() ])
+
+ def __init__(self, defaultuser, pathFactory):
+ """
+ Initialize a L{PropertyStore}.
+
+ @param pathFactory: a 0-arg callable that returns the L{CachingFilePath} to set extended attributes on.
+ """
+ super(PropertyStore, self).__init__(defaultuser)
+
+ self._pathFactory = pathFactory
+ # self.attrs = xattr(path.path)
+ self.removed = set()
+ self.modified = {}
+
+
+ @property
+ def path(self):
+ return self._pathFactory()
+
+ @property
+ def attrs(self):
+ return xattr(self.path.path)
+
+ def __str__(self):
+ return "<%s %s>" % (self.__class__.__name__, self.path.path)
+
+ def _encodeKey(self, effective, compressNamespace=True):
+
+ qname, uid = effective
+ namespace = self._namespaceCompress.get(qname.namespace, qname.namespace) if compressNamespace else qname.namespace
+ result = urllib.quote("{%s}%s" % (namespace, qname.name), safe="{}:")
+ if uid:
+ result = uid + result
+ r = self.deadPropertyXattrPrefix + result
+ return r
+
+ def _decodeKey(self, name):
+
+ name = urllib.unquote(name[len(self.deadPropertyXattrPrefix):])
+
+ index1 = name.find("{")
+ index2 = name.find("}")
+
+ if (index1 is - 1 or index2 is - 1 or not len(name) > index2):
+ raise ValueError("Invalid encoded name: %r" % (name,))
+ if index1 == 0:
+ uid = None
+ else:
+ uid = name[:index1]
+ propnamespace = name[index1 + 1:index2]
+ propnamespace = self._namespaceExpand.get(propnamespace, propnamespace)
+ propname = name[index2 + 1:]
+
+ return PropertyName(propnamespace, propname), uid
+
+ #
+ # Required implementations
+ #
+
+ def _getitem_uid(self, key, uid):
+ validKey(key)
+ effectiveKey = (key, uid)
+
+ if effectiveKey in self.modified:
+ return self.modified[effectiveKey]
+
+ if effectiveKey in self.removed:
+ raise KeyError(key)
+
+ try:
+ try:
+ data = self.attrs[self._encodeKey(effectiveKey)]
+ except IOError, e:
+ if e.errno in [_ERRNO_NO_ATTR, errno.ENOENT]:
+ raise KeyError(key)
+ raise PropertyStoreError(e)
+ except KeyError:
+ # Check for uncompressed namespace
+ if effectiveKey[0].namespace in self._namespaceCompress:
+ try:
+ data = self.attrs[self._encodeKey(effectiveKey, compressNamespace=False)]
+ except IOError, e:
+ raise KeyError(key)
+
+ try:
+ # Write it back using the compressed format
+ self.attrs[self._encodeKey(effectiveKey)] = data
+ del self.attrs[self._encodeKey(effectiveKey, compressNamespace=False)]
+ except IOError, e:
+ msg = "Unable to upgrade property to compressed namespace: %s" % (
+ key.toString()
+ )
+ self.log_error(msg)
+ raise PropertyStoreError(msg)
+ else:
+ raise
+
+ #
+ # Unserialize XML data from an xattr. The storage format has changed
+ # over time:
+ #
+ # 1- Started with XML
+ # 2- Started compressing the XML due to limits on xattr size
+ # 3- Switched to pickle which is faster, still compressing
+ # 4- Back to compressed XML for interoperability, size
+ #
+ # We only write the current format, but we also read the old
+ # ones for compatibility.
+ #
+ legacy = False
+
+ try:
+ data = decompress(data)
+ except ZlibError:
+ legacy = True
+
+ try:
+ doc = WebDAVDocument.fromString(data)
+ except ValueError:
+ try:
+ doc = unpickle(data)
+ except UnpicklingError:
+ msg = "Invalid property value stored on server: %s %s" % (
+ key.toString(), data
+ )
+ self.log_error(msg)
+ raise PropertyStoreError(msg)
+ else:
+ legacy = True
+
+ if legacy:
+ # XXX untested: CDT catches this though.
+ self._setitem_uid(key, doc.root_element, uid)
+
+ return doc.root_element
+
+ def _setitem_uid(self, key, value, uid):
+ validKey(key)
+ effectiveKey = (key, uid)
+
+ if effectiveKey in self.removed:
+ self.removed.remove(effectiveKey)
+ self.modified[effectiveKey] = value
+
+ def _delitem_uid(self, key, uid):
+ validKey(key)
+ effectiveKey = (key, uid)
+
+ if effectiveKey in self.modified:
+ del self.modified[effectiveKey]
+ elif self._encodeKey(effectiveKey) not in self.attrs:
+ raise KeyError(key)
+
+ self.removed.add(effectiveKey)
+
+ def _keys_uid(self, uid):
+ seen = set()
+
+ try:
+ iterattr = iter(self.attrs)
+ except IOError, e:
+ if e.errno != errno.ENOENT:
+ raise
+ iterattr = iter(())
+
+ for key in iterattr:
+ effectivekey = self._decodeKey(key)
+ if effectivekey[1] == uid and effectivekey not in self.removed:
+ seen.add(effectivekey)
+ yield effectivekey[0]
+
+ for effectivekey in self.modified:
+ if effectivekey[1] == uid and effectivekey not in seen:
+ yield effectivekey[0]
+
+ #
+ # I/O
+ #
+
+ def flush(self):
+ # FIXME: The transaction may have deleted the file, and then obviously
+ # flushing would fail. Let's try to detect that scenario. The
+ # transaction should not attempt to flush properties if it is also
+ # deleting the resource, though, and there are other reasons we might
+ # want to know about that the file doesn't exist, so this should be
+ # fixed.
+ self.path.changed()
+ if not self.path.exists():
+ return
+
+ attrs = self.attrs
+ removed = self.removed
+ modified = self.modified
+
+ for key in removed:
+ assert key not in modified
+ try:
+ del attrs[self._encodeKey(key)]
+ except KeyError:
+ pass
+ except IOError, e:
+ if e.errno != _ERRNO_NO_ATTR:
+ raise
+
+ for key in modified:
+ assert key not in removed
+ value = modified[key]
+ attrs[self._encodeKey(key)] = compress(value.toxml())
+
+ self.removed.clear()
+ self.modified.clear()
+
+ def abort(self):
+ self.removed.clear()
+ self.modified.clear()
Deleted: CalendarServer/trunk/txdav/caldav/__init__.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/__init__.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,19 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-CalDAV support for Twisted.
-"""
Copied: CalendarServer/trunk/txdav/caldav/__init__.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/__init__.py)
===================================================================
--- CalendarServer/trunk/txdav/caldav/__init__.py (rev 0)
+++ CalendarServer/trunk/txdav/caldav/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,19 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+CalDAV support for Twisted.
+"""
Deleted: CalendarServer/trunk/txdav/caldav/datastore/__init__.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/__init__.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,19 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Calendar stores.
-"""
Copied: CalendarServer/trunk/txdav/caldav/datastore/__init__.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/__init__.py)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/__init__.py (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,19 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Calendar stores.
+"""
Deleted: CalendarServer/trunk/txdav/caldav/datastore/file.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/file.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/file.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,542 +0,0 @@
-# -*- test-case-name: txdav.caldav.datastore.test.test_file -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-File calendar store.
-"""
-
-__all__ = [
- "CalendarStore",
- "CalendarStoreTransaction",
- "CalendarHome",
- "Calendar",
- "CalendarObject",
-]
-
-import hashlib
-
-from errno import ENOENT
-
-from twisted.internet.interfaces import ITransport
-from twisted.python.failure import Failure
-
-from txdav.base.propertystore.xattr import PropertyStore
-
-from twext.python.vcomponent import InvalidICalendarDataError
-from twext.python.vcomponent import VComponent
-from twext.web2.dav.element.rfc2518 import ResourceType, GETContentType
-from twext.web2.dav.resource import TwistedGETContentMD5
-from twext.web2.http_headers import generateContentType
-
-from twistedcaldav import caldavxml, customxml
-from twistedcaldav.caldavxml import ScheduleCalendarTransp, Opaque
-from twistedcaldav.index import Index as OldIndex, IndexSchedule as OldInboxIndex
-from twistedcaldav.sharing import InvitesDatabase
-
-from txdav.caldav.icalendarstore import IAttachment
-from txdav.caldav.icalendarstore import ICalendar, ICalendarObject
-from txdav.caldav.icalendarstore import ICalendarHome
-
-from txdav.caldav.datastore.util import (
- validateCalendarComponent, dropboxIDFromCalendarObject
-)
-
-from txdav.common.datastore.file import (
- CommonDataStore, CommonStoreTransaction, CommonHome, CommonHomeChild,
- CommonObjectResource
-, CommonStubResource)
-
-from txdav.common.icommondatastore import (NoSuchObjectResourceError,
- InternalDataStoreError)
-from txdav.base.datastore.file import writeOperation, hidden, FileMetaDataMixin
-from txdav.base.propertystore.base import PropertyName
-
-from zope.interface import implements
-
-CalendarStore = CommonDataStore
-
-CalendarStoreTransaction = CommonStoreTransaction
-
-class CalendarHome(CommonHome):
- implements(ICalendarHome)
-
- def __init__(self, uid, path, calendarStore, transaction, notifier):
- super(CalendarHome, self).__init__(uid, path, calendarStore, transaction, notifier)
-
- self._childClass = Calendar
-
-
- def calendarWithName(self, name):
- if name in ('dropbox', 'notifications', 'freebusy'):
- # "dropbox" is a file storage area, not a calendar.
- return None
- else:
- return self.childWithName(name)
-
-
- createCalendarWithName = CommonHome.createChildWithName
- removeCalendarWithName = CommonHome.removeChildWithName
-
- def calendars(self):
- """
- Return a generator of the child resource objects.
- """
- for child in self.children():
- if child.name() in ('dropbox', 'notification'):
- continue
- yield child
-
- def listCalendars(self):
- """
- Return a generator of the child resource names.
- """
- for name in self.listChildren():
- if name in ('dropbox', 'notification'):
- continue
- yield name
-
-
- def calendarObjectWithDropboxID(self, dropboxID):
- """
- Implement lookup with brute-force scanning.
- """
- for calendar in self.calendars():
- for calendarObject in calendar.calendarObjects():
- if dropboxID == calendarObject.dropboxID():
- return calendarObject
-
-
- @property
- def _calendarStore(self):
- return self._dataStore
-
-
- def createdHome(self):
- self.createCalendarWithName("calendar")
- defaultCal = self.calendarWithName("calendar")
- props = defaultCal.properties()
- props[PropertyName(*ScheduleCalendarTransp.qname())] = ScheduleCalendarTransp(
- Opaque())
- self.createCalendarWithName("inbox")
-
-
-
-class Calendar(CommonHomeChild):
- """
- File-based implementation of L{ICalendar}.
- """
- implements(ICalendar)
-
- def __init__(self, name, calendarHome, notifier, realName=None):
- """
- Initialize a calendar pointing at a path on disk.
-
- @param name: the subdirectory of calendarHome where this calendar
- resides.
- @type name: C{str}
-
- @param calendarHome: the home containing this calendar.
- @type calendarHome: L{CalendarHome}
-
- @param realName: If this calendar was just created, the name which it
- will eventually have on disk.
- @type realName: C{str}
- """
- super(Calendar, self).__init__(name, calendarHome, notifier,
- realName=realName)
-
- self._index = Index(self)
- self._invites = Invites(self)
- self._objectResourceClass = CalendarObject
-
-
- @property
- def _calendarHome(self):
- return self._home
-
-
- def resourceType(self):
- return ResourceType.calendar #@UndefinedVariable
-
-
- ownerCalendarHome = CommonHomeChild.ownerHome
- calendarObjects = CommonHomeChild.objectResources
- listCalendarObjects = CommonHomeChild.listObjectResources
- calendarObjectWithName = CommonHomeChild.objectResourceWithName
- calendarObjectWithUID = CommonHomeChild.objectResourceWithUID
- createCalendarObjectWithName = CommonHomeChild.createObjectResourceWithName
- removeCalendarObjectWithName = CommonHomeChild.removeObjectResourceWithName
- removeCalendarObjectWithUID = CommonHomeChild.removeObjectResourceWithUID
- calendarObjectsSinceToken = CommonHomeChild.objectResourcesSinceToken
-
-
- def calendarObjectsInTimeRange(self, start, end, timeZone):
- raise NotImplementedError()
-
-
- def initPropertyStore(self, props):
- # Setup peruser special properties
- props.setSpecialProperties(
- (
- PropertyName.fromElement(caldavxml.CalendarDescription),
- PropertyName.fromElement(caldavxml.CalendarTimeZone),
- ),
- (
- PropertyName.fromElement(customxml.GETCTag),
- PropertyName.fromElement(caldavxml.SupportedCalendarComponentSet),
- PropertyName.fromElement(caldavxml.ScheduleCalendarTransp),
- ),
- )
-
-
-
-class CalendarObject(CommonObjectResource):
- """
- @ivar _path: The path of the .ics file on disk
-
- @type _path: L{FilePath}
- """
- implements(ICalendarObject)
-
- def __init__(self, name, calendar):
- super(CalendarObject, self).__init__(name, calendar)
- self._attachments = {}
-
-
- @property
- def _calendar(self):
- return self._parentCollection
-
-
- def calendar(self):
- return self._calendar
-
-
- @writeOperation
- def setComponent(self, component, inserting=False):
- validateCalendarComponent(self, self._calendar, component, inserting)
-
- self._calendar.retrieveOldIndex().addResource(
- self.name(), component
- )
-
- self._component = component
- # FIXME: needs to clear text cache
-
- def do():
- # Mark all properties as dirty, so they can be added back
- # to the newly updated file.
- self.properties().update(self.properties())
-
- backup = None
- if self._path.exists():
- backup = hidden(self._path.temporarySibling())
- self._path.moveTo(backup)
- fh = self._path.open("w")
- try:
- # FIXME: concurrency problem; if this write is interrupted
- # halfway through, the underlying file will be corrupt.
- fh.write(str(component))
- finally:
- fh.close()
-
- # Now re-write the original properties on the updated file
- self.properties().flush()
-
- def undo():
- if backup:
- backup.moveTo(self._path)
- else:
- self._path.remove()
- return undo
- self._transaction.addOperation(do, "set calendar component %r" % (self.name(),))
- if self._calendar._notifier:
- self._transaction.postCommit(self._calendar._notifier.notify)
-
- def component(self):
- if self._component is not None:
- return self._component
- text = self.text()
-
- try:
- component = VComponent.fromString(text)
- except InvalidICalendarDataError, e:
- raise InternalDataStoreError(
- "File corruption detected (%s) in file: %s"
- % (e, self._path.path)
- )
- return component
-
-
- def text(self):
- if self._component is not None:
- return str(self._component)
- try:
- fh = self._path.open()
- except IOError, e:
- if e[0] == ENOENT:
- raise NoSuchObjectResourceError(self)
- else:
- raise
-
- try:
- text = fh.read()
- finally:
- fh.close()
-
- if not (
- text.startswith("BEGIN:VCALENDAR\r\n") or
- text.endswith("\r\nEND:VCALENDAR\r\n")
- ):
- raise InternalDataStoreError(
- "File corruption detected (improper start) in file: %s"
- % (self._path.path,)
- )
- return text
-
- iCalendarText = text
-
- def uid(self):
- if not hasattr(self, "_uid"):
- self._uid = self.component().resourceUID()
- return self._uid
-
- def componentType(self):
- if not hasattr(self, "_componentType"):
- self._componentType = self.component().mainType()
- return self._componentType
-
- def organizer(self):
- return self.component().getOrganizer()
-
-
- def createAttachmentWithName(self, name, contentType):
- """
- Implement L{ICalendarObject.removeAttachmentWithName}.
- """
- # Make a (FIXME: temp, remember rollbacks) file in dropbox-land
- attachment = Attachment(self, name)
- self._attachments[name] = attachment
- return attachment.store(contentType)
-
-
- def removeAttachmentWithName(self, name):
- """
- Implement L{ICalendarObject.removeAttachmentWithName}.
- """
- # FIXME: rollback, tests for rollback
- self._dropboxPath().child(name).remove()
- if name in self._attachments:
- del self._attachments[name]
-
-
- def attachmentWithName(self, name):
- # Attachments can be local or remote, but right now we only care about
- # local. So we're going to base this on the listing of files in the
- # dropbox and not on the calendar data. However, we COULD examine the
- # 'attach' properties.
-
- if name in self._attachments:
- return self._attachments[name]
- # FIXME: cache consistently (put it in self._attachments)
- if self._dropboxPath().child(name).exists():
- return Attachment(self, name)
- else:
- # FIXME: test for non-existent attachment.
- return None
-
-
- def attendeesCanManageAttachments(self):
- return self.component().hasPropertyInAnyComponent("X-APPLE-DROPBOX")
-
-
- def dropboxID(self):
- return dropboxIDFromCalendarObject(self)
-
-
- def _dropboxPath(self):
- dropboxPath = self._parentCollection._home._path.child(
- "dropbox"
- ).child(self.dropboxID())
- if not dropboxPath.isdir():
- dropboxPath.makedirs()
- return dropboxPath
-
-
- def attachments(self):
- # See comment on attachmentWithName.
- return [Attachment(self, name)
- for name in self._dropboxPath().listdir()]
-
- def initPropertyStore(self, props):
- # Setup peruser special properties
- props.setSpecialProperties(
- (
- ),
- (
- PropertyName.fromElement(customxml.TwistedCalendarAccessProperty),
- PropertyName.fromElement(customxml.TwistedSchedulingObjectResource),
- PropertyName.fromElement(caldavxml.ScheduleTag),
- PropertyName.fromElement(customxml.TwistedScheduleMatchETags),
- PropertyName.fromElement(customxml.TwistedCalendarHasPrivateCommentsProperty),
- PropertyName.fromElement(caldavxml.Originator),
- PropertyName.fromElement(caldavxml.Recipient),
- PropertyName.fromElement(customxml.ScheduleChanges),
- ),
- )
-
-
-contentTypeKey = PropertyName.fromElement(GETContentType)
-md5key = PropertyName.fromElement(TwistedGETContentMD5)
-
-class AttachmentStorageTransport(object):
-
- implements(ITransport)
-
- def __init__(self, attachment, contentType):
- """
-
- @param attachment:
- @type attachment:
- """
- self._attachment = attachment
- self._contentType = contentType
- self._file = self._attachment._path.open("w")
-
-
- def write(self, data):
- # FIXME: multiple chunks
- self._file.write(data)
-
-
- def loseConnection(self):
- # FIXME: do anything
- self._file.close()
-
- md5 = hashlib.md5(self._attachment._path.getContent()).hexdigest()
- props = self._attachment.properties()
- props[contentTypeKey] = GETContentType(generateContentType(self._contentType))
- props[md5key] = TwistedGETContentMD5.fromString(md5)
- props.flush()
-
-
-
-class Attachment(FileMetaDataMixin):
- """
- An L{Attachment} is a container for the data associated with a I{locally-
- stored} calendar attachment. That is to say, there will only be
- L{Attachment} objects present on the I{organizer's} copy of and event, and
- only for C{ATTACH} properties where this server is storing the resource.
- (For example, the organizer may specify an C{ATTACH} property that
- references an URI on a remote server.)
- """
-
- implements(IAttachment)
-
- def __init__(self, calendarObject, name):
- self._calendarObject = calendarObject
- self._name = name
-
-
- def name(self):
- return self._name
-
-
- def properties(self):
- uid = self._calendarObject._parentCollection._home.uid()
- return PropertyStore(uid, lambda :self._path)
-
-
- def store(self, contentType):
- return AttachmentStorageTransport(self, contentType)
-
- def retrieve(self, protocol):
- # FIXME: makeConnection
- # FIXME: actually stream
- # FIMXE: connectionLost
- protocol.dataReceived(self._path.getContent())
- # FIXME: ConnectionDone, not NotImplementedError
- protocol.connectionLost(Failure(NotImplementedError()))
-
- @property
- def _path(self):
- dropboxPath = self._calendarObject._dropboxPath()
- return dropboxPath.child(self.name())
-
-
-
-class CalendarStubResource(CommonStubResource):
- """
- Just enough resource to keep the calendar's sql DB classes going.
- """
-
- def isCalendarCollection(self):
- return True
-
-
- def getChild(self, name):
- calendarObject = self.resource.calendarObjectWithName(name)
- if calendarObject:
- class ChildResource(object):
- def __init__(self, calendarObject):
- self.calendarObject = calendarObject
-
- def iCalendar(self):
- return self.calendarObject.component()
-
- return ChildResource(calendarObject)
- else:
- return None
-
-
-
-class Index(object):
- #
- # OK, here's where we get ugly.
- # The index code needs to be rewritten also, but in the meantime...
- #
- def __init__(self, calendar):
- self.calendar = calendar
- stubResource = CalendarStubResource(calendar)
- if self.calendar.name() == 'inbox':
- indexClass = OldInboxIndex
- else:
- indexClass = OldIndex
- self._oldIndex = indexClass(stubResource)
-
-
- def calendarObjects(self):
- calendar = self.calendar
- for name, uid, componentType in self._oldIndex.bruteForceSearch():
- calendarObject = calendar.calendarObjectWithName(name)
-
- # Precache what we found in the index
- calendarObject._uid = uid
- calendarObject._componentType = componentType
-
- yield calendarObject
-
-
-class Invites(object):
- #
- # OK, here's where we get ugly.
- # The index code needs to be rewritten also, but in the meantime...
- #
- def __init__(self, calendar):
- self.calendar = calendar
- stubResource = CalendarStubResource(calendar)
- self._oldInvites = InvitesDatabase(stubResource)
Copied: CalendarServer/trunk/txdav/caldav/datastore/file.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/file.py)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/file.py (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/file.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,542 @@
+# -*- test-case-name: txdav.caldav.datastore.test.test_file -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+File calendar store.
+"""
+
+__all__ = [
+ "CalendarStore",
+ "CalendarStoreTransaction",
+ "CalendarHome",
+ "Calendar",
+ "CalendarObject",
+]
+
+import hashlib
+
+from errno import ENOENT
+
+from twisted.internet.interfaces import ITransport
+from twisted.python.failure import Failure
+
+from txdav.base.propertystore.xattr import PropertyStore
+
+from twext.python.vcomponent import InvalidICalendarDataError
+from twext.python.vcomponent import VComponent
+from twext.web2.dav.element.rfc2518 import ResourceType, GETContentType
+from twext.web2.dav.resource import TwistedGETContentMD5
+from twext.web2.http_headers import generateContentType
+
+from twistedcaldav import caldavxml, customxml
+from twistedcaldav.caldavxml import ScheduleCalendarTransp, Opaque
+from twistedcaldav.index import Index as OldIndex, IndexSchedule as OldInboxIndex
+from twistedcaldav.sharing import InvitesDatabase
+
+from txdav.caldav.icalendarstore import IAttachment
+from txdav.caldav.icalendarstore import ICalendar, ICalendarObject
+from txdav.caldav.icalendarstore import ICalendarHome
+
+from txdav.caldav.datastore.util import (
+ validateCalendarComponent, dropboxIDFromCalendarObject
+)
+
+from txdav.common.datastore.file import (
+ CommonDataStore, CommonStoreTransaction, CommonHome, CommonHomeChild,
+ CommonObjectResource
+, CommonStubResource)
+
+from txdav.common.icommondatastore import (NoSuchObjectResourceError,
+ InternalDataStoreError)
+from txdav.base.datastore.file import writeOperation, hidden, FileMetaDataMixin
+from txdav.base.propertystore.base import PropertyName
+
+from zope.interface import implements
+
+CalendarStore = CommonDataStore
+
+CalendarStoreTransaction = CommonStoreTransaction
+
+class CalendarHome(CommonHome):
+ implements(ICalendarHome)
+
+ def __init__(self, uid, path, calendarStore, transaction, notifier):
+ super(CalendarHome, self).__init__(uid, path, calendarStore, transaction, notifier)
+
+ self._childClass = Calendar
+
+
+ def calendarWithName(self, name):
+ if name in ('dropbox', 'notifications', 'freebusy'):
+ # "dropbox" is a file storage area, not a calendar.
+ return None
+ else:
+ return self.childWithName(name)
+
+
+ createCalendarWithName = CommonHome.createChildWithName
+ removeCalendarWithName = CommonHome.removeChildWithName
+
+ def calendars(self):
+ """
+ Return a generator of the child resource objects.
+ """
+ for child in self.children():
+ if child.name() in ('dropbox', 'notification'):
+ continue
+ yield child
+
+ def listCalendars(self):
+ """
+ Return a generator of the child resource names.
+ """
+ for name in self.listChildren():
+ if name in ('dropbox', 'notification'):
+ continue
+ yield name
+
+
+ def calendarObjectWithDropboxID(self, dropboxID):
+ """
+ Implement lookup with brute-force scanning.
+ """
+ for calendar in self.calendars():
+ for calendarObject in calendar.calendarObjects():
+ if dropboxID == calendarObject.dropboxID():
+ return calendarObject
+
+
+ @property
+ def _calendarStore(self):
+ return self._dataStore
+
+
+ def createdHome(self):
+ self.createCalendarWithName("calendar")
+ defaultCal = self.calendarWithName("calendar")
+ props = defaultCal.properties()
+ props[PropertyName(*ScheduleCalendarTransp.qname())] = ScheduleCalendarTransp(
+ Opaque())
+ self.createCalendarWithName("inbox")
+
+
+
+class Calendar(CommonHomeChild):
+ """
+ File-based implementation of L{ICalendar}.
+ """
+ implements(ICalendar)
+
+ def __init__(self, name, calendarHome, notifier, realName=None):
+ """
+ Initialize a calendar pointing at a path on disk.
+
+ @param name: the subdirectory of calendarHome where this calendar
+ resides.
+ @type name: C{str}
+
+ @param calendarHome: the home containing this calendar.
+ @type calendarHome: L{CalendarHome}
+
+ @param realName: If this calendar was just created, the name which it
+ will eventually have on disk.
+ @type realName: C{str}
+ """
+ super(Calendar, self).__init__(name, calendarHome, notifier,
+ realName=realName)
+
+ self._index = Index(self)
+ self._invites = Invites(self)
+ self._objectResourceClass = CalendarObject
+
+
+ @property
+ def _calendarHome(self):
+ return self._home
+
+
+ def resourceType(self):
+ return ResourceType.calendar #@UndefinedVariable
+
+
+ ownerCalendarHome = CommonHomeChild.ownerHome
+ calendarObjects = CommonHomeChild.objectResources
+ listCalendarObjects = CommonHomeChild.listObjectResources
+ calendarObjectWithName = CommonHomeChild.objectResourceWithName
+ calendarObjectWithUID = CommonHomeChild.objectResourceWithUID
+ createCalendarObjectWithName = CommonHomeChild.createObjectResourceWithName
+ removeCalendarObjectWithName = CommonHomeChild.removeObjectResourceWithName
+ removeCalendarObjectWithUID = CommonHomeChild.removeObjectResourceWithUID
+ calendarObjectsSinceToken = CommonHomeChild.objectResourcesSinceToken
+
+
+ def calendarObjectsInTimeRange(self, start, end, timeZone):
+ raise NotImplementedError()
+
+
+ def initPropertyStore(self, props):
+ # Setup peruser special properties
+ props.setSpecialProperties(
+ (
+ PropertyName.fromElement(caldavxml.CalendarDescription),
+ PropertyName.fromElement(caldavxml.CalendarTimeZone),
+ ),
+ (
+ PropertyName.fromElement(customxml.GETCTag),
+ PropertyName.fromElement(caldavxml.SupportedCalendarComponentSet),
+ PropertyName.fromElement(caldavxml.ScheduleCalendarTransp),
+ ),
+ )
+
+
+
+class CalendarObject(CommonObjectResource):
+ """
+ @ivar _path: The path of the .ics file on disk
+
+ @type _path: L{FilePath}
+ """
+ implements(ICalendarObject)
+
+ def __init__(self, name, calendar):
+ super(CalendarObject, self).__init__(name, calendar)
+ self._attachments = {}
+
+
+ @property
+ def _calendar(self):
+ return self._parentCollection
+
+
+ def calendar(self):
+ return self._calendar
+
+
+ @writeOperation
+ def setComponent(self, component, inserting=False):
+ validateCalendarComponent(self, self._calendar, component, inserting)
+
+ self._calendar.retrieveOldIndex().addResource(
+ self.name(), component
+ )
+
+ self._component = component
+ # FIXME: needs to clear text cache
+
+ def do():
+ # Mark all properties as dirty, so they can be added back
+ # to the newly updated file.
+ self.properties().update(self.properties())
+
+ backup = None
+ if self._path.exists():
+ backup = hidden(self._path.temporarySibling())
+ self._path.moveTo(backup)
+ fh = self._path.open("w")
+ try:
+ # FIXME: concurrency problem; if this write is interrupted
+ # halfway through, the underlying file will be corrupt.
+ fh.write(str(component))
+ finally:
+ fh.close()
+
+ # Now re-write the original properties on the updated file
+ self.properties().flush()
+
+ def undo():
+ if backup:
+ backup.moveTo(self._path)
+ else:
+ self._path.remove()
+ return undo
+ self._transaction.addOperation(do, "set calendar component %r" % (self.name(),))
+ if self._calendar._notifier:
+ self._transaction.postCommit(self._calendar._notifier.notify)
+
+ def component(self):
+ if self._component is not None:
+ return self._component
+ text = self.text()
+
+ try:
+ component = VComponent.fromString(text)
+ except InvalidICalendarDataError, e:
+ raise InternalDataStoreError(
+ "File corruption detected (%s) in file: %s"
+ % (e, self._path.path)
+ )
+ return component
+
+
+ def text(self):
+ if self._component is not None:
+ return str(self._component)
+ try:
+ fh = self._path.open()
+ except IOError, e:
+ if e[0] == ENOENT:
+ raise NoSuchObjectResourceError(self)
+ else:
+ raise
+
+ try:
+ text = fh.read()
+ finally:
+ fh.close()
+
+ if not (
+ text.startswith("BEGIN:VCALENDAR\r\n") or
+ text.endswith("\r\nEND:VCALENDAR\r\n")
+ ):
+ raise InternalDataStoreError(
+ "File corruption detected (improper start) in file: %s"
+ % (self._path.path,)
+ )
+ return text
+
+ iCalendarText = text
+
+ def uid(self):
+ if not hasattr(self, "_uid"):
+ self._uid = self.component().resourceUID()
+ return self._uid
+
+ def componentType(self):
+ if not hasattr(self, "_componentType"):
+ self._componentType = self.component().mainType()
+ return self._componentType
+
+ def organizer(self):
+ return self.component().getOrganizer()
+
+
+ def createAttachmentWithName(self, name, contentType):
+ """
+ Implement L{ICalendarObject.removeAttachmentWithName}.
+ """
+ # Make a (FIXME: temp, remember rollbacks) file in dropbox-land
+ attachment = Attachment(self, name)
+ self._attachments[name] = attachment
+ return attachment.store(contentType)
+
+
+ def removeAttachmentWithName(self, name):
+ """
+ Implement L{ICalendarObject.removeAttachmentWithName}.
+ """
+ # FIXME: rollback, tests for rollback
+ self._dropboxPath().child(name).remove()
+ if name in self._attachments:
+ del self._attachments[name]
+
+
+ def attachmentWithName(self, name):
+ # Attachments can be local or remote, but right now we only care about
+ # local. So we're going to base this on the listing of files in the
+ # dropbox and not on the calendar data. However, we COULD examine the
+ # 'attach' properties.
+
+ if name in self._attachments:
+ return self._attachments[name]
+ # FIXME: cache consistently (put it in self._attachments)
+ if self._dropboxPath().child(name).exists():
+ return Attachment(self, name)
+ else:
+ # FIXME: test for non-existent attachment.
+ return None
+
+
+ def attendeesCanManageAttachments(self):
+ return self.component().hasPropertyInAnyComponent("X-APPLE-DROPBOX")
+
+
+ def dropboxID(self):
+ return dropboxIDFromCalendarObject(self)
+
+
+ def _dropboxPath(self):
+ dropboxPath = self._parentCollection._home._path.child(
+ "dropbox"
+ ).child(self.dropboxID())
+ if not dropboxPath.isdir():
+ dropboxPath.makedirs()
+ return dropboxPath
+
+
+ def attachments(self):
+ # See comment on attachmentWithName.
+ return [Attachment(self, name)
+ for name in self._dropboxPath().listdir()]
+
+ def initPropertyStore(self, props):
+ # Setup peruser special properties
+ props.setSpecialProperties(
+ (
+ ),
+ (
+ PropertyName.fromElement(customxml.TwistedCalendarAccessProperty),
+ PropertyName.fromElement(customxml.TwistedSchedulingObjectResource),
+ PropertyName.fromElement(caldavxml.ScheduleTag),
+ PropertyName.fromElement(customxml.TwistedScheduleMatchETags),
+ PropertyName.fromElement(customxml.TwistedCalendarHasPrivateCommentsProperty),
+ PropertyName.fromElement(caldavxml.Originator),
+ PropertyName.fromElement(caldavxml.Recipient),
+ PropertyName.fromElement(customxml.ScheduleChanges),
+ ),
+ )
+
+
+contentTypeKey = PropertyName.fromElement(GETContentType)
+md5key = PropertyName.fromElement(TwistedGETContentMD5)
+
+class AttachmentStorageTransport(object):
+
+ implements(ITransport)
+
+ def __init__(self, attachment, contentType):
+ """
+
+ @param attachment:
+ @type attachment:
+ """
+ self._attachment = attachment
+ self._contentType = contentType
+ self._file = self._attachment._path.open("w")
+
+
+ def write(self, data):
+ # FIXME: multiple chunks
+ self._file.write(data)
+
+
+ def loseConnection(self):
+ # FIXME: do anything
+ self._file.close()
+
+ md5 = hashlib.md5(self._attachment._path.getContent()).hexdigest()
+ props = self._attachment.properties()
+ props[contentTypeKey] = GETContentType(generateContentType(self._contentType))
+ props[md5key] = TwistedGETContentMD5.fromString(md5)
+ props.flush()
+
+
+
+class Attachment(FileMetaDataMixin):
+ """
+ An L{Attachment} is a container for the data associated with a I{locally-
+ stored} calendar attachment. That is to say, there will only be
+ L{Attachment} objects present on the I{organizer's} copy of and event, and
+ only for C{ATTACH} properties where this server is storing the resource.
+ (For example, the organizer may specify an C{ATTACH} property that
+ references an URI on a remote server.)
+ """
+
+ implements(IAttachment)
+
+ def __init__(self, calendarObject, name):
+ self._calendarObject = calendarObject
+ self._name = name
+
+
+ def name(self):
+ return self._name
+
+
+ def properties(self):
+ uid = self._calendarObject._parentCollection._home.uid()
+ return PropertyStore(uid, lambda :self._path)
+
+
+ def store(self, contentType):
+ return AttachmentStorageTransport(self, contentType)
+
+ def retrieve(self, protocol):
+ # FIXME: makeConnection
+ # FIXME: actually stream
+ # FIMXE: connectionLost
+ protocol.dataReceived(self._path.getContent())
+ # FIXME: ConnectionDone, not NotImplementedError
+ protocol.connectionLost(Failure(NotImplementedError()))
+
+ @property
+ def _path(self):
+ dropboxPath = self._calendarObject._dropboxPath()
+ return dropboxPath.child(self.name())
+
+
+
+class CalendarStubResource(CommonStubResource):
+ """
+ Just enough resource to keep the calendar's sql DB classes going.
+ """
+
+ def isCalendarCollection(self):
+ return True
+
+
+ def getChild(self, name):
+ calendarObject = self.resource.calendarObjectWithName(name)
+ if calendarObject:
+ class ChildResource(object):
+ def __init__(self, calendarObject):
+ self.calendarObject = calendarObject
+
+ def iCalendar(self):
+ return self.calendarObject.component()
+
+ return ChildResource(calendarObject)
+ else:
+ return None
+
+
+
+class Index(object):
+ #
+ # OK, here's where we get ugly.
+ # The index code needs to be rewritten also, but in the meantime...
+ #
+ def __init__(self, calendar):
+ self.calendar = calendar
+ stubResource = CalendarStubResource(calendar)
+ if self.calendar.name() == 'inbox':
+ indexClass = OldInboxIndex
+ else:
+ indexClass = OldIndex
+ self._oldIndex = indexClass(stubResource)
+
+
+ def calendarObjects(self):
+ calendar = self.calendar
+ for name, uid, componentType in self._oldIndex.bruteForceSearch():
+ calendarObject = calendar.calendarObjectWithName(name)
+
+ # Precache what we found in the index
+ calendarObject._uid = uid
+ calendarObject._componentType = componentType
+
+ yield calendarObject
+
+
+class Invites(object):
+ #
+ # OK, here's where we get ugly.
+ # The index code needs to be rewritten also, but in the meantime...
+ #
+ def __init__(self, calendar):
+ self.calendar = calendar
+ stubResource = CalendarStubResource(calendar)
+ self._oldInvites = InvitesDatabase(stubResource)
Deleted: CalendarServer/trunk/txdav/caldav/datastore/scheduling.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/scheduling.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/scheduling.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,161 +0,0 @@
-# -*- test-case-name: txdav.caldav.datastore.test.test_scheduling -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-from zope.interface.declarations import implements
-from txdav.caldav.icalendarstore import ICalendarHome, ICalendar, ICalendarObject,\
- ICalendarTransaction
-from txdav.idav import IDataStore
-from twisted.python.util import FancyEqMixin
-from twisted.python.components import proxyForInterface
-
-
-
-class ImplicitTransaction(
- proxyForInterface(ICalendarTransaction,
- originalAttribute="_transaction")):
- """
- Wrapper around an L{ICalendarStoreTransaction}.
- """
-
- def __init__(self, transaction):
- """
- Initialize an L{ImplicitTransaction}.
-
- @type transaction: L{ICalendarStoreTransaction}
- """
- self._transaction = transaction
-
-
- def calendarHomeWithUID(self, uid, create=False):
- # FIXME: 'create' flag
- newHome = super(ImplicitTransaction, self
- ).calendarHomeWithUID(uid, create)
-# return ImplicitCalendarHome(newHome, self)
- if newHome is None:
- return None
- else:
- # FIXME: relay transaction
- return ImplicitCalendarHome(newHome, None)
-
-
-
-class ImplicitCalendarHome(
- proxyForInterface(ICalendarHome, "_calendarHome")
- ):
-
- implements(ICalendarHome)
-
- def __init__(self, calendarHome, transaction):
- """
- Initialize L{ImplicitCalendarHome} with an underlying
- calendar home and L{ImplicitTransaction}.
- """
- self._calendarHome = calendarHome
- self._transaction = transaction
-
-
-# def properties(self):
-# # FIXME: wrap?
-# return self._calendarHome.properties()
-
- def calendars(self):
- for calendar in super(ImplicitCalendarHome, self).calendars():
- yield ImplicitCalendar(self, calendar)
-
- def createCalendarWithName(self, name):
- self._calendarHome.createCalendarWithName(name)
-
- def removeCalendarWithName(self, name):
- self._calendarHome.removeCalendarWithName(name)
-
-
- def calendarWithName(self, name):
- calendar = self._calendarHome.calendarWithName(name)
- if calendar is not None:
- return ImplicitCalendar(self, calendar)
- else:
- return None
-
-
-
-class ImplicitCalendarObject(object):
- implements(ICalendarObject)
- def setComponent(self, component): ""
- def component(self): ""
- def iCalendarText(self): ""
- def uid(self): ""
- def componentType(self): ""
- def organizer(self): ""
- def properties(self):""
-
-
-
-class ImplicitCalendar(FancyEqMixin,
- proxyForInterface(ICalendar, "_subCalendar")):
-
- compareAttributes = ['_subCalendar', '_parentHome']
-
- def __init__(self, parentHome, subCalendar):
- self._parentHome = parentHome
- self._subCalendar = subCalendar
-
-# def ownerCalendarHome(self):
-# return self._parentHome
-# def calendarObjects(self):
-# # FIXME: wrap
-# return self._subCalendar.calendarObjects()
-# def calendarObjectWithUID(self, uid): ""
-# def createCalendarObjectWithName(self, name, component):
-# # FIXME: implement most of StoreCalendarObjectResource here!
-# self._subCalendar.createCalendarObjectWithName(name, component)
-# def removeCalendarObjectWithName(self, name):
-# # FIXME: implement deletion logic here!
-# return self._subCalendar.removeCalendarObjectWithName(name)
-# def removeCalendarObjectWithUID(self, uid): ""
-# def syncToken(self): ""
-# def calendarObjectsInTimeRange(self, start, end, timeZone): ""
-# def calendarObjectsSinceToken(self, token): ""
-# def properties(self):
-# # FIXME: probably need to wrap this as well
-# return self._subCalendar.properties()
-#
-# def calendarObjectWithName(self, name):
-# #FIXME: wrap
-# return self._subCalendar.calendarObjectWithName(name)
-
-
-class ImplicitStore(object):
- """
- This is a wrapper around an L{ICalendarStore} that implements implicit
- scheduling.
- """
-
- implements(IDataStore)
-
- def __init__(self, calendarStore):
- """
- Create an L{ImplicitStore} wrapped around another
- L{ICalendarStore} provider.
- """
- self._calendarStore = calendarStore
-
-
- def newTransaction(self, label="unlabeled"):
- """
- Wrap an underlying L{ITransaction}.
- """
- return ImplicitTransaction(
- self._calendarStore.newTransaction(label))
Copied: CalendarServer/trunk/txdav/caldav/datastore/scheduling.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/scheduling.py)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/scheduling.py (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/scheduling.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,161 @@
+# -*- test-case-name: txdav.caldav.datastore.test.test_scheduling -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+from zope.interface.declarations import implements
+from txdav.caldav.icalendarstore import ICalendarHome, ICalendar, ICalendarObject,\
+ ICalendarTransaction
+from txdav.idav import IDataStore
+from twisted.python.util import FancyEqMixin
+from twisted.python.components import proxyForInterface
+
+
+
+class ImplicitTransaction(
+ proxyForInterface(ICalendarTransaction,
+ originalAttribute="_transaction")):
+ """
+ Wrapper around an L{ICalendarStoreTransaction}.
+ """
+
+ def __init__(self, transaction):
+ """
+ Initialize an L{ImplicitTransaction}.
+
+ @type transaction: L{ICalendarStoreTransaction}
+ """
+ self._transaction = transaction
+
+
+ def calendarHomeWithUID(self, uid, create=False):
+ # FIXME: 'create' flag
+ newHome = super(ImplicitTransaction, self
+ ).calendarHomeWithUID(uid, create)
+# return ImplicitCalendarHome(newHome, self)
+ if newHome is None:
+ return None
+ else:
+ # FIXME: relay transaction
+ return ImplicitCalendarHome(newHome, None)
+
+
+
+class ImplicitCalendarHome(
+ proxyForInterface(ICalendarHome, "_calendarHome")
+ ):
+
+ implements(ICalendarHome)
+
+ def __init__(self, calendarHome, transaction):
+ """
+ Initialize L{ImplicitCalendarHome} with an underlying
+ calendar home and L{ImplicitTransaction}.
+ """
+ self._calendarHome = calendarHome
+ self._transaction = transaction
+
+
+# def properties(self):
+# # FIXME: wrap?
+# return self._calendarHome.properties()
+
+ def calendars(self):
+ for calendar in super(ImplicitCalendarHome, self).calendars():
+ yield ImplicitCalendar(self, calendar)
+
+ def createCalendarWithName(self, name):
+ self._calendarHome.createCalendarWithName(name)
+
+ def removeCalendarWithName(self, name):
+ self._calendarHome.removeCalendarWithName(name)
+
+
+ def calendarWithName(self, name):
+ calendar = self._calendarHome.calendarWithName(name)
+ if calendar is not None:
+ return ImplicitCalendar(self, calendar)
+ else:
+ return None
+
+
+
+class ImplicitCalendarObject(object):
+ implements(ICalendarObject)
+ def setComponent(self, component): ""
+ def component(self): ""
+ def iCalendarText(self): ""
+ def uid(self): ""
+ def componentType(self): ""
+ def organizer(self): ""
+ def properties(self):""
+
+
+
+class ImplicitCalendar(FancyEqMixin,
+ proxyForInterface(ICalendar, "_subCalendar")):
+
+ compareAttributes = ['_subCalendar', '_parentHome']
+
+ def __init__(self, parentHome, subCalendar):
+ self._parentHome = parentHome
+ self._subCalendar = subCalendar
+
+# def ownerCalendarHome(self):
+# return self._parentHome
+# def calendarObjects(self):
+# # FIXME: wrap
+# return self._subCalendar.calendarObjects()
+# def calendarObjectWithUID(self, uid): ""
+# def createCalendarObjectWithName(self, name, component):
+# # FIXME: implement most of StoreCalendarObjectResource here!
+# self._subCalendar.createCalendarObjectWithName(name, component)
+# def removeCalendarObjectWithName(self, name):
+# # FIXME: implement deletion logic here!
+# return self._subCalendar.removeCalendarObjectWithName(name)
+# def removeCalendarObjectWithUID(self, uid): ""
+# def syncToken(self): ""
+# def calendarObjectsInTimeRange(self, start, end, timeZone): ""
+# def calendarObjectsSinceToken(self, token): ""
+# def properties(self):
+# # FIXME: probably need to wrap this as well
+# return self._subCalendar.properties()
+#
+# def calendarObjectWithName(self, name):
+# #FIXME: wrap
+# return self._subCalendar.calendarObjectWithName(name)
+
+
+class ImplicitStore(object):
+ """
+ This is a wrapper around an L{ICalendarStore} that implements implicit
+ scheduling.
+ """
+
+ implements(IDataStore)
+
+ def __init__(self, calendarStore):
+ """
+ Create an L{ImplicitStore} wrapped around another
+ L{ICalendarStore} provider.
+ """
+ self._calendarStore = calendarStore
+
+
+ def newTransaction(self, label="unlabeled"):
+ """
+ Wrap an underlying L{ITransaction}.
+ """
+ return ImplicitTransaction(
+ self._calendarStore.newTransaction(label))
Deleted: CalendarServer/trunk/txdav/caldav/datastore/sql.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/sql.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/sql.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,634 +0,0 @@
-# -*- test-case-name: txdav.caldav.datastore.test.test_sql -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-__all__ = [
- "CalendarHome",
- "Calendar",
- "CalendarObject",
-]
-
-from twext.python.vcomponent import VComponent
-from twext.web2.dav.element.rfc2518 import ResourceType
-from twext.web2.http_headers import MimeType, generateContentType
-
-from twisted.internet.error import ConnectionLost
-from twisted.internet.interfaces import ITransport
-from twisted.python import hashlib
-from twisted.python.failure import Failure
-
-from twistedcaldav import caldavxml, customxml
-from twistedcaldav.caldavxml import ScheduleCalendarTransp, Opaque
-from twistedcaldav.dateops import normalizeForIndex
-from twistedcaldav.index import IndexedSearchException
-from twistedcaldav.instance import InvalidOverriddenInstanceError
-
-from txdav.caldav.datastore.util import validateCalendarComponent,\
- dropboxIDFromCalendarObject
-from txdav.caldav.icalendarstore import ICalendarHome, ICalendar, ICalendarObject,\
- IAttachment
-
-from txdav.common.datastore.sql import CommonHome, CommonHomeChild,\
- CommonObjectResource
-from txdav.common.datastore.sql_legacy import \
- PostgresLegacyIndexEmulator, PostgresLegacyInvitesEmulator,\
- PostgresLegacySharesEmulator
-from txdav.common.datastore.sql_tables import CALENDAR_TABLE,\
- CALENDAR_BIND_TABLE, CALENDAR_OBJECT_REVISIONS_TABLE, CALENDAR_OBJECT_TABLE,\
- _ATTACHMENTS_MODE_WRITE
-from txdav.base.propertystore.base import PropertyName
-
-from vobject.icalendar import utc
-
-import datetime
-
-from zope.interface.declarations import implements
-
-class CalendarHome(CommonHome):
-
- implements(ICalendarHome)
-
- def __init__(self, transaction, ownerUID, resourceID, notifier):
- super(CalendarHome, self).__init__(transaction, ownerUID, resourceID, notifier)
-
- self._shares = PostgresLegacySharesEmulator(self)
- self._childClass = Calendar
- self._childTable = CALENDAR_TABLE
- self._bindTable = CALENDAR_BIND_TABLE
-
- createCalendarWithName = CommonHome.createChildWithName
- removeCalendarWithName = CommonHome.removeChildWithName
- calendarWithName = CommonHome.childWithName
- calendars = CommonHome.children
- listCalendars = CommonHome.listChildren
-
- def calendarObjectWithDropboxID(self, dropboxID):
- """
- Implement lookup with brute-force scanning.
- """
- for calendar in self.calendars():
- for calendarObject in calendar.calendarObjects():
- if dropboxID == calendarObject.dropboxID():
- return calendarObject
-
-
- def createdHome(self):
- self.createCalendarWithName("calendar")
- defaultCal = self.calendarWithName("calendar")
- props = defaultCal.properties()
- props[PropertyName(*ScheduleCalendarTransp.qname())] = ScheduleCalendarTransp(
- Opaque())
- self.createCalendarWithName("inbox")
-
-class Calendar(CommonHomeChild):
- """
- File-based implementation of L{ICalendar}.
- """
- implements(ICalendar)
-
- def __init__(self, home, name, resourceID, notifier):
- """
- Initialize a calendar pointing at a path on disk.
-
- @param name: the subdirectory of calendarHome where this calendar
- resides.
- @type name: C{str}
-
- @param calendarHome: the home containing this calendar.
- @type calendarHome: L{CalendarHome}
-
- @param realName: If this calendar was just created, the name which it
- will eventually have on disk.
- @type realName: C{str}
- """
- super(Calendar, self).__init__(home, name, resourceID, notifier)
-
- self._index = PostgresLegacyIndexEmulator(self)
- self._invites = PostgresLegacyInvitesEmulator(self)
- self._objectResourceClass = CalendarObject
- self._bindTable = CALENDAR_BIND_TABLE
- self._homeChildTable = CALENDAR_TABLE
- self._revisionsTable = CALENDAR_OBJECT_REVISIONS_TABLE
- self._objectTable = CALENDAR_OBJECT_TABLE
-
-
- @property
- def _calendarHome(self):
- return self._home
-
-
- def resourceType(self):
- return ResourceType.calendar #@UndefinedVariable
-
-
- ownerCalendarHome = CommonHomeChild.ownerHome
- calendarObjects = CommonHomeChild.objectResources
- listCalendarObjects = CommonHomeChild.listObjectResources
- calendarObjectWithName = CommonHomeChild.objectResourceWithName
- calendarObjectWithUID = CommonHomeChild.objectResourceWithUID
- createCalendarObjectWithName = CommonHomeChild.createObjectResourceWithName
- removeCalendarObjectWithName = CommonHomeChild.removeObjectResourceWithName
- removeCalendarObjectWithUID = CommonHomeChild.removeObjectResourceWithUID
- calendarObjectsSinceToken = CommonHomeChild.objectResourcesSinceToken
-
-
- def calendarObjectsInTimeRange(self, start, end, timeZone):
- raise NotImplementedError()
-
-
- def initPropertyStore(self, props):
- # Setup peruser special properties
- props.setSpecialProperties(
- (
- PropertyName.fromElement(caldavxml.CalendarDescription),
- PropertyName.fromElement(caldavxml.CalendarTimeZone),
- ),
- (
- PropertyName.fromElement(customxml.GETCTag),
- PropertyName.fromElement(caldavxml.SupportedCalendarComponentSet),
- PropertyName.fromElement(caldavxml.ScheduleCalendarTransp),
- ),
- )
-
- def contentType(self):
- """
- The content type of Calendar objects is text/calendar.
- """
- return MimeType.fromString("text/calendar; charset=utf-8")
-
-#
-# Duration into the future through which recurrences are expanded in the index
-# by default. This is a caching parameter which affects the size of the index;
-# it does not affect search results beyond this period, but it may affect
-# performance of such a search.
-#
-default_future_expansion_duration = datetime.timedelta(days=365 * 1)
-
-#
-# Maximum duration into the future through which recurrences are expanded in the
-# index. This is a caching parameter which affects the size of the index; it
-# does not affect search results beyond this period, but it may affect
-# performance of such a search.
-#
-# When a search is performed on a time span that goes beyond that which is
-# expanded in the index, we have to open each resource which may have data in
-# that time period. In order to avoid doing that multiple times, we want to
-# cache those results. However, we don't necessarily want to cache all
-# occurrences into some obscenely far-in-the-future date, so we cap the caching
-# period. Searches beyond this period will always be relatively expensive for
-# resources with occurrences beyond this period.
-#
-maximum_future_expansion_duration = datetime.timedelta(days=365 * 5)
-
-icalfbtype_to_indexfbtype = {
- "UNKNOWN" : 0,
- "FREE" : 1,
- "BUSY" : 2,
- "BUSY-UNAVAILABLE": 3,
- "BUSY-TENTATIVE" : 4,
-}
-
-indexfbtype_to_icalfbtype = {
- 0: '?',
- 1: 'F',
- 2: 'B',
- 3: 'U',
- 4: 'T',
-}
-
-def _pathToName(path):
- return path.rsplit(".", 1)[0].split("-", 3)[-1]
-
-class CalendarObject(CommonObjectResource):
- implements(ICalendarObject)
-
- def __init__(self, name, calendar, resid):
- super(CalendarObject, self).__init__(name, calendar, resid)
-
- self._objectTable = CALENDAR_OBJECT_TABLE
-
- @property
- def _calendar(self):
- return self._parentCollection
-
- def calendar(self):
- return self._calendar
-
- def setComponent(self, component, inserting=False):
- validateCalendarComponent(self, self._calendar, component, inserting)
-
- self.updateDatabase(component, inserting=inserting)
- if inserting:
- self._calendar._insertRevision(self._name)
- else:
- self._calendar._updateRevision(self._name)
-
- if self._calendar._notifier:
- self._txn.postCommit(self._calendar._notifier.notify)
-
- def updateDatabase(self, component, expand_until=None, reCreate=False, inserting=False):
- """
- Update the database tables for the new data being written.
-
- @param component: calendar data to store
- @type component: L{Component}
- """
-
- # Decide how far to expand based on the component
- master = component.masterComponent()
- if master is None or not component.isRecurring() and not component.isRecurringUnbounded():
- # When there is no master we have a set of overridden components - index them all.
- # When there is one instance - index it.
- # When bounded - index all.
- expand = datetime.datetime(2100, 1, 1, 0, 0, 0, tzinfo=utc)
- else:
- if expand_until:
- expand = expand_until
- else:
- expand = datetime.date.today() + default_future_expansion_duration
-
- if expand > (datetime.date.today() + maximum_future_expansion_duration):
- raise IndexedSearchException
-
- try:
- instances = component.expandTimeRanges(expand, ignoreInvalidInstances=reCreate)
- except InvalidOverriddenInstanceError, e:
- self.log_err("Invalid instance %s when indexing %s in %s" % (e.rid, self._name, self.resource,))
- raise
-
- componentText = str(component)
- self._objectText = componentText
- organizer = component.getOrganizer()
- if not organizer:
- organizer = ""
-
- # CALENDAR_OBJECT table update
- if inserting:
- self._resourceID = self._txn.execSQL(
- """
- insert into CALENDAR_OBJECT
- (CALENDAR_RESOURCE_ID, RESOURCE_NAME, ICALENDAR_TEXT, ICALENDAR_UID, ICALENDAR_TYPE, ATTACHMENTS_MODE, ORGANIZER, RECURRANCE_MAX)
- values
- (%s, %s, %s, %s, %s, %s, %s, %s)
- returning RESOURCE_ID
- """,
- # FIXME: correct ATTACHMENTS_MODE based on X-APPLE-
- # DROPBOX
- [
- self._calendar._resourceID,
- self._name,
- componentText,
- component.resourceUID(),
- component.resourceType(),
- _ATTACHMENTS_MODE_WRITE,
- organizer,
- normalizeForIndex(instances.limit) if instances.limit else None,
- ]
- )[0][0]
- else:
- self._txn.execSQL(
- """
- update CALENDAR_OBJECT set
- (ICALENDAR_TEXT, ICALENDAR_UID, ICALENDAR_TYPE, ATTACHMENTS_MODE, ORGANIZER, RECURRANCE_MAX, MODIFIED)
- =
- (%s, %s, %s, %s, %s, %s, timezone('UTC', CURRENT_TIMESTAMP))
- where RESOURCE_ID = %s
- """,
- # should really be filling out more fields: ORGANIZER,
- # ORGANIZER_OBJECT, a correct ATTACHMENTS_MODE based on X-APPLE-
- # DROPBOX
- [
- componentText,
- component.resourceUID(),
- component.resourceType(),
- _ATTACHMENTS_MODE_WRITE,
- organizer,
- normalizeForIndex(instances.limit) if instances.limit else None,
- self._resourceID
- ]
- )
-
- # Need to wipe the existing time-range for this and rebuild
- self._txn.execSQL(
- """
- delete from TIME_RANGE where CALENDAR_OBJECT_RESOURCE_ID = %s
- """,
- [
- self._resourceID,
- ],
- )
-
-
- # CALENDAR_OBJECT table update
- for key in instances:
- instance = instances[key]
- start = instance.start.replace(tzinfo=utc)
- end = instance.end.replace(tzinfo=utc)
- float = instance.start.tzinfo is None
- transp = instance.component.propertyValue("TRANSP") == "TRANSPARENT"
- instanceid = self._txn.execSQL(
- """
- insert into TIME_RANGE
- (CALENDAR_RESOURCE_ID, CALENDAR_OBJECT_RESOURCE_ID, FLOATING, START_DATE, END_DATE, FBTYPE, TRANSPARENT)
- values
- (%s, %s, %s, %s, %s, %s, %s)
- returning
- INSTANCE_ID
- """,
- [
- self._calendar._resourceID,
- self._resourceID,
- float,
- start,
- end,
- icalfbtype_to_indexfbtype.get(instance.component.getFBType(), icalfbtype_to_indexfbtype["FREE"]),
- transp,
- ],
- )[0][0]
- peruserdata = component.perUserTransparency(instance.rid)
- for useruid, transp in peruserdata:
- self._txn.execSQL(
- """
- insert into TRANSPARENCY
- (TIME_RANGE_INSTANCE_ID, USER_ID, TRANSPARENT)
- values
- (%s, %s, %s)
- """,
- [
- instanceid,
- useruid,
- transp,
- ],
- )
-
- # Special - for unbounded recurrence we insert a value for "infinity"
- # that will allow an open-ended time-range to always match it.
- if component.isRecurringUnbounded():
- start = datetime.datetime(2100, 1, 1, 0, 0, 0, tzinfo=utc)
- end = datetime.datetime(2100, 1, 1, 1, 0, 0, tzinfo=utc)
- float = False
- instanceid = self._txn.execSQL(
- """
- insert into TIME_RANGE
- (CALENDAR_RESOURCE_ID, CALENDAR_OBJECT_RESOURCE_ID, FLOATING, START_DATE, END_DATE, FBTYPE, TRANSPARENT)
- values
- (%s, %s, %s, %s, %s, %s, %s)
- returning
- INSTANCE_ID
- """,
- [
- self._calendar._resourceID,
- self._resourceID,
- float,
- start,
- end,
- icalfbtype_to_indexfbtype["UNKNOWN"],
- True,
- ],
- )[0][0]
- peruserdata = component.perUserTransparency(None)
- for useruid, transp in peruserdata:
- self._txn.execSQL(
- """
- insert into TRANSPARENCY
- (TIME_RANGE_INSTANCE_ID, USER_ID, TRANSPARENT)
- values
- (%s, %s, %s)
- """,
- [
- instanceid,
- useruid,
- transp,
- ],
- )
-
- def component(self):
- return VComponent.fromString(self.iCalendarText())
-
- def text(self):
- if self._objectText is None:
- text = self._txn.execSQL(
- "select ICALENDAR_TEXT from CALENDAR_OBJECT where "
- "RESOURCE_ID = %s", [self._resourceID]
- )[0][0]
- self._objectText = text
- return text
- else:
- return self._objectText
-
- iCalendarText = text
-
- def uid(self):
- return self.component().resourceUID()
-
- def name(self):
- return self._name
-
- def componentType(self):
- return self.component().mainType()
-
- def organizer(self):
- return self.component().getOrganizer()
-
- def createAttachmentWithName(self, name, contentType):
- path = self._attachmentPath(name)
- attachment = Attachment(self, path)
- self._txn.execSQL("""
- insert into ATTACHMENT (CALENDAR_OBJECT_RESOURCE_ID, CONTENT_TYPE,
- SIZE, MD5, PATH)
- values (%s, %s, %s, %s, %s)
- """,
- [
- self._resourceID, generateContentType(contentType), 0, "",
- attachment._pathValue()
- ]
- )
- return attachment.store(contentType)
-
- def removeAttachmentWithName(self, name):
- attachment = Attachment(self, self._attachmentPath(name))
- self._txn.postCommit(attachment._path.remove)
- self._txn.execSQL("""
- delete from ATTACHMENT where CALENDAR_OBJECT_RESOURCE_ID = %s AND
- PATH = %s
- """, [self._resourceID, attachment._pathValue()])
-
- def attachmentWithName(self, name):
- attachment = Attachment(self, self._attachmentPath(name))
- if attachment._populate():
- return attachment
- else:
- return None
-
- def attendeesCanManageAttachments(self):
- return self.component().hasPropertyInAnyComponent("X-APPLE-DROPBOX")
-
- def dropboxID(self):
- return dropboxIDFromCalendarObject(self)
-
- def _attachmentPath(self, name):
- attachmentRoot = self._txn._store.attachmentsPath
- try:
- attachmentRoot.createDirectory()
- except:
- pass
- return attachmentRoot.child(
- "%s-%s-%s-%s.attachment" % (
- self._calendar._home.uid(), self._calendar.name(),
- self.name(), name
- )
- )
-
- def attachments(self):
- rows = self._txn.execSQL("""
- select PATH from ATTACHMENT where CALENDAR_OBJECT_RESOURCE_ID = %s
- """, [self._resourceID])
- for row in rows:
- demangledName = _pathToName(row[0])
- yield self.attachmentWithName(demangledName)
-
- def initPropertyStore(self, props):
- # Setup peruser special properties
- props.setSpecialProperties(
- (
- ),
- (
- PropertyName.fromElement(customxml.TwistedCalendarAccessProperty),
- PropertyName.fromElement(customxml.TwistedSchedulingObjectResource),
- PropertyName.fromElement(caldavxml.ScheduleTag),
- PropertyName.fromElement(customxml.TwistedScheduleMatchETags),
- PropertyName.fromElement(customxml.TwistedCalendarHasPrivateCommentsProperty),
- PropertyName.fromElement(caldavxml.Originator),
- PropertyName.fromElement(caldavxml.Recipient),
- PropertyName.fromElement(customxml.ScheduleChanges),
- ),
- )
-
- # IDataStoreResource
- def contentType(self):
- """
- The content type of Calendar objects is text/calendar.
- """
- return MimeType.fromString("text/calendar; charset=utf-8")
-
-class AttachmentStorageTransport(object):
-
- implements(ITransport)
-
- def __init__(self, attachment, contentType):
- self.attachment = attachment
- self.contentType = contentType
- self.buf = ''
- self.hash = hashlib.md5()
-
-
- @property
- def _txn(self):
- return self.attachment._txn
-
-
- def write(self, data):
- self.buf += data
- self.hash.update(data)
-
-
- def loseConnection(self):
- self.attachment._path.setContent(self.buf)
- pathValue = self.attachment._pathValue()
- contentTypeString = generateContentType(self.contentType)
- self._txn.execSQL(
- "update ATTACHMENT set CONTENT_TYPE = %s, SIZE = %s, MD5 = %s, MODIFIED = timezone('UTC', CURRENT_TIMESTAMP) "
- "WHERE PATH = %s",
- [contentTypeString, len(self.buf), self.hash.hexdigest(), pathValue]
- )
-
-class Attachment(object):
-
- implements(IAttachment)
-
- def __init__(self, calendarObject, path):
- self._calendarObject = calendarObject
- self._path = path
-
-
- @property
- def _txn(self):
- return self._calendarObject._txn
-
-
- def _populate(self):
- """
- Execute necessary SQL queries to retrieve attributes.
-
- @return: C{True} if this attachment exists, C{False} otherwise.
- """
- rows = self._txn.execSQL(
- """
- select CONTENT_TYPE, SIZE, MD5, extract(EPOCH from CREATED), extract(EPOCH from MODIFIED) from ATTACHMENT where PATH = %s
- """, [self._pathValue()])
- if not rows:
- return False
- self._contentType = MimeType.fromString(rows[0][0])
- self._size = rows[0][1]
- self._md5 = rows[0][2]
- self._created = int(rows[0][3])
- self._modified = int(rows[0][4])
- return True
-
-
- def name(self):
- return _pathToName(self._pathValue())
-
-
- def _pathValue(self):
- """
- Compute the value which should go into the 'path' column for this
- attachment.
- """
- root = self._txn._store.attachmentsPath
- return '/'.join(self._path.segmentsFrom(root))
-
- def properties(self):
- pass # stub
-
-
- def store(self, contentType):
- return AttachmentStorageTransport(self, contentType)
-
-
- def retrieve(self, protocol):
- protocol.dataReceived(self._path.getContent())
- protocol.connectionLost(Failure(ConnectionLost()))
-
-
- # IDataStoreResource
- def contentType(self):
- return self._contentType
-
-
- def md5(self):
- return self._md5
-
-
- def size(self):
- return self._size
-
-
- def created(self):
- return self._created
-
- def modified(self):
- return self._modified
Copied: CalendarServer/trunk/txdav/caldav/datastore/sql.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/sql.py)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/sql.py (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/sql.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,634 @@
+# -*- test-case-name: txdav.caldav.datastore.test.test_sql -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+__all__ = [
+ "CalendarHome",
+ "Calendar",
+ "CalendarObject",
+]
+
+from twext.python.vcomponent import VComponent
+from twext.web2.dav.element.rfc2518 import ResourceType
+from twext.web2.http_headers import MimeType, generateContentType
+
+from twisted.internet.error import ConnectionLost
+from twisted.internet.interfaces import ITransport
+from twisted.python import hashlib
+from twisted.python.failure import Failure
+
+from twistedcaldav import caldavxml, customxml
+from twistedcaldav.caldavxml import ScheduleCalendarTransp, Opaque
+from twistedcaldav.dateops import normalizeForIndex
+from twistedcaldav.index import IndexedSearchException
+from twistedcaldav.instance import InvalidOverriddenInstanceError
+
+from txdav.caldav.datastore.util import validateCalendarComponent,\
+ dropboxIDFromCalendarObject
+from txdav.caldav.icalendarstore import ICalendarHome, ICalendar, ICalendarObject,\
+ IAttachment
+
+from txdav.common.datastore.sql import CommonHome, CommonHomeChild,\
+ CommonObjectResource
+from txdav.common.datastore.sql_legacy import \
+ PostgresLegacyIndexEmulator, PostgresLegacyInvitesEmulator,\
+ PostgresLegacySharesEmulator
+from txdav.common.datastore.sql_tables import CALENDAR_TABLE,\
+ CALENDAR_BIND_TABLE, CALENDAR_OBJECT_REVISIONS_TABLE, CALENDAR_OBJECT_TABLE,\
+ _ATTACHMENTS_MODE_WRITE
+from txdav.base.propertystore.base import PropertyName
+
+from vobject.icalendar import utc
+
+import datetime
+
+from zope.interface.declarations import implements
+
+class CalendarHome(CommonHome):
+
+ implements(ICalendarHome)
+
+ def __init__(self, transaction, ownerUID, resourceID, notifier):
+ super(CalendarHome, self).__init__(transaction, ownerUID, resourceID, notifier)
+
+ self._shares = PostgresLegacySharesEmulator(self)
+ self._childClass = Calendar
+ self._childTable = CALENDAR_TABLE
+ self._bindTable = CALENDAR_BIND_TABLE
+
+ createCalendarWithName = CommonHome.createChildWithName
+ removeCalendarWithName = CommonHome.removeChildWithName
+ calendarWithName = CommonHome.childWithName
+ calendars = CommonHome.children
+ listCalendars = CommonHome.listChildren
+
+ def calendarObjectWithDropboxID(self, dropboxID):
+ """
+ Implement lookup with brute-force scanning.
+ """
+ for calendar in self.calendars():
+ for calendarObject in calendar.calendarObjects():
+ if dropboxID == calendarObject.dropboxID():
+ return calendarObject
+
+
+ def createdHome(self):
+ self.createCalendarWithName("calendar")
+ defaultCal = self.calendarWithName("calendar")
+ props = defaultCal.properties()
+ props[PropertyName(*ScheduleCalendarTransp.qname())] = ScheduleCalendarTransp(
+ Opaque())
+ self.createCalendarWithName("inbox")
+
+class Calendar(CommonHomeChild):
+ """
+ File-based implementation of L{ICalendar}.
+ """
+ implements(ICalendar)
+
+ def __init__(self, home, name, resourceID, notifier):
+ """
+ Initialize a calendar pointing at a path on disk.
+
+ @param name: the subdirectory of calendarHome where this calendar
+ resides.
+ @type name: C{str}
+
+ @param calendarHome: the home containing this calendar.
+ @type calendarHome: L{CalendarHome}
+
+ @param realName: If this calendar was just created, the name which it
+ will eventually have on disk.
+ @type realName: C{str}
+ """
+ super(Calendar, self).__init__(home, name, resourceID, notifier)
+
+ self._index = PostgresLegacyIndexEmulator(self)
+ self._invites = PostgresLegacyInvitesEmulator(self)
+ self._objectResourceClass = CalendarObject
+ self._bindTable = CALENDAR_BIND_TABLE
+ self._homeChildTable = CALENDAR_TABLE
+ self._revisionsTable = CALENDAR_OBJECT_REVISIONS_TABLE
+ self._objectTable = CALENDAR_OBJECT_TABLE
+
+
+ @property
+ def _calendarHome(self):
+ return self._home
+
+
+ def resourceType(self):
+ return ResourceType.calendar #@UndefinedVariable
+
+
+ ownerCalendarHome = CommonHomeChild.ownerHome
+ calendarObjects = CommonHomeChild.objectResources
+ listCalendarObjects = CommonHomeChild.listObjectResources
+ calendarObjectWithName = CommonHomeChild.objectResourceWithName
+ calendarObjectWithUID = CommonHomeChild.objectResourceWithUID
+ createCalendarObjectWithName = CommonHomeChild.createObjectResourceWithName
+ removeCalendarObjectWithName = CommonHomeChild.removeObjectResourceWithName
+ removeCalendarObjectWithUID = CommonHomeChild.removeObjectResourceWithUID
+ calendarObjectsSinceToken = CommonHomeChild.objectResourcesSinceToken
+
+
+ def calendarObjectsInTimeRange(self, start, end, timeZone):
+ raise NotImplementedError()
+
+
+ def initPropertyStore(self, props):
+ # Setup peruser special properties
+ props.setSpecialProperties(
+ (
+ PropertyName.fromElement(caldavxml.CalendarDescription),
+ PropertyName.fromElement(caldavxml.CalendarTimeZone),
+ ),
+ (
+ PropertyName.fromElement(customxml.GETCTag),
+ PropertyName.fromElement(caldavxml.SupportedCalendarComponentSet),
+ PropertyName.fromElement(caldavxml.ScheduleCalendarTransp),
+ ),
+ )
+
+ def contentType(self):
+ """
+ The content type of Calendar objects is text/calendar.
+ """
+ return MimeType.fromString("text/calendar; charset=utf-8")
+
+#
+# Duration into the future through which recurrences are expanded in the index
+# by default. This is a caching parameter which affects the size of the index;
+# it does not affect search results beyond this period, but it may affect
+# performance of such a search.
+#
+default_future_expansion_duration = datetime.timedelta(days=365 * 1)
+
+#
+# Maximum duration into the future through which recurrences are expanded in the
+# index. This is a caching parameter which affects the size of the index; it
+# does not affect search results beyond this period, but it may affect
+# performance of such a search.
+#
+# When a search is performed on a time span that goes beyond that which is
+# expanded in the index, we have to open each resource which may have data in
+# that time period. In order to avoid doing that multiple times, we want to
+# cache those results. However, we don't necessarily want to cache all
+# occurrences into some obscenely far-in-the-future date, so we cap the caching
+# period. Searches beyond this period will always be relatively expensive for
+# resources with occurrences beyond this period.
+#
+maximum_future_expansion_duration = datetime.timedelta(days=365 * 5)
+
+icalfbtype_to_indexfbtype = {
+ "UNKNOWN" : 0,
+ "FREE" : 1,
+ "BUSY" : 2,
+ "BUSY-UNAVAILABLE": 3,
+ "BUSY-TENTATIVE" : 4,
+}
+
+indexfbtype_to_icalfbtype = {
+ 0: '?',
+ 1: 'F',
+ 2: 'B',
+ 3: 'U',
+ 4: 'T',
+}
+
+def _pathToName(path):
+ return path.rsplit(".", 1)[0].split("-", 3)[-1]
+
+class CalendarObject(CommonObjectResource):
+ implements(ICalendarObject)
+
+ def __init__(self, name, calendar, resid):
+ super(CalendarObject, self).__init__(name, calendar, resid)
+
+ self._objectTable = CALENDAR_OBJECT_TABLE
+
+ @property
+ def _calendar(self):
+ return self._parentCollection
+
+ def calendar(self):
+ return self._calendar
+
+ def setComponent(self, component, inserting=False):
+ validateCalendarComponent(self, self._calendar, component, inserting)
+
+ self.updateDatabase(component, inserting=inserting)
+ if inserting:
+ self._calendar._insertRevision(self._name)
+ else:
+ self._calendar._updateRevision(self._name)
+
+ if self._calendar._notifier:
+ self._txn.postCommit(self._calendar._notifier.notify)
+
+ def updateDatabase(self, component, expand_until=None, reCreate=False, inserting=False):
+ """
+ Update the database tables for the new data being written.
+
+ @param component: calendar data to store
+ @type component: L{Component}
+ """
+
+ # Decide how far to expand based on the component
+ master = component.masterComponent()
+ if master is None or not component.isRecurring() and not component.isRecurringUnbounded():
+ # When there is no master we have a set of overridden components - index them all.
+ # When there is one instance - index it.
+ # When bounded - index all.
+ expand = datetime.datetime(2100, 1, 1, 0, 0, 0, tzinfo=utc)
+ else:
+ if expand_until:
+ expand = expand_until
+ else:
+ expand = datetime.date.today() + default_future_expansion_duration
+
+ if expand > (datetime.date.today() + maximum_future_expansion_duration):
+ raise IndexedSearchException
+
+ try:
+ instances = component.expandTimeRanges(expand, ignoreInvalidInstances=reCreate)
+ except InvalidOverriddenInstanceError, e:
+ self.log_err("Invalid instance %s when indexing %s in %s" % (e.rid, self._name, self.resource,))
+ raise
+
+ componentText = str(component)
+ self._objectText = componentText
+ organizer = component.getOrganizer()
+ if not organizer:
+ organizer = ""
+
+ # CALENDAR_OBJECT table update
+ if inserting:
+ self._resourceID = self._txn.execSQL(
+ """
+ insert into CALENDAR_OBJECT
+ (CALENDAR_RESOURCE_ID, RESOURCE_NAME, ICALENDAR_TEXT, ICALENDAR_UID, ICALENDAR_TYPE, ATTACHMENTS_MODE, ORGANIZER, RECURRANCE_MAX)
+ values
+ (%s, %s, %s, %s, %s, %s, %s, %s)
+ returning RESOURCE_ID
+ """,
+ # FIXME: correct ATTACHMENTS_MODE based on X-APPLE-
+ # DROPBOX
+ [
+ self._calendar._resourceID,
+ self._name,
+ componentText,
+ component.resourceUID(),
+ component.resourceType(),
+ _ATTACHMENTS_MODE_WRITE,
+ organizer,
+ normalizeForIndex(instances.limit) if instances.limit else None,
+ ]
+ )[0][0]
+ else:
+ self._txn.execSQL(
+ """
+ update CALENDAR_OBJECT set
+ (ICALENDAR_TEXT, ICALENDAR_UID, ICALENDAR_TYPE, ATTACHMENTS_MODE, ORGANIZER, RECURRANCE_MAX, MODIFIED)
+ =
+ (%s, %s, %s, %s, %s, %s, timezone('UTC', CURRENT_TIMESTAMP))
+ where RESOURCE_ID = %s
+ """,
+ # should really be filling out more fields: ORGANIZER,
+ # ORGANIZER_OBJECT, a correct ATTACHMENTS_MODE based on X-APPLE-
+ # DROPBOX
+ [
+ componentText,
+ component.resourceUID(),
+ component.resourceType(),
+ _ATTACHMENTS_MODE_WRITE,
+ organizer,
+ normalizeForIndex(instances.limit) if instances.limit else None,
+ self._resourceID
+ ]
+ )
+
+ # Need to wipe the existing time-range for this and rebuild
+ self._txn.execSQL(
+ """
+ delete from TIME_RANGE where CALENDAR_OBJECT_RESOURCE_ID = %s
+ """,
+ [
+ self._resourceID,
+ ],
+ )
+
+
+ # CALENDAR_OBJECT table update
+ for key in instances:
+ instance = instances[key]
+ start = instance.start.replace(tzinfo=utc)
+ end = instance.end.replace(tzinfo=utc)
+ float = instance.start.tzinfo is None
+ transp = instance.component.propertyValue("TRANSP") == "TRANSPARENT"
+ instanceid = self._txn.execSQL(
+ """
+ insert into TIME_RANGE
+ (CALENDAR_RESOURCE_ID, CALENDAR_OBJECT_RESOURCE_ID, FLOATING, START_DATE, END_DATE, FBTYPE, TRANSPARENT)
+ values
+ (%s, %s, %s, %s, %s, %s, %s)
+ returning
+ INSTANCE_ID
+ """,
+ [
+ self._calendar._resourceID,
+ self._resourceID,
+ float,
+ start,
+ end,
+ icalfbtype_to_indexfbtype.get(instance.component.getFBType(), icalfbtype_to_indexfbtype["FREE"]),
+ transp,
+ ],
+ )[0][0]
+ peruserdata = component.perUserTransparency(instance.rid)
+ for useruid, transp in peruserdata:
+ self._txn.execSQL(
+ """
+ insert into TRANSPARENCY
+ (TIME_RANGE_INSTANCE_ID, USER_ID, TRANSPARENT)
+ values
+ (%s, %s, %s)
+ """,
+ [
+ instanceid,
+ useruid,
+ transp,
+ ],
+ )
+
+ # Special - for unbounded recurrence we insert a value for "infinity"
+ # that will allow an open-ended time-range to always match it.
+ if component.isRecurringUnbounded():
+ start = datetime.datetime(2100, 1, 1, 0, 0, 0, tzinfo=utc)
+ end = datetime.datetime(2100, 1, 1, 1, 0, 0, tzinfo=utc)
+ float = False
+ instanceid = self._txn.execSQL(
+ """
+ insert into TIME_RANGE
+ (CALENDAR_RESOURCE_ID, CALENDAR_OBJECT_RESOURCE_ID, FLOATING, START_DATE, END_DATE, FBTYPE, TRANSPARENT)
+ values
+ (%s, %s, %s, %s, %s, %s, %s)
+ returning
+ INSTANCE_ID
+ """,
+ [
+ self._calendar._resourceID,
+ self._resourceID,
+ float,
+ start,
+ end,
+ icalfbtype_to_indexfbtype["UNKNOWN"],
+ True,
+ ],
+ )[0][0]
+ peruserdata = component.perUserTransparency(None)
+ for useruid, transp in peruserdata:
+ self._txn.execSQL(
+ """
+ insert into TRANSPARENCY
+ (TIME_RANGE_INSTANCE_ID, USER_ID, TRANSPARENT)
+ values
+ (%s, %s, %s)
+ """,
+ [
+ instanceid,
+ useruid,
+ transp,
+ ],
+ )
+
+ def component(self):
+ return VComponent.fromString(self.iCalendarText())
+
+ def text(self):
+ if self._objectText is None:
+ text = self._txn.execSQL(
+ "select ICALENDAR_TEXT from CALENDAR_OBJECT where "
+ "RESOURCE_ID = %s", [self._resourceID]
+ )[0][0]
+ self._objectText = text
+ return text
+ else:
+ return self._objectText
+
+ iCalendarText = text
+
+ def uid(self):
+ return self.component().resourceUID()
+
+ def name(self):
+ return self._name
+
+ def componentType(self):
+ return self.component().mainType()
+
+ def organizer(self):
+ return self.component().getOrganizer()
+
+ def createAttachmentWithName(self, name, contentType):
+ path = self._attachmentPath(name)
+ attachment = Attachment(self, path)
+ self._txn.execSQL("""
+ insert into ATTACHMENT (CALENDAR_OBJECT_RESOURCE_ID, CONTENT_TYPE,
+ SIZE, MD5, PATH)
+ values (%s, %s, %s, %s, %s)
+ """,
+ [
+ self._resourceID, generateContentType(contentType), 0, "",
+ attachment._pathValue()
+ ]
+ )
+ return attachment.store(contentType)
+
+ def removeAttachmentWithName(self, name):
+ attachment = Attachment(self, self._attachmentPath(name))
+ self._txn.postCommit(attachment._path.remove)
+ self._txn.execSQL("""
+ delete from ATTACHMENT where CALENDAR_OBJECT_RESOURCE_ID = %s AND
+ PATH = %s
+ """, [self._resourceID, attachment._pathValue()])
+
+ def attachmentWithName(self, name):
+ attachment = Attachment(self, self._attachmentPath(name))
+ if attachment._populate():
+ return attachment
+ else:
+ return None
+
+ def attendeesCanManageAttachments(self):
+ return self.component().hasPropertyInAnyComponent("X-APPLE-DROPBOX")
+
+ def dropboxID(self):
+ return dropboxIDFromCalendarObject(self)
+
+ def _attachmentPath(self, name):
+ attachmentRoot = self._txn._store.attachmentsPath
+ try:
+ attachmentRoot.createDirectory()
+ except:
+ pass
+ return attachmentRoot.child(
+ "%s-%s-%s-%s.attachment" % (
+ self._calendar._home.uid(), self._calendar.name(),
+ self.name(), name
+ )
+ )
+
+ def attachments(self):
+ rows = self._txn.execSQL("""
+ select PATH from ATTACHMENT where CALENDAR_OBJECT_RESOURCE_ID = %s
+ """, [self._resourceID])
+ for row in rows:
+ demangledName = _pathToName(row[0])
+ yield self.attachmentWithName(demangledName)
+
+ def initPropertyStore(self, props):
+ # Setup peruser special properties
+ props.setSpecialProperties(
+ (
+ ),
+ (
+ PropertyName.fromElement(customxml.TwistedCalendarAccessProperty),
+ PropertyName.fromElement(customxml.TwistedSchedulingObjectResource),
+ PropertyName.fromElement(caldavxml.ScheduleTag),
+ PropertyName.fromElement(customxml.TwistedScheduleMatchETags),
+ PropertyName.fromElement(customxml.TwistedCalendarHasPrivateCommentsProperty),
+ PropertyName.fromElement(caldavxml.Originator),
+ PropertyName.fromElement(caldavxml.Recipient),
+ PropertyName.fromElement(customxml.ScheduleChanges),
+ ),
+ )
+
+ # IDataStoreResource
+ def contentType(self):
+ """
+ The content type of Calendar objects is text/calendar.
+ """
+ return MimeType.fromString("text/calendar; charset=utf-8")
+
+class AttachmentStorageTransport(object):
+
+ implements(ITransport)
+
+ def __init__(self, attachment, contentType):
+ self.attachment = attachment
+ self.contentType = contentType
+ self.buf = ''
+ self.hash = hashlib.md5()
+
+
+ @property
+ def _txn(self):
+ return self.attachment._txn
+
+
+ def write(self, data):
+ self.buf += data
+ self.hash.update(data)
+
+
+ def loseConnection(self):
+ self.attachment._path.setContent(self.buf)
+ pathValue = self.attachment._pathValue()
+ contentTypeString = generateContentType(self.contentType)
+ self._txn.execSQL(
+ "update ATTACHMENT set CONTENT_TYPE = %s, SIZE = %s, MD5 = %s, MODIFIED = timezone('UTC', CURRENT_TIMESTAMP) "
+ "WHERE PATH = %s",
+ [contentTypeString, len(self.buf), self.hash.hexdigest(), pathValue]
+ )
+
+class Attachment(object):
+
+ implements(IAttachment)
+
+ def __init__(self, calendarObject, path):
+ self._calendarObject = calendarObject
+ self._path = path
+
+
+ @property
+ def _txn(self):
+ return self._calendarObject._txn
+
+
+ def _populate(self):
+ """
+ Execute necessary SQL queries to retrieve attributes.
+
+ @return: C{True} if this attachment exists, C{False} otherwise.
+ """
+ rows = self._txn.execSQL(
+ """
+ select CONTENT_TYPE, SIZE, MD5, extract(EPOCH from CREATED), extract(EPOCH from MODIFIED) from ATTACHMENT where PATH = %s
+ """, [self._pathValue()])
+ if not rows:
+ return False
+ self._contentType = MimeType.fromString(rows[0][0])
+ self._size = rows[0][1]
+ self._md5 = rows[0][2]
+ self._created = int(rows[0][3])
+ self._modified = int(rows[0][4])
+ return True
+
+
+ def name(self):
+ return _pathToName(self._pathValue())
+
+
+ def _pathValue(self):
+ """
+ Compute the value which should go into the 'path' column for this
+ attachment.
+ """
+ root = self._txn._store.attachmentsPath
+ return '/'.join(self._path.segmentsFrom(root))
+
+ def properties(self):
+ pass # stub
+
+
+ def store(self, contentType):
+ return AttachmentStorageTransport(self, contentType)
+
+
+ def retrieve(self, protocol):
+ protocol.dataReceived(self._path.getContent())
+ protocol.connectionLost(Failure(ConnectionLost()))
+
+
+ # IDataStoreResource
+ def contentType(self):
+ return self._contentType
+
+
+ def md5(self):
+ return self._md5
+
+
+ def size(self):
+ return self._size
+
+
+ def created(self):
+ return self._created
+
+ def modified(self):
+ return self._modified
Deleted: CalendarServer/trunk/txdav/caldav/datastore/test/__init__.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/__init__.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,20 +0,0 @@
-# -*- test-case-name: txdav.caldav.datastore.test -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Calendar store tests.
-"""
Copied: CalendarServer/trunk/txdav/caldav/datastore/test/__init__.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/__init__.py)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/test/__init__.py (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,20 @@
+# -*- test-case-name: txdav.caldav.datastore.test -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Calendar store tests.
+"""
Deleted: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/1.ics
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/1.ics 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/1.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,44 +0,0 @@
-BEGIN:VCALENDAR
-VERSION:2.0
-PRODID:-//Apple Inc.//iCal 4.0.1//EN
-CALSCALE:GREGORIAN
-BEGIN:VTIMEZONE
-TZID:US/Pacific
-BEGIN:DAYLIGHT
-TZOFFSETFROM:-0800
-RRULE:FREQ=YEARLY;BYMONTH=3;BYDAY=2SU
-DTSTART:20070311T020000
-TZNAME:PDT
-TZOFFSETTO:-0700
-END:DAYLIGHT
-BEGIN:STANDARD
-TZOFFSETFROM:-0700
-RRULE:FREQ=YEARLY;BYMONTH=11;BYDAY=1SU
-DTSTART:20071104T020000
-TZNAME:PST
-TZOFFSETTO:-0800
-END:STANDARD
-END:VTIMEZONE
-BEGIN:VEVENT
-ATTENDEE;CN="Wilfredo Sanchez";CUTYPE=INDIVIDUAL;PARTSTAT=ACCEPTED:mailt
- o:wsanchez at apple.com
-ATTENDEE;CN="Cyrus Daboo";CUTYPE=INDIVIDUAL;PARTSTAT=ACCEPTED:mailto:cda
- boo at apple.com
-DTEND;TZID=US/Pacific:20090324T124500
-TRANSP:OPAQUE
-ORGANIZER;CN="Wilfredo Sanchez":mailto:wsanchez at apple.com
-UID:uid1
-DTSTAMP:20090326T145447Z
-LOCATION:Wilfredo's Office
-SEQUENCE:2
-X-APPLE-EWS-BUSYSTATUS:BUSY
-SUMMARY:CalDAV protocol updates
-DTSTART;TZID=US/Pacific:20090324T121500
-CREATED:20090326T145440Z
-BEGIN:VALARM
-X-WR-ALARMUID:DB39AB67-449C-441C-89D2-D740B5F41A73
-TRIGGER;VALUE=DATE-TIME:20090324T180009Z
-ACTION:AUDIO
-END:VALARM
-END:VEVENT
-END:VCALENDAR
Copied: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/1.ics (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/1.ics)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/1.ics (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/1.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,44 @@
+BEGIN:VCALENDAR
+VERSION:2.0
+PRODID:-//Apple Inc.//iCal 4.0.1//EN
+CALSCALE:GREGORIAN
+BEGIN:VTIMEZONE
+TZID:US/Pacific
+BEGIN:DAYLIGHT
+TZOFFSETFROM:-0800
+RRULE:FREQ=YEARLY;BYMONTH=3;BYDAY=2SU
+DTSTART:20070311T020000
+TZNAME:PDT
+TZOFFSETTO:-0700
+END:DAYLIGHT
+BEGIN:STANDARD
+TZOFFSETFROM:-0700
+RRULE:FREQ=YEARLY;BYMONTH=11;BYDAY=1SU
+DTSTART:20071104T020000
+TZNAME:PST
+TZOFFSETTO:-0800
+END:STANDARD
+END:VTIMEZONE
+BEGIN:VEVENT
+ATTENDEE;CN="Wilfredo Sanchez";CUTYPE=INDIVIDUAL;PARTSTAT=ACCEPTED:mailt
+ o:wsanchez at apple.com
+ATTENDEE;CN="Cyrus Daboo";CUTYPE=INDIVIDUAL;PARTSTAT=ACCEPTED:mailto:cda
+ boo at apple.com
+DTEND;TZID=US/Pacific:20090324T124500
+TRANSP:OPAQUE
+ORGANIZER;CN="Wilfredo Sanchez":mailto:wsanchez at apple.com
+UID:uid1
+DTSTAMP:20090326T145447Z
+LOCATION:Wilfredo's Office
+SEQUENCE:2
+X-APPLE-EWS-BUSYSTATUS:BUSY
+SUMMARY:CalDAV protocol updates
+DTSTART;TZID=US/Pacific:20090324T121500
+CREATED:20090326T145440Z
+BEGIN:VALARM
+X-WR-ALARMUID:DB39AB67-449C-441C-89D2-D740B5F41A73
+TRIGGER;VALUE=DATE-TIME:20090324T180009Z
+ACTION:AUDIO
+END:VALARM
+END:VEVENT
+END:VCALENDAR
Deleted: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/2.ics
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/2.ics 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/2.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,48 +0,0 @@
-BEGIN:VCALENDAR
-VERSION:2.0
-CALSCALE:GREGORIAN
-PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
-BEGIN:VTIMEZONE
-TZID:US/Eastern
-LAST-MODIFIED:20040110T032845Z
-BEGIN:STANDARD
-DTSTART:20001026T020000
-RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
-TZNAME:EST
-TZOFFSETFROM:-0400
-TZOFFSETTO:-0500
-END:STANDARD
-BEGIN:DAYLIGHT
-DTSTART:20000404T020000
-RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
-TZNAME:EDT
-TZOFFSETFROM:-0500
-TZOFFSETTO:-0400
-END:DAYLIGHT
-END:VTIMEZONE
-BEGIN:VEVENT
-UID:uid2
-DTSTART;TZID=US/Eastern:20060102T140000
-DURATION:PT1H
-CREATED:20060102T190000Z
-DTSTAMP:20051222T210507Z
-RRULE:FREQ=DAILY;COUNT=5
-SUMMARY:event 6-%ctr
-END:VEVENT
-BEGIN:VEVENT
-UID:uid2
-RECURRENCE-ID;TZID=US/Eastern:20060104T140000
-DTSTART;TZID=US/Eastern:20060104T160000
-DURATION:PT1H
-CREATED:20060102T190000Z
-DESCRIPTION:Some notes
-DTSTAMP:20051222T210507Z
-SUMMARY:event 6-%ctr changed
-BEGIN:VALARM
-ACTION:AUDIO
-TRIGGER;RELATED=START:-PT10M
-X-MULBERRY-ALARM-STATUS:PENDING
-X-MULBERRY-SPEAK-TEXT:
-END:VALARM
-END:VEVENT
-END:VCALENDAR
Copied: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/2.ics (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/2.ics)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/2.ics (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/2.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,48 @@
+BEGIN:VCALENDAR
+VERSION:2.0
+CALSCALE:GREGORIAN
+PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
+BEGIN:VTIMEZONE
+TZID:US/Eastern
+LAST-MODIFIED:20040110T032845Z
+BEGIN:STANDARD
+DTSTART:20001026T020000
+RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
+TZNAME:EST
+TZOFFSETFROM:-0400
+TZOFFSETTO:-0500
+END:STANDARD
+BEGIN:DAYLIGHT
+DTSTART:20000404T020000
+RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
+TZNAME:EDT
+TZOFFSETFROM:-0500
+TZOFFSETTO:-0400
+END:DAYLIGHT
+END:VTIMEZONE
+BEGIN:VEVENT
+UID:uid2
+DTSTART;TZID=US/Eastern:20060102T140000
+DURATION:PT1H
+CREATED:20060102T190000Z
+DTSTAMP:20051222T210507Z
+RRULE:FREQ=DAILY;COUNT=5
+SUMMARY:event 6-%ctr
+END:VEVENT
+BEGIN:VEVENT
+UID:uid2
+RECURRENCE-ID;TZID=US/Eastern:20060104T140000
+DTSTART;TZID=US/Eastern:20060104T160000
+DURATION:PT1H
+CREATED:20060102T190000Z
+DESCRIPTION:Some notes
+DTSTAMP:20051222T210507Z
+SUMMARY:event 6-%ctr changed
+BEGIN:VALARM
+ACTION:AUDIO
+TRIGGER;RELATED=START:-PT10M
+X-MULBERRY-ALARM-STATUS:PENDING
+X-MULBERRY-SPEAK-TEXT:
+END:VALARM
+END:VEVENT
+END:VCALENDAR
Deleted: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/3.ics
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/3.ics 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/3.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,33 +0,0 @@
-BEGIN:VCALENDAR
-VERSION:2.0
-CALSCALE:GREGORIAN
-PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
-BEGIN:VTIMEZONE
-TZID:US/Pacific
-LAST-MODIFIED:20040110T032845Z
-BEGIN:STANDARD
-DTSTART:20001026T020000
-RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
-TZNAME:PST
-TZOFFSETFROM:-0700
-TZOFFSETTO:-0800
-END:STANDARD
-BEGIN:DAYLIGHT
-DTSTART:20000404T020000
-RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
-TZNAME:PDT
-TZOFFSETFROM:-0800
-TZOFFSETTO:-0700
-END:DAYLIGHT
-END:VTIMEZONE
-BEGIN:VEVENT
-UID:uid3
-DTSTART;TZID=US/Pacific:20060101T130000
-DURATION:PT1H
-CREATED:20060101T210000Z
-DTSTAMP:20051222T210146Z
-LAST-MODIFIED:20051222T210203Z
-SEQUENCE:1
-SUMMARY:event 3-%ctr
-END:VEVENT
-END:VCALENDAR
Copied: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/3.ics (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/3.ics)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/3.ics (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_1/3.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,33 @@
+BEGIN:VCALENDAR
+VERSION:2.0
+CALSCALE:GREGORIAN
+PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
+BEGIN:VTIMEZONE
+TZID:US/Pacific
+LAST-MODIFIED:20040110T032845Z
+BEGIN:STANDARD
+DTSTART:20001026T020000
+RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
+TZNAME:PST
+TZOFFSETFROM:-0700
+TZOFFSETTO:-0800
+END:STANDARD
+BEGIN:DAYLIGHT
+DTSTART:20000404T020000
+RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
+TZNAME:PDT
+TZOFFSETFROM:-0800
+TZOFFSETTO:-0700
+END:DAYLIGHT
+END:VTIMEZONE
+BEGIN:VEVENT
+UID:uid3
+DTSTART;TZID=US/Pacific:20060101T130000
+DURATION:PT1H
+CREATED:20060101T210000Z
+DTSTAMP:20051222T210146Z
+LAST-MODIFIED:20051222T210203Z
+SEQUENCE:1
+SUMMARY:event 3-%ctr
+END:VEVENT
+END:VCALENDAR
Deleted: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/24204e8682b99527cbda64d7423acda7.ics
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/24204e8682b99527cbda64d7423acda7.ics 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/24204e8682b99527cbda64d7423acda7.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,32 +0,0 @@
-BEGIN:VCALENDAR
-VERSION:2.0
-CALSCALE:GREGORIAN
-PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
-BEGIN:VTIMEZONE
-TZID:US/Mountain
-LAST-MODIFIED:20040110T032845Z
-BEGIN:STANDARD
-DTSTART:20001026T020000
-RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
-TZNAME:MST
-TZOFFSETFROM:-0600
-TZOFFSETTO:-0700
-END:STANDARD
-BEGIN:DAYLIGHT
-DTSTART:20000404T020000
-RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
-TZNAME:MDT
-TZOFFSETFROM:-0700
-TZOFFSETTO:-0600
-END:DAYLIGHT
-END:VTIMEZONE
-BEGIN:VEVENT
-UID:9A6519F71822CD45840C3440-%ctr at ninevah.local
-DTSTART;TZID=US/Mountain:20060101T110000
-DURATION:PT1H
-CREATED:20060101T160000Z
-DESCRIPTION:Some notes
-DTSTAMP:20051222T210052Z
-SUMMARY:event 2-%ctr
-END:VEVENT
-END:VCALENDAR
Copied: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/24204e8682b99527cbda64d7423acda7.ics (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/24204e8682b99527cbda64d7423acda7.ics)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/24204e8682b99527cbda64d7423acda7.ics (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/24204e8682b99527cbda64d7423acda7.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,32 @@
+BEGIN:VCALENDAR
+VERSION:2.0
+CALSCALE:GREGORIAN
+PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
+BEGIN:VTIMEZONE
+TZID:US/Mountain
+LAST-MODIFIED:20040110T032845Z
+BEGIN:STANDARD
+DTSTART:20001026T020000
+RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
+TZNAME:MST
+TZOFFSETFROM:-0600
+TZOFFSETTO:-0700
+END:STANDARD
+BEGIN:DAYLIGHT
+DTSTART:20000404T020000
+RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
+TZNAME:MDT
+TZOFFSETFROM:-0700
+TZOFFSETTO:-0600
+END:DAYLIGHT
+END:VTIMEZONE
+BEGIN:VEVENT
+UID:9A6519F71822CD45840C3440-%ctr at ninevah.local
+DTSTART;TZID=US/Mountain:20060101T110000
+DURATION:PT1H
+CREATED:20060101T160000Z
+DESCRIPTION:Some notes
+DTSTAMP:20051222T210052Z
+SUMMARY:event 2-%ctr
+END:VEVENT
+END:VCALENDAR
Deleted: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/61038c41bd02ae5daf9f7fe9d54199fd.ics
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/61038c41bd02ae5daf9f7fe9d54199fd.ics 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/61038c41bd02ae5daf9f7fe9d54199fd.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,31 +0,0 @@
-BEGIN:VCALENDAR
-VERSION:2.0
-CALSCALE:GREGORIAN
-PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
-BEGIN:VTIMEZONE
-TZID:US/Eastern
-LAST-MODIFIED:20040110T032845Z
-BEGIN:STANDARD
-DTSTART:20001026T020000
-RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
-TZNAME:EST
-TZOFFSETFROM:-0400
-TZOFFSETTO:-0500
-END:STANDARD
-BEGIN:DAYLIGHT
-DTSTART:20000404T020000
-RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
-TZNAME:EDT
-TZOFFSETFROM:-0500
-TZOFFSETTO:-0400
-END:DAYLIGHT
-END:VTIMEZONE
-BEGIN:VEVENT
-UID:54E181BC7CCC373042B28842-%ctr at ninevah.local
-DTSTART;TZID=US/Eastern:20060101T100000
-DURATION:PT1H
-CREATED:20060101T150000Z
-DTSTAMP:20051222T205953Z
-SUMMARY:event 1-%ctr
-END:VEVENT
-END:VCALENDAR
Copied: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/61038c41bd02ae5daf9f7fe9d54199fd.ics (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/61038c41bd02ae5daf9f7fe9d54199fd.ics)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/61038c41bd02ae5daf9f7fe9d54199fd.ics (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/61038c41bd02ae5daf9f7fe9d54199fd.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,31 @@
+BEGIN:VCALENDAR
+VERSION:2.0
+CALSCALE:GREGORIAN
+PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
+BEGIN:VTIMEZONE
+TZID:US/Eastern
+LAST-MODIFIED:20040110T032845Z
+BEGIN:STANDARD
+DTSTART:20001026T020000
+RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
+TZNAME:EST
+TZOFFSETFROM:-0400
+TZOFFSETTO:-0500
+END:STANDARD
+BEGIN:DAYLIGHT
+DTSTART:20000404T020000
+RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
+TZNAME:EDT
+TZOFFSETFROM:-0500
+TZOFFSETTO:-0400
+END:DAYLIGHT
+END:VTIMEZONE
+BEGIN:VEVENT
+UID:54E181BC7CCC373042B28842-%ctr at ninevah.local
+DTSTART;TZID=US/Eastern:20060101T100000
+DURATION:PT1H
+CREATED:20060101T150000Z
+DTSTAMP:20051222T205953Z
+SUMMARY:event 1-%ctr
+END:VEVENT
+END:VCALENDAR
Deleted: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/84be58ced1f1bb34057e1bd7e602c9c8.ics
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/84be58ced1f1bb34057e1bd7e602c9c8.ics 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/84be58ced1f1bb34057e1bd7e602c9c8.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,31 +0,0 @@
-BEGIN:VCALENDAR
-VERSION:2.0
-CALSCALE:GREGORIAN
-PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
-BEGIN:VTIMEZONE
-TZID:US/Eastern
-LAST-MODIFIED:20040110T032845Z
-BEGIN:STANDARD
-DTSTART:20001026T020000
-RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
-TZNAME:EST
-TZOFFSETFROM:-0400
-TZOFFSETTO:-0500
-END:STANDARD
-BEGIN:DAYLIGHT
-DTSTART:20000404T020000
-RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
-TZNAME:EDT
-TZOFFSETFROM:-0500
-TZOFFSETTO:-0400
-END:DAYLIGHT
-END:VTIMEZONE
-BEGIN:VEVENT
-UID:54E181BC7CCC373042B28842-8-%ctr at ninevah.local
-DTSTART;TZID=US/Eastern:20060107T100000
-DURATION:PT1H
-CREATED:20060101T150000Z
-DTSTAMP:20051222T205953Z
-SUMMARY:event 8-%ctr
-END:VEVENT
-END:VCALENDAR
Copied: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/84be58ced1f1bb34057e1bd7e602c9c8.ics (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/84be58ced1f1bb34057e1bd7e602c9c8.ics)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/84be58ced1f1bb34057e1bd7e602c9c8.ics (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/84be58ced1f1bb34057e1bd7e602c9c8.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,31 @@
+BEGIN:VCALENDAR
+VERSION:2.0
+CALSCALE:GREGORIAN
+PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
+BEGIN:VTIMEZONE
+TZID:US/Eastern
+LAST-MODIFIED:20040110T032845Z
+BEGIN:STANDARD
+DTSTART:20001026T020000
+RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
+TZNAME:EST
+TZOFFSETFROM:-0400
+TZOFFSETTO:-0500
+END:STANDARD
+BEGIN:DAYLIGHT
+DTSTART:20000404T020000
+RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
+TZNAME:EDT
+TZOFFSETFROM:-0500
+TZOFFSETTO:-0400
+END:DAYLIGHT
+END:VTIMEZONE
+BEGIN:VEVENT
+UID:54E181BC7CCC373042B28842-8-%ctr at ninevah.local
+DTSTART;TZID=US/Eastern:20060107T100000
+DURATION:PT1H
+CREATED:20060101T150000Z
+DTSTAMP:20051222T205953Z
+SUMMARY:event 8-%ctr
+END:VEVENT
+END:VCALENDAR
Deleted: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/acc1015b7dc300c1b5665f6833960994.ics
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/acc1015b7dc300c1b5665f6833960994.ics 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/acc1015b7dc300c1b5665f6833960994.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,31 +0,0 @@
-BEGIN:VCALENDAR
-VERSION:2.0
-CALSCALE:GREGORIAN
-PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
-BEGIN:VTIMEZONE
-TZID:US/Eastern
-LAST-MODIFIED:20040110T032845Z
-BEGIN:STANDARD
-DTSTART:20001026T020000
-RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
-TZNAME:EST
-TZOFFSETFROM:-0400
-TZOFFSETTO:-0500
-END:STANDARD
-BEGIN:DAYLIGHT
-DTSTART:20000404T020000
-RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
-TZNAME:EDT
-TZOFFSETFROM:-0500
-TZOFFSETTO:-0400
-END:DAYLIGHT
-END:VTIMEZONE
-BEGIN:VEVENT
-UID:54E181BC7CCC373042B28842-9-%ctr at ninevah.local
-DTSTART;TZID=US/Eastern:20060107T103000
-DURATION:PT1H
-CREATED:20060101T150000Z
-DTSTAMP:20051222T205953Z
-SUMMARY:event 9-%ctr
-END:VEVENT
-END:VCALENDAR
Copied: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/acc1015b7dc300c1b5665f6833960994.ics (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/acc1015b7dc300c1b5665f6833960994.ics)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/acc1015b7dc300c1b5665f6833960994.ics (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/acc1015b7dc300c1b5665f6833960994.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,31 @@
+BEGIN:VCALENDAR
+VERSION:2.0
+CALSCALE:GREGORIAN
+PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
+BEGIN:VTIMEZONE
+TZID:US/Eastern
+LAST-MODIFIED:20040110T032845Z
+BEGIN:STANDARD
+DTSTART:20001026T020000
+RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
+TZNAME:EST
+TZOFFSETFROM:-0400
+TZOFFSETTO:-0500
+END:STANDARD
+BEGIN:DAYLIGHT
+DTSTART:20000404T020000
+RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
+TZNAME:EDT
+TZOFFSETFROM:-0500
+TZOFFSETTO:-0400
+END:DAYLIGHT
+END:VTIMEZONE
+BEGIN:VEVENT
+UID:54E181BC7CCC373042B28842-9-%ctr at ninevah.local
+DTSTART;TZID=US/Eastern:20060107T103000
+DURATION:PT1H
+CREATED:20060101T150000Z
+DTSTAMP:20051222T205953Z
+SUMMARY:event 9-%ctr
+END:VEVENT
+END:VCALENDAR
Deleted: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b0d5785f275c064117ffd1fc20f4ed40.ics
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b0d5785f275c064117ffd1fc20f4ed40.ics 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b0d5785f275c064117ffd1fc20f4ed40.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,39 +0,0 @@
-BEGIN:VCALENDAR
-VERSION:2.0
-CALSCALE:GREGORIAN
-PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
-BEGIN:VTIMEZONE
-TZID:US/Eastern
-LAST-MODIFIED:20040110T032845Z
-BEGIN:STANDARD
-DTSTART:20001026T020000
-RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
-TZNAME:EST
-TZOFFSETFROM:-0400
-TZOFFSETTO:-0500
-END:STANDARD
-BEGIN:DAYLIGHT
-DTSTART:20000404T020000
-RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
-TZNAME:EDT
-TZOFFSETFROM:-0500
-TZOFFSETTO:-0400
-END:DAYLIGHT
-END:VTIMEZONE
-BEGIN:VEVENT
-UID:A3217B429B4D2FF2DC2EEE66-%ctr at ninevah.local
-DTSTART;TZID=US/Eastern:20060101T180000
-DURATION:PT1H
-CREATED:20060101T230000Z
-DTSTAMP:20051222T210310Z
-SUMMARY:event 4-%ctr
-BEGIN:VALARM
-ACTION:AUDIO
-DURATION:PT10M
-REPEAT:5
-TRIGGER;RELATED=START:-PT1H
-X-MULBERRY-ALARM-STATUS:PENDING
-X-MULBERRY-SPEAK-TEXT:
-END:VALARM
-END:VEVENT
-END:VCALENDAR
Copied: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b0d5785f275c064117ffd1fc20f4ed40.ics (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b0d5785f275c064117ffd1fc20f4ed40.ics)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b0d5785f275c064117ffd1fc20f4ed40.ics (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b0d5785f275c064117ffd1fc20f4ed40.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,39 @@
+BEGIN:VCALENDAR
+VERSION:2.0
+CALSCALE:GREGORIAN
+PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
+BEGIN:VTIMEZONE
+TZID:US/Eastern
+LAST-MODIFIED:20040110T032845Z
+BEGIN:STANDARD
+DTSTART:20001026T020000
+RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
+TZNAME:EST
+TZOFFSETFROM:-0400
+TZOFFSETTO:-0500
+END:STANDARD
+BEGIN:DAYLIGHT
+DTSTART:20000404T020000
+RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
+TZNAME:EDT
+TZOFFSETFROM:-0500
+TZOFFSETTO:-0400
+END:DAYLIGHT
+END:VTIMEZONE
+BEGIN:VEVENT
+UID:A3217B429B4D2FF2DC2EEE66-%ctr at ninevah.local
+DTSTART;TZID=US/Eastern:20060101T180000
+DURATION:PT1H
+CREATED:20060101T230000Z
+DTSTAMP:20051222T210310Z
+SUMMARY:event 4-%ctr
+BEGIN:VALARM
+ACTION:AUDIO
+DURATION:PT10M
+REPEAT:5
+TRIGGER;RELATED=START:-PT1H
+X-MULBERRY-ALARM-STATUS:PENDING
+X-MULBERRY-SPEAK-TEXT:
+END:VALARM
+END:VEVENT
+END:VCALENDAR
Deleted: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b495c5dd5aa53392078eb43b1f906a80.ics
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b495c5dd5aa53392078eb43b1f906a80.ics 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b495c5dd5aa53392078eb43b1f906a80.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,38 +0,0 @@
-BEGIN:VCALENDAR
-VERSION:2.0
-CALSCALE:GREGORIAN
-PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
-BEGIN:VTIMEZONE
-TZID:US/Eastern
-LAST-MODIFIED:20040110T032845Z
-BEGIN:STANDARD
-DTSTART:20001026T020000
-RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
-TZNAME:EST
-TZOFFSETFROM:-0400
-TZOFFSETTO:-0500
-END:STANDARD
-BEGIN:DAYLIGHT
-DTSTART:20000404T020000
-RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
-TZNAME:EDT
-TZOFFSETFROM:-0500
-TZOFFSETTO:-0400
-END:DAYLIGHT
-END:VTIMEZONE
-BEGIN:VEVENT
-UID:945113826375CBB89184DC36-%ctr at ninevah.local
-DTSTART;TZID=US/Eastern:20060102T100000
-DURATION:PT1H
-CREATED:20060102T150000Z
-DTSTAMP:20051222T210412Z
-RRULE:FREQ=DAILY;COUNT=5
-SUMMARY:event 5-%ctr
-BEGIN:VALARM
-ACTION:AUDIO
-TRIGGER;RELATED=START:-PT10M
-X-MULBERRY-ALARM-STATUS:PENDING
-X-MULBERRY-SPEAK-TEXT:
-END:VALARM
-END:VEVENT
-END:VCALENDAR
Copied: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b495c5dd5aa53392078eb43b1f906a80.ics (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b495c5dd5aa53392078eb43b1f906a80.ics)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b495c5dd5aa53392078eb43b1f906a80.ics (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b495c5dd5aa53392078eb43b1f906a80.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,38 @@
+BEGIN:VCALENDAR
+VERSION:2.0
+CALSCALE:GREGORIAN
+PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
+BEGIN:VTIMEZONE
+TZID:US/Eastern
+LAST-MODIFIED:20040110T032845Z
+BEGIN:STANDARD
+DTSTART:20001026T020000
+RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
+TZNAME:EST
+TZOFFSETFROM:-0400
+TZOFFSETTO:-0500
+END:STANDARD
+BEGIN:DAYLIGHT
+DTSTART:20000404T020000
+RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
+TZNAME:EDT
+TZOFFSETFROM:-0500
+TZOFFSETTO:-0400
+END:DAYLIGHT
+END:VTIMEZONE
+BEGIN:VEVENT
+UID:945113826375CBB89184DC36-%ctr at ninevah.local
+DTSTART;TZID=US/Eastern:20060102T100000
+DURATION:PT1H
+CREATED:20060102T150000Z
+DTSTAMP:20051222T210412Z
+RRULE:FREQ=DAILY;COUNT=5
+SUMMARY:event 5-%ctr
+BEGIN:VALARM
+ACTION:AUDIO
+TRIGGER;RELATED=START:-PT10M
+X-MULBERRY-ALARM-STATUS:PENDING
+X-MULBERRY-SPEAK-TEXT:
+END:VALARM
+END:VEVENT
+END:VCALENDAR
Deleted: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b88dd50941e4a31520ee396fd7894c96.ics
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b88dd50941e4a31520ee396fd7894c96.ics 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b88dd50941e4a31520ee396fd7894c96.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,31 +0,0 @@
-BEGIN:VCALENDAR
-VERSION:2.0
-CALSCALE:GREGORIAN
-PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
-BEGIN:VTIMEZONE
-TZID:US/Eastern
-LAST-MODIFIED:20040110T032845Z
-BEGIN:STANDARD
-DTSTART:20001026T020000
-RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
-TZNAME:EST
-TZOFFSETFROM:-0400
-TZOFFSETTO:-0500
-END:STANDARD
-BEGIN:DAYLIGHT
-DTSTART:20000404T020000
-RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
-TZNAME:EDT
-TZOFFSETFROM:-0500
-TZOFFSETTO:-0400
-END:DAYLIGHT
-END:VTIMEZONE
-BEGIN:VEVENT
-UID:54E181BC7CCC373042B28842-10-%ctr at ninevah.local
-DTSTART;TZID=US/Eastern:20060108T100000
-DURATION:PT1H
-CREATED:20060101T150000Z
-DTSTAMP:20051222T205953Z
-SUMMARY:event 10-%ctr
-END:VEVENT
-END:VCALENDAR
Copied: CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b88dd50941e4a31520ee396fd7894c96.ics (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b88dd50941e4a31520ee396fd7894c96.ics)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b88dd50941e4a31520ee396fd7894c96.ics (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/calendar_store/ho/me/home1/calendar_2/b88dd50941e4a31520ee396fd7894c96.ics 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,31 @@
+BEGIN:VCALENDAR
+VERSION:2.0
+CALSCALE:GREGORIAN
+PRODID:-//Cyrusoft International\, Inc.//Mulberry v4.0//EN
+BEGIN:VTIMEZONE
+TZID:US/Eastern
+LAST-MODIFIED:20040110T032845Z
+BEGIN:STANDARD
+DTSTART:20001026T020000
+RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
+TZNAME:EST
+TZOFFSETFROM:-0400
+TZOFFSETTO:-0500
+END:STANDARD
+BEGIN:DAYLIGHT
+DTSTART:20000404T020000
+RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
+TZNAME:EDT
+TZOFFSETFROM:-0500
+TZOFFSETTO:-0400
+END:DAYLIGHT
+END:VTIMEZONE
+BEGIN:VEVENT
+UID:54E181BC7CCC373042B28842-10-%ctr at ninevah.local
+DTSTART;TZID=US/Eastern:20060108T100000
+DURATION:PT1H
+CREATED:20060101T150000Z
+DTSTAMP:20051222T205953Z
+SUMMARY:event 10-%ctr
+END:VEVENT
+END:VCALENDAR
Deleted: CalendarServer/trunk/txdav/caldav/datastore/test/common.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/common.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/common.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,1202 +0,0 @@
-# -*- test-case-name: txdav.caldav.datastore -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Tests for common calendar store API functions.
-"""
-
-from zope.interface.verify import verifyObject
-from zope.interface.exceptions import (
- BrokenMethodImplementation, DoesNotImplement)
-
-from twisted.internet.defer import Deferred, inlineCallbacks
-from twisted.internet.protocol import Protocol
-
-from txdav.idav import IPropertyStore, IDataStore, AlreadyFinishedError
-from txdav.base.propertystore.base import PropertyName
-
-from txdav.common.icommondatastore import HomeChildNameAlreadyExistsError, \
- ICommonTransaction
-from txdav.common.icommondatastore import InvalidObjectResourceError
-from txdav.common.icommondatastore import NoSuchHomeChildError
-from txdav.common.icommondatastore import NoSuchObjectResourceError
-from txdav.common.icommondatastore import ObjectResourceNameAlreadyExistsError
-from txdav.common.inotifications import INotificationObject
-
-from txdav.caldav.icalendarstore import (
- ICalendarObject, ICalendarHome,
- ICalendar, IAttachment, ICalendarTransaction)
-
-from twext.python.filepath import CachingFilePath as FilePath
-from twext.web2.dav import davxml
-from twext.web2.http_headers import MimeType
-from twext.web2.dav.element.base import WebDAVUnknownElement
-from twext.python.vcomponent import VComponent
-
-from twistedcaldav.notify import Notifier
-from twistedcaldav.customxml import InviteNotification, InviteSummary
-
-storePath = FilePath(__file__).parent().child("calendar_store")
-
-homeRoot = storePath.child("ho").child("me").child("home1")
-
-cal1Root = homeRoot.child("calendar_1")
-
-calendar1_objectNames = [
- "1.ics",
- "2.ics",
- "3.ics",
-]
-
-
-home1_calendarNames = [
- "calendar_1",
- "calendar_2",
- "calendar_empty",
-]
-
-
-event4_text = (
- "BEGIN:VCALENDAR\r\n"
- "VERSION:2.0\r\n"
- "PRODID:-//Apple Inc.//iCal 4.0.1//EN\r\n"
- "CALSCALE:GREGORIAN\r\n"
- "BEGIN:VTIMEZONE\r\n"
- "TZID:US/Pacific\r\n"
- "BEGIN:DAYLIGHT\r\n"
- "TZOFFSETFROM:-0800\r\n"
- "RRULE:FREQ=YEARLY;BYMONTH=3;BYDAY=2SU\r\n"
- "DTSTART:20070311T020000\r\n"
- "TZNAME:PDT\r\n"
- "TZOFFSETTO:-0700\r\n"
- "END:DAYLIGHT\r\n"
- "BEGIN:STANDARD\r\n"
- "TZOFFSETFROM:-0700\r\n"
- "RRULE:FREQ=YEARLY;BYMONTH=11;BYDAY=1SU\r\n"
- "DTSTART:20071104T020000\r\n"
- "TZNAME:PST\r\n"
- "TZOFFSETTO:-0800\r\n"
- "END:STANDARD\r\n"
- "END:VTIMEZONE\r\n"
- "BEGIN:VEVENT\r\n"
- "CREATED:20100203T013849Z\r\n"
- "UID:uid4\r\n"
- "DTEND;TZID=US/Pacific:20100207T173000\r\n"
- "TRANSP:OPAQUE\r\n"
- "SUMMARY:New Event\r\n"
- "DTSTART;TZID=US/Pacific:20100207T170000\r\n"
- "DTSTAMP:20100203T013909Z\r\n"
- "SEQUENCE:3\r\n"
- "BEGIN:VALARM\r\n"
- "X-WR-ALARMUID:1377CCC7-F85C-4610-8583-9513D4B364E1\r\n"
- "TRIGGER:-PT20M\r\n"
- "ATTACH;VALUE=URI:Basso\r\n"
- "ACTION:AUDIO\r\n"
- "END:VALARM\r\n"
- "END:VEVENT\r\n"
- "END:VCALENDAR\r\n"
-)
-
-
-
-event4notCalDAV_text = (
- "BEGIN:VCALENDAR\r\n"
- "VERSION:2.0\r\n"
- "PRODID:-//Apple Inc.//iCal 4.0.1//EN\r\n"
- "CALSCALE:GREGORIAN\r\n"
- "BEGIN:VEVENT\r\n"
- "CREATED:20100203T013849Z\r\n"
- "UID:4\r\n"
- "DTEND;TZID=US/Pacific:20100207T173000\r\n" # TZID without VTIMEZONE
- "TRANSP:OPAQUE\r\n"
- "SUMMARY:New Event\r\n"
- "DTSTART;TZID=US/Pacific:20100207T170000\r\n"
- "DTSTAMP:20100203T013909Z\r\n"
- "SEQUENCE:3\r\n"
- "BEGIN:VALARM\r\n"
- "X-WR-ALARMUID:1377CCC7-F85C-4610-8583-9513D4B364E1\r\n"
- "TRIGGER:-PT20M\r\n"
- "ATTACH;VALUE=URI:Basso\r\n"
- "ACTION:AUDIO\r\n"
- "END:VALARM\r\n"
- "END:VEVENT\r\n"
- "END:VCALENDAR\r\n"
-)
-
-
-
-event1modified_text = event4_text.replace(
- "\r\nUID:uid4\r\n",
- "\r\nUID:uid1\r\n"
-)
-
-
-
-def assertProvides(testCase, interface, provider):
- """
- Verify that C{provider} properly provides C{interface}
-
- @type interface: L{zope.interface.Interface}
- @type provider: C{provider}
- """
- try:
- verifyObject(interface, provider)
- except BrokenMethodImplementation, e:
- testCase.fail(e)
- except DoesNotImplement, e:
- testCase.fail("%r does not provide %s.%s" %
- (provider, interface.__module__, interface.getName()))
-
-
-class CommonTests(object):
- """
- Tests for common functionality of interfaces defined in
- L{txdav.caldav.icalendarstore}.
- """
-
- requirements = {
- "home1": {
- "calendar_1": {
- "1.ics": cal1Root.child("1.ics").getContent(),
- "2.ics": cal1Root.child("2.ics").getContent(),
- "3.ics": cal1Root.child("3.ics").getContent()
- },
- "calendar_2": {},
- "calendar_empty": {},
- "not_a_calendar": None
- },
- "not_a_home": None
- }
-
- def storeUnderTest(self):
- """
- Subclasses must override this to return an L{ICommonDataStore} provider
- which adheres to the structure detailed by L{CommonTests.requirements}.
- This attribute is a dict of dict of dicts; the outermost layer
- representing UIDs mapping to calendar homes, then calendar names mapping
- to calendar collections, and finally calendar object names mapping to
- calendar object text.
- """
- raise NotImplementedError()
-
-
- lastTransaction = None
- savedStore = None
-
- def transactionUnderTest(self):
- """
- Create a transaction from C{storeUnderTest} and save it as
- C[lastTransaction}. Also makes sure to use the same store, saving the
- value from C{storeUnderTest}.
- """
- if self.lastTransaction is not None:
- return self.lastTransaction
- if self.savedStore is None:
- self.savedStore = self.storeUnderTest()
- txn = self.lastTransaction = self.savedStore.newTransaction(self.id())
- return txn
-
-
- def commit(self):
- """
- Commit the last transaction created from C{transactionUnderTest}, and
- clear it.
- """
- self.lastTransaction.commit()
- self.lastTransaction = None
-
-
- def abort(self):
- """
- Abort the last transaction created from C[transactionUnderTest}, and
- clear it.
- """
- self.lastTransaction.abort()
- self.lastTransaction = None
-
-
- def setUp(self):
- self.notifierFactory = StubNotifierFactory()
-
-
- def tearDown(self):
- if self.lastTransaction is not None:
- self.commit()
-
-
- def homeUnderTest(self):
- """
- Get the calendar home detailed by C{requirements['home1']}.
- """
- return self.transactionUnderTest().calendarHomeWithUID("home1")
-
-
- def calendarUnderTest(self):
- """
- Get the calendar detailed by C{requirements['home1']['calendar_1']}.
- """
- return self.homeUnderTest().calendarWithName("calendar_1")
-
-
- def calendarObjectUnderTest(self):
- """
- Get the calendar detailed by
- C{requirements['home1']['calendar_1']['1.ics']}.
- """
- return self.calendarUnderTest().calendarObjectWithName("1.ics")
-
-
- assertProvides = assertProvides
-
- def test_calendarStoreProvides(self):
- """
- The calendar store provides L{IDataStore} and its required attributes.
- """
- calendarStore = self.storeUnderTest()
- self.assertProvides(IDataStore, calendarStore)
-
-
- def test_transactionProvides(self):
- """
- The transactions generated by the calendar store provide
- L{ICommonStoreTransaction}, L{ICalendarTransaction}, and their
- respectively required attributes.
- """
- txn = self.transactionUnderTest()
- self.assertProvides(ICommonTransaction, txn)
- self.assertProvides(ICalendarTransaction, txn)
-
-
- def test_homeProvides(self):
- """
- The calendar homes generated by the calendar store provide
- L{ICalendarHome} and its required attributes.
- """
- self.assertProvides(ICalendarHome, self.homeUnderTest())
-
-
- def test_calendarProvides(self):
- """
- The calendars generated by the calendar store provide L{ICalendar} and
- its required attributes.
- """
- self.assertProvides(ICalendar, self.calendarUnderTest())
-
-
- def test_calendarObjectProvides(self):
- """
- The calendar objects generated by the calendar store provide
- L{ICalendarObject} and its required attributes.
- """
- self.assertProvides(ICalendarObject, self.calendarObjectUnderTest())
-
-
- def notificationUnderTest(self):
- txn = self.transactionUnderTest()
- notifications = txn.notificationsWithUID("home1")
- inviteNotification = InviteNotification()
- notifications.writeNotificationObject("abc", inviteNotification,
- inviteNotification.toxml())
- notificationObject = notifications.notificationObjectWithUID("abc")
- return notificationObject
-
-
- def test_notificationObjectProvides(self):
- """
- The objects retrieved from the notification home (the object returned
- from L{notificationsWithUID}) provide L{INotificationObject}.
- """
- notificationObject = self.notificationUnderTest()
- self.assertProvides(INotificationObject, notificationObject)
-
-
- def test_replaceNotification(self):
- """
- L{INotificationCollection.writeNotificationObject} will silently
- overwrite the notification object.
- """
- notifications = self.transactionUnderTest().notificationsWithUID(
- "home1"
- )
- inviteNotification = InviteNotification()
- notifications.writeNotificationObject("abc", inviteNotification,
- inviteNotification.toxml())
- inviteNotification2 = InviteNotification(InviteSummary("a summary"))
- notifications.writeNotificationObject(
- "abc", inviteNotification, inviteNotification2.toxml())
- abc = notifications.notificationObjectWithUID("abc")
- self.assertEquals(abc.xmldata(), inviteNotification2.toxml())
-
-
- def test_notificationObjectModified(self):
- """
- The objects retrieved from the notification home have a C{modified}
- method which returns the timestamp of their last modification.
- """
- notification = self.notificationUnderTest()
- self.assertIsInstance(notification.modified(), int)
-
-
- def test_notificationObjectParent(self):
- """
- L{INotificationObject.notificationCollection} returns the
- L{INotificationCollection} that the object was retrieved from.
- """
- txn = self.transactionUnderTest()
- collection = txn.notificationsWithUID("home1")
- notification = self.notificationUnderTest()
- self.assertIdentical(collection, notification.notificationCollection())
-
-
- def test_notifierID(self):
- home = self.homeUnderTest()
- self.assertEquals(home.notifierID(), "home1")
- calendar = home.calendarWithName("calendar_1")
- self.assertEquals(calendar.notifierID(), "home1")
- self.assertEquals(calendar.notifierID(label="collection"), "home1/calendar_1")
-
-
- def test_calendarHomeWithUID_exists(self):
- """
- Finding an existing calendar home by UID results in an object that
- provides L{ICalendarHome} and has a C{uid()} method that returns the
- same value that was passed in.
- """
- calendarHome = (self.transactionUnderTest()
- .calendarHomeWithUID("home1"))
- self.assertEquals(calendarHome.uid(), "home1")
- self.assertProvides(ICalendarHome, calendarHome)
-
-
- def test_calendarHomeWithUID_absent(self):
- """
- L{ICommonStoreTransaction.calendarHomeWithUID} should return C{None}
- when asked for a non-existent calendar home.
- """
- txn = self.transactionUnderTest()
- self.assertEquals(txn.calendarHomeWithUID("xyzzy"), None)
-
-
- def test_calendarWithName_exists(self):
- """
- L{ICalendarHome.calendarWithName} returns an L{ICalendar} provider,
- whose name matches the one passed in.
- """
- home = self.homeUnderTest()
- for name in home1_calendarNames:
- calendar = home.calendarWithName(name)
- if calendar is None:
- self.fail("calendar %r didn't exist" % (name,))
- self.assertProvides(ICalendar, calendar)
- self.assertEquals(calendar.name(), name)
-
-
- def test_calendarRename(self):
- """
- L{ICalendar.rename} changes the name of the L{ICalendar}.
- """
- home = self.homeUnderTest()
- calendar = home.calendarWithName("calendar_1")
- calendar.rename("some_other_name")
- def positiveAssertions():
- self.assertEquals(calendar.name(), "some_other_name")
- self.assertEquals(calendar, home.calendarWithName("some_other_name"))
- self.assertEquals(None, home.calendarWithName("calendar_1"))
- positiveAssertions()
- self.commit()
- home = self.homeUnderTest()
- calendar = home.calendarWithName("some_other_name")
- positiveAssertions()
- # FIXME: revert
- # FIXME: test for multiple renames
- # FIXME: test for conflicting renames (a->b, c->a in the same txn)
-
-
- def test_calendarWithName_absent(self):
- """
- L{ICalendarHome.calendarWithName} returns C{None} for calendars which
- do not exist.
- """
- self.assertEquals(self.homeUnderTest().calendarWithName("xyzzy"),
- None)
-
-
- def test_createCalendarWithName_absent(self):
- """
- L{ICalendarHome.createCalendarWithName} creates a new L{ICalendar} that
- can be retrieved with L{ICalendarHome.calendarWithName}.
- """
- home = self.homeUnderTest()
- name = "new"
- self.assertIdentical(home.calendarWithName(name), None)
- home.createCalendarWithName(name)
- self.assertNotIdentical(home.calendarWithName(name), None)
- def checkProperties():
- calendarProperties = home.calendarWithName(name).properties()
- self.assertEquals(
- calendarProperties[
- PropertyName.fromString(davxml.ResourceType.sname())
- ],
- davxml.ResourceType.calendar #@UndefinedVariable
- )
- checkProperties()
-
- self.commit()
-
- # Make sure notification fired after commit
- self.assertEquals(self.notifierFactory.history, [("update", "home1")])
-
- # Make sure it's available in a new transaction; i.e. test the commit.
- home = self.homeUnderTest()
- self.assertNotIdentical(home.calendarWithName(name), None)
-
- # Sanity check: are the properties actually persisted? Check in
- # subsequent transaction.
- checkProperties()
-
- # FIXME: no independent testing of the property store's persistence
- # right now
-
-
- def test_createCalendarWithName_exists(self):
- """
- L{ICalendarHome.createCalendarWithName} raises
- L{CalendarAlreadyExistsError} when the name conflicts with an already-
- existing
- """
- for name in home1_calendarNames:
- self.assertRaises(
- HomeChildNameAlreadyExistsError,
- self.homeUnderTest().createCalendarWithName, name
- )
-
-
- def test_removeCalendarWithName_exists(self):
- """
- L{ICalendarHome.removeCalendarWithName} removes a calendar that already
- exists.
- """
- home = self.homeUnderTest()
-
- # FIXME: test transactions
- for name in home1_calendarNames:
- self.assertNotIdentical(home.calendarWithName(name), None)
- home.removeCalendarWithName(name)
- self.assertEquals(home.calendarWithName(name), None)
-
- self.commit()
-
- # Make sure notification fired after commit
- self.assertEquals(
- self.notifierFactory.history,
- [("update", "home1"), ("update", "home1"), ("update", "home1")]
- )
-
-
- def test_removeCalendarWithName_absent(self):
- """
- Attempt to remove an non-existing calendar should raise.
- """
- home = self.homeUnderTest()
- self.assertRaises(NoSuchHomeChildError,
- home.removeCalendarWithName, "xyzzy")
-
-
- def test_calendarObjects(self):
- """
- L{ICalendar.calendarObjects} will enumerate the calendar objects present
- in the filesystem, in name order, but skip those with hidden names.
- """
- calendar1 = self.calendarUnderTest()
- calendarObjects = list(calendar1.calendarObjects())
-
- for calendarObject in calendarObjects:
- self.assertProvides(ICalendarObject, calendarObject)
- self.assertEquals(
- calendar1.calendarObjectWithName(calendarObject.name()),
- calendarObject
- )
-
- self.assertEquals(
- set(list(o.name() for o in calendarObjects)),
- set(calendar1_objectNames)
- )
-
-
- def test_calendarObjectsWithRemovedObject(self):
- """
- L{ICalendar.calendarObjects} skips those objects which have been
- removed by L{Calendar.removeCalendarObjectWithName} in the same
- transaction, even if it has not yet been committed.
- """
- calendar1 = self.calendarUnderTest()
- calendar1.removeCalendarObjectWithName("2.ics")
- calendarObjects = list(calendar1.calendarObjects())
- self.assertEquals(set(o.name() for o in calendarObjects),
- set(calendar1_objectNames) - set(["2.ics"]))
-
-
- def test_ownerCalendarHome(self):
- """
- L{ICalendar.ownerCalendarHome} should match the home UID.
- """
- self.assertEquals(
- self.calendarUnderTest().ownerCalendarHome().uid(),
- self.homeUnderTest().uid()
- )
-
-
- def test_calendarObjectWithName_exists(self):
- """
- L{ICalendar.calendarObjectWithName} returns an L{ICalendarObject}
- provider for calendars which already exist.
- """
- calendar1 = self.calendarUnderTest()
- for name in calendar1_objectNames:
- calendarObject = calendar1.calendarObjectWithName(name)
- self.assertProvides(ICalendarObject, calendarObject)
- self.assertEquals(calendarObject.name(), name)
- # FIXME: add more tests based on CommonTests.requirements
-
-
- def test_calendarObjectWithName_absent(self):
- """
- L{ICalendar.calendarObjectWithName} returns C{None} for calendars which
- don't exist.
- """
- calendar1 = self.calendarUnderTest()
- self.assertEquals(calendar1.calendarObjectWithName("xyzzy"), None)
-
-
- def test_removeCalendarObjectWithUID_exists(self):
- """
- Remove an existing calendar object.
- """
- calendar = self.calendarUnderTest()
- for name in calendar1_objectNames:
- uid = (u'uid' + name.rstrip(".ics"))
- self.assertNotIdentical(calendar.calendarObjectWithUID(uid),
- None)
- calendar.removeCalendarObjectWithUID(uid)
- self.assertEquals(
- calendar.calendarObjectWithUID(uid),
- None
- )
- self.assertEquals(
- calendar.calendarObjectWithName(name),
- None
- )
-
- # Make sure notifications are fired after commit
- self.commit()
- self.assertEquals(
- self.notifierFactory.history,
- [
- ("update", "home1"),
- ("update", "home1/calendar_1"),
- ("update", "home1"),
- ("update", "home1/calendar_1"),
- ("update", "home1"),
- ("update", "home1/calendar_1"),
- ]
- )
-
- def test_removeCalendarObjectWithName_exists(self):
- """
- Remove an existing calendar object.
- """
- calendar = self.calendarUnderTest()
- for name in calendar1_objectNames:
- self.assertNotIdentical(
- calendar.calendarObjectWithName(name), None
- )
- calendar.removeCalendarObjectWithName(name)
- self.assertIdentical(
- calendar.calendarObjectWithName(name), None
- )
-
-
- def test_removeCalendarObjectWithName_absent(self):
- """
- Attempt to remove an non-existing calendar object should raise.
- """
- calendar = self.calendarUnderTest()
- self.assertRaises(
- NoSuchObjectResourceError,
- calendar.removeCalendarObjectWithName, "xyzzy"
- )
-
-
- def test_calendarName(self):
- """
- L{Calendar.name} reflects the name of the calendar.
- """
- self.assertEquals(self.calendarUnderTest().name(), "calendar_1")
-
-
- def test_calendarObjectName(self):
- """
- L{ICalendarObject.name} reflects the name of the calendar object.
- """
- self.assertEquals(self.calendarObjectUnderTest().name(), "1.ics")
-
-
- def test_component(self):
- """
- L{ICalendarObject.component} returns a L{VComponent} describing the
- calendar data underlying that calendar object.
- """
- component = self.calendarObjectUnderTest().component()
-
- self.failUnless(
- isinstance(component, VComponent),
- component
- )
-
- self.assertEquals(component.name(), "VCALENDAR")
- self.assertEquals(component.mainType(), "VEVENT")
- self.assertEquals(component.resourceUID(), "uid1")
-
-
- def test_iCalendarText(self):
- """
- L{ICalendarObject.iCalendarText} returns a C{str} describing the same
- data provided by L{ICalendarObject.component}.
- """
- text = self.calendarObjectUnderTest().iCalendarText()
- self.assertIsInstance(text, str)
- self.failUnless(text.startswith("BEGIN:VCALENDAR\r\n"))
- self.assertIn("\r\nUID:uid1\r\n", text)
- self.failUnless(text.endswith("\r\nEND:VCALENDAR\r\n"))
-
-
- def test_calendarObjectUID(self):
- """
- L{ICalendarObject.uid} returns a C{str} describing the C{UID} property
- of the calendar object's component.
- """
- self.assertEquals(self.calendarObjectUnderTest().uid(), "uid1")
-
-
- def test_organizer(self):
- """
- L{ICalendarObject.organizer} returns a C{str} describing the calendar
- user address of the C{ORGANIZER} property of the calendar object's
- component.
- """
- self.assertEquals(
- self.calendarObjectUnderTest().organizer(),
- "mailto:wsanchez at apple.com"
- )
-
-
- def test_calendarObjectWithUID_absent(self):
- """
- L{ICalendar.calendarObjectWithUID} returns C{None} for calendars which
- don't exist.
- """
- calendar1 = self.calendarUnderTest()
- self.assertEquals(calendar1.calendarObjectWithUID("xyzzy"), None)
-
-
- def test_calendars(self):
- """
- L{ICalendarHome.calendars} returns an iterable of L{ICalendar}
- providers, which are consistent with the results from
- L{ICalendar.calendarWithName}.
- """
- # Add a dot directory to make sure we don't find it
- # self.home1._path.child(".foo").createDirectory()
- home = self.homeUnderTest()
- calendars = list(home.calendars())
-
- for calendar in calendars:
- self.assertProvides(ICalendar, calendar)
- self.assertEquals(calendar,
- home.calendarWithName(calendar.name()))
-
- self.assertEquals(
- set(c.name() for c in calendars),
- set(home1_calendarNames)
- )
-
-
- def test_calendarsAfterAddCalendar(self):
- """
- L{ICalendarHome.calendars} includes calendars recently added with
- L{ICalendarHome.createCalendarWithName}.
- """
- home = self.homeUnderTest()
- before = set(x.name() for x in home.calendars())
- home.createCalendarWithName("new-name")
- after = set(x.name() for x in home.calendars())
- self.assertEquals(before | set(['new-name']), after)
-
-
- def test_createCalendarObjectWithName_absent(self):
- """
- L{ICalendar.createCalendarObjectWithName} creates a new
- L{ICalendarObject}.
- """
- calendar1 = self.calendarUnderTest()
- name = "4.ics"
- self.assertIdentical(calendar1.calendarObjectWithName(name), None)
- component = VComponent.fromString(event4_text)
- calendar1.createCalendarObjectWithName(name, component)
-
- calendarObject = calendar1.calendarObjectWithName(name)
- self.assertEquals(calendarObject.component(), component)
-
- self.commit()
-
- # Make sure notifications fire after commit
- self.assertEquals(
- self.notifierFactory.history,
- [
- ("update", "home1"),
- ("update", "home1/calendar_1"),
- ]
- )
-
-
- def test_createCalendarObjectWithName_exists(self):
- """
- L{ICalendar.createCalendarObjectWithName} raises
- L{CalendarObjectNameAlreadyExistsError} if a calendar object with the
- given name already exists in that calendar.
- """
- cal = self.calendarUnderTest()
- comp = VComponent.fromString(event4_text)
- self.assertRaises(
- ObjectResourceNameAlreadyExistsError,
- cal.createCalendarObjectWithName,
- "1.ics", comp
- )
-
-
- def test_createCalendarObjectWithName_invalid(self):
- """
- L{ICalendar.createCalendarObjectWithName} raises
- L{InvalidCalendarComponentError} if presented with invalid iCalendar
- text.
- """
- self.assertRaises(
- InvalidObjectResourceError,
- self.calendarUnderTest().createCalendarObjectWithName,
- "new", VComponent.fromString(event4notCalDAV_text)
- )
-
-
- def test_setComponent_invalid(self):
- """
- L{ICalendarObject.setComponent} raises L{InvalidICalendarDataError} if
- presented with invalid iCalendar text.
- """
- calendarObject = self.calendarObjectUnderTest()
- self.assertRaises(
- InvalidObjectResourceError,
- calendarObject.setComponent,
- VComponent.fromString(event4notCalDAV_text)
- )
-
-
- def test_setComponent_uidchanged(self):
- """
- L{ICalendarObject.setComponent} raises L{InvalidCalendarComponentError}
- when given a L{VComponent} whose UID does not match its existing UID.
- """
- calendar1 = self.calendarUnderTest()
- component = VComponent.fromString(event4_text)
- calendarObject = calendar1.calendarObjectWithName("1.ics")
- self.assertRaises(
- InvalidObjectResourceError,
- calendarObject.setComponent, component
- )
-
-
- def test_calendarHomeWithUID_create(self):
- """
- L{ICommonStoreTransaction.calendarHomeWithUID} with C{create=True}
- will create a calendar home that doesn't exist yet.
- """
- txn = self.transactionUnderTest()
- noHomeUID = "xyzzy"
- calendarHome = txn.calendarHomeWithUID(
- noHomeUID,
- create=True
- )
- def readOtherTxn():
- otherTxn = self.savedStore.newTransaction(self.id() + "other txn")
- self.addCleanup(otherTxn.commit)
- return otherTxn.calendarHomeWithUID(noHomeUID)
- self.assertProvides(ICalendarHome, calendarHome)
- # Default calendar should be automatically created.
- self.assertProvides(ICalendar,
- calendarHome.calendarWithName("calendar"))
- # A concurrent transaction shouldn't be able to read it yet:
- self.assertIdentical(readOtherTxn(), None)
- self.commit()
- # But once it's committed, other transactions should see it.
- self.assertProvides(ICalendarHome, readOtherTxn())
-
-
- def test_setComponent(self):
- """
- L{CalendarObject.setComponent} changes the result of
- L{CalendarObject.component} within the same transaction.
- """
- component = VComponent.fromString(event1modified_text)
-
- calendar1 = self.calendarUnderTest()
- calendarObject = calendar1.calendarObjectWithName("1.ics")
- oldComponent = calendarObject.component()
- self.assertNotEqual(component, oldComponent)
- calendarObject.setComponent(component)
- self.assertEquals(calendarObject.component(), component)
-
- # Also check a new instance
- calendarObject = calendar1.calendarObjectWithName("1.ics")
- self.assertEquals(calendarObject.component(), component)
-
- self.commit()
-
- # Make sure notification fired after commit
- self.assertEquals(
- self.notifierFactory.history,
- [
- ("update", "home1"),
- ("update", "home1/calendar_1"),
- ]
- )
-
-
- def checkPropertiesMethod(self, thunk):
- """
- Verify that the given object has a properties method that returns an
- L{IPropertyStore}.
- """
- properties = thunk.properties()
- self.assertProvides(IPropertyStore, properties)
-
-
- def test_homeProperties(self):
- """
- L{ICalendarHome.properties} returns a property store.
- """
- self.checkPropertiesMethod(self.homeUnderTest())
-
-
- def test_calendarProperties(self):
- """
- L{ICalendar.properties} returns a property store.
- """
- self.checkPropertiesMethod(self.calendarUnderTest())
-
-
- def test_calendarObjectProperties(self):
- """
- L{ICalendarObject.properties} returns a property store.
- """
- self.checkPropertiesMethod(self.calendarObjectUnderTest())
-
-
- def test_newCalendarObjectProperties(self):
- """
- L{ICalendarObject.properties} returns an empty property store for a
- calendar object which has been created but not committed.
- """
- calendar = self.calendarUnderTest()
- calendar.createCalendarObjectWithName(
- "4.ics", VComponent.fromString(event4_text)
- )
- newEvent = calendar.calendarObjectWithName("4.ics")
- self.assertEquals(newEvent.properties().items(), [])
-
-
- def test_setComponentPreservesProperties(self):
- """
- L{ICalendarObject.setComponent} preserves properties.
-
- (Some implementations must go to extra trouble to provide this
- behavior; for example, file storage must copy extended attributes from
- the existing file to the temporary file replacing it.)
- """
- propertyName = PropertyName("http://example.com/ns", "example")
- propertyContent = WebDAVUnknownElement("sample content")
- propertyContent.name = propertyName.name
- propertyContent.namespace = propertyName.namespace
-
- self.calendarObjectUnderTest().properties()[
- propertyName] = propertyContent
- self.commit()
- # Sanity check; are properties even readable in a separate transaction?
- # Should probably be a separate test.
- self.assertEquals(
- self.calendarObjectUnderTest().properties()[propertyName],
- propertyContent)
- obj = self.calendarObjectUnderTest()
- event1_text = obj.iCalendarText()
- event1_text_withDifferentSubject = event1_text.replace(
- "SUMMARY:CalDAV protocol updates",
- "SUMMARY:Changed"
- )
- # Sanity check; make sure the test has the right idea of the subject.
- self.assertNotEquals(event1_text, event1_text_withDifferentSubject)
- newComponent = VComponent.fromString(event1_text_withDifferentSubject)
- obj.setComponent(newComponent)
-
- # Putting everything into a separate transaction to account for any
- # caching that may take place.
- self.commit()
- self.assertEquals(
- self.calendarObjectUnderTest().properties()[propertyName],
- propertyContent
- )
-
-
- eventWithDropbox = "\r\n".join("""
-BEGIN:VCALENDAR
-CALSCALE:GREGORIAN
-PRODID:-//Example Inc.//Example Calendar//EN
-VERSION:2.0
-BEGIN:VTIMEZONE
-LAST-MODIFIED:20040110T032845Z
-TZID:US/Eastern
-BEGIN:DAYLIGHT
-DTSTART:20000404T020000
-RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
-TZNAME:EDT
-TZOFFSETFROM:-0500
-TZOFFSETTO:-0400
-END:DAYLIGHT
-BEGIN:STANDARD
-DTSTART:20001026T020000
-RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
-TZNAME:EST
-TZOFFSETFROM:-0400
-TZOFFSETTO:-0500
-END:STANDARD
-END:VTIMEZONE
-BEGIN:VEVENT
-DTSTAMP:20051222T205953Z
-CREATED:20060101T150000Z
-DTSTART;TZID=US/Eastern:20060101T100000
-DURATION:PT1H
-SUMMARY:event 1
-UID:event1 at ninevah.local
-ORGANIZER:user01
-ATTENDEE;PARTSTAT=ACCEPTED:user01
-ATTACH;VALUE=URI:/calendars/users/home1/some-dropbox-id/some-dropbox-id/caldavd.plist
-X-APPLE-DROPBOX:/calendars/users/home1/dropbox/some-dropbox-id
-END:VEVENT
-END:VCALENDAR
- """.strip().split("\n"))
-
- def test_dropboxID(self):
- """
- L{ICalendarObject.dropboxID} should synthesize its dropbox from the X
- -APPLE-DROPBOX property, if available.
- """
- cal = self.calendarUnderTest()
- cal.createCalendarObjectWithName("drop.ics", VComponent.fromString(
- self.eventWithDropbox
- )
- )
- obj = cal.calendarObjectWithName("drop.ics")
- self.assertEquals(obj.dropboxID(), "some-dropbox-id")
-
-
- def test_indexByDropboxProperty(self):
- """
- L{ICalendarHome.calendarObjectWithDropboxID} will return a calendar
- object in the calendar home with the given final segment in its C{X
- -APPLE-DROPBOX} property URI.
- """
- objName = "with-dropbox.ics"
- cal = self.calendarUnderTest()
- cal.createCalendarObjectWithName(
- objName, VComponent.fromString(
- self.eventWithDropbox
- )
- )
- self.commit()
- home = self.homeUnderTest()
- cal = self.calendarUnderTest()
- fromName = cal.calendarObjectWithName(objName)
- fromDropbox = home.calendarObjectWithDropboxID("some-dropbox-id")
- self.assertEquals(fromName, fromDropbox)
-
-
- @inlineCallbacks
- def createAttachmentTest(self, refresh):
- """
- Common logic for attachment-creation tests.
- """
- obj = self.calendarObjectUnderTest()
- t = obj.createAttachmentWithName("new.attachment", MimeType("text", "x-fixture"))
- t.write("new attachment")
- t.write(" text")
- t.loseConnection()
- obj = refresh(obj)
- class CaptureProtocol(Protocol):
- buf = ''
- def dataReceived(self, data):
- self.buf += data
- def connectionLost(self, reason):
- self.deferred.callback(self.buf)
- capture = CaptureProtocol()
- capture.deferred = Deferred()
- attachment = obj.attachmentWithName("new.attachment")
- self.assertProvides(IAttachment, attachment)
- attachment.retrieve(capture)
- data = yield capture.deferred
- self.assertEquals(data, "new attachment text")
- contentType = attachment.contentType()
- self.assertIsInstance(contentType, MimeType)
- self.assertEquals(contentType, MimeType("text", "x-fixture"))
- self.assertEquals(attachment.md5(), '50a9f27aeed9247a0833f30a631f1858')
- self.assertEquals(
- [attachment.name() for attachment in obj.attachments()],
- ['new.attachment']
- )
-
-
- def test_createAttachment(self):
- """
- L{ICalendarObject.createAttachmentWithName} will store an
- L{IAttachment} object that can be retrieved by
- L{ICalendarObject.attachmentWithName}.
- """
- return self.createAttachmentTest(lambda x: x)
-
-
- def test_createAttachmentCommit(self):
- """
- L{ICalendarObject.createAttachmentWithName} will store an
- L{IAttachment} object that can be retrieved by
- L{ICalendarObject.attachmentWithName} in subsequent transactions.
- """
- def refresh(obj):
- self.commit()
- return self.calendarObjectUnderTest()
- return self.createAttachmentTest(refresh)
-
-
- def test_removeAttachmentWithName(self, refresh=lambda x:x):
- """
- L{ICalendarObject.removeAttachmentWithName} will remove the calendar
- object with the given name.
- """
- def deleteIt(ignored):
- obj = self.calendarObjectUnderTest()
- obj.removeAttachmentWithName("new.attachment")
- obj = refresh(obj)
- self.assertIdentical(
- None, obj.attachmentWithName("new.attachment")
- )
- self.assertEquals(list(obj.attachments()), [])
- return self.test_createAttachmentCommit().addCallback(deleteIt)
-
-
- def test_removeAttachmentWithNameCommit(self):
- """
- L{ICalendarObject.removeAttachmentWithName} will remove the calendar
- object with the given name. (After commit, it will still be gone.)
- """
- def refresh(obj):
- self.commit()
- return self.calendarObjectUnderTest()
- return self.test_removeAttachmentWithName(refresh)
-
-
- def test_noDropboxCalendar(self):
- """
- L{ICalendarObject.createAttachmentWithName} may create a directory
- named 'dropbox', but this should not be seen as a calendar by
- L{ICalendarHome.calendarWithName} or L{ICalendarHome.calendars}.
- """
- obj = self.calendarObjectUnderTest()
- t = obj.createAttachmentWithName("new.attachment", MimeType("text", "plain"))
- t.write("new attachment text")
- t.loseConnection()
- self.commit()
- self.assertEquals(self.homeUnderTest().calendarWithName("dropbox"),
- None)
- self.assertEquals(
- set([n.name() for n in self.homeUnderTest().calendars()]),
- set(home1_calendarNames))
-
-
- def test_finishedOnCommit(self):
- """
- Calling L{ITransaction.abort} or L{ITransaction.commit} after
- L{ITransaction.commit} has already been called raises an
- L{AlreadyFinishedError}.
- """
- self.calendarObjectUnderTest()
- txn = self.lastTransaction
- self.commit()
- self.assertRaises(AlreadyFinishedError, txn.commit)
- self.assertRaises(AlreadyFinishedError, txn.abort)
-
-
- def test_dontLeakCalendars(self):
- """
- Calendars in one user's calendar home should not show up in another
- user's calendar home.
- """
- home2 = self.transactionUnderTest().calendarHomeWithUID(
- "home2", create=True)
- self.assertIdentical(home2.calendarWithName("calendar_1"), None)
-
-
- def test_dontLeakObjects(self):
- """
- Calendar objects in one user's calendar should not show up in another
- user's via uid or name queries.
- """
- home1 = self.homeUnderTest()
- home2 = self.transactionUnderTest().calendarHomeWithUID(
- "home2", create=True)
- calendar1 = home1.calendarWithName("calendar_1")
- calendar2 = home2.calendarWithName("calendar")
- objects = list(home2.calendarWithName("calendar").calendarObjects())
- self.assertEquals(objects, [])
- for resourceName in self.requirements['home1']['calendar_1'].keys():
- obj = calendar1.calendarObjectWithName(resourceName)
- self.assertIdentical(
- calendar2.calendarObjectWithName(resourceName), None)
- self.assertIdentical(
- calendar2.calendarObjectWithUID(obj.uid()), None)
-
-
-
-class StubNotifierFactory(object):
-
- """ For testing push notifications without an XMPP server """
-
- def __init__(self):
- self.reset()
-
- def newNotifier(self, label="default", id=None):
- return Notifier(self, label=label, id=id)
-
- def send(self, op, id):
- self.history.append((op, id))
-
- def reset(self):
- self.history = []
Copied: CalendarServer/trunk/txdav/caldav/datastore/test/common.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/common.py)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/test/common.py (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/common.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,1202 @@
+# -*- test-case-name: txdav.caldav.datastore -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Tests for common calendar store API functions.
+"""
+
+from zope.interface.verify import verifyObject
+from zope.interface.exceptions import (
+ BrokenMethodImplementation, DoesNotImplement)
+
+from twisted.internet.defer import Deferred, inlineCallbacks
+from twisted.internet.protocol import Protocol
+
+from txdav.idav import IPropertyStore, IDataStore, AlreadyFinishedError
+from txdav.base.propertystore.base import PropertyName
+
+from txdav.common.icommondatastore import HomeChildNameAlreadyExistsError, \
+ ICommonTransaction
+from txdav.common.icommondatastore import InvalidObjectResourceError
+from txdav.common.icommondatastore import NoSuchHomeChildError
+from txdav.common.icommondatastore import NoSuchObjectResourceError
+from txdav.common.icommondatastore import ObjectResourceNameAlreadyExistsError
+from txdav.common.inotifications import INotificationObject
+
+from txdav.caldav.icalendarstore import (
+ ICalendarObject, ICalendarHome,
+ ICalendar, IAttachment, ICalendarTransaction)
+
+from twext.python.filepath import CachingFilePath as FilePath
+from twext.web2.dav import davxml
+from twext.web2.http_headers import MimeType
+from twext.web2.dav.element.base import WebDAVUnknownElement
+from twext.python.vcomponent import VComponent
+
+from twistedcaldav.notify import Notifier
+from twistedcaldav.customxml import InviteNotification, InviteSummary
+
+storePath = FilePath(__file__).parent().child("calendar_store")
+
+homeRoot = storePath.child("ho").child("me").child("home1")
+
+cal1Root = homeRoot.child("calendar_1")
+
+calendar1_objectNames = [
+ "1.ics",
+ "2.ics",
+ "3.ics",
+]
+
+
+home1_calendarNames = [
+ "calendar_1",
+ "calendar_2",
+ "calendar_empty",
+]
+
+
+event4_text = (
+ "BEGIN:VCALENDAR\r\n"
+ "VERSION:2.0\r\n"
+ "PRODID:-//Apple Inc.//iCal 4.0.1//EN\r\n"
+ "CALSCALE:GREGORIAN\r\n"
+ "BEGIN:VTIMEZONE\r\n"
+ "TZID:US/Pacific\r\n"
+ "BEGIN:DAYLIGHT\r\n"
+ "TZOFFSETFROM:-0800\r\n"
+ "RRULE:FREQ=YEARLY;BYMONTH=3;BYDAY=2SU\r\n"
+ "DTSTART:20070311T020000\r\n"
+ "TZNAME:PDT\r\n"
+ "TZOFFSETTO:-0700\r\n"
+ "END:DAYLIGHT\r\n"
+ "BEGIN:STANDARD\r\n"
+ "TZOFFSETFROM:-0700\r\n"
+ "RRULE:FREQ=YEARLY;BYMONTH=11;BYDAY=1SU\r\n"
+ "DTSTART:20071104T020000\r\n"
+ "TZNAME:PST\r\n"
+ "TZOFFSETTO:-0800\r\n"
+ "END:STANDARD\r\n"
+ "END:VTIMEZONE\r\n"
+ "BEGIN:VEVENT\r\n"
+ "CREATED:20100203T013849Z\r\n"
+ "UID:uid4\r\n"
+ "DTEND;TZID=US/Pacific:20100207T173000\r\n"
+ "TRANSP:OPAQUE\r\n"
+ "SUMMARY:New Event\r\n"
+ "DTSTART;TZID=US/Pacific:20100207T170000\r\n"
+ "DTSTAMP:20100203T013909Z\r\n"
+ "SEQUENCE:3\r\n"
+ "BEGIN:VALARM\r\n"
+ "X-WR-ALARMUID:1377CCC7-F85C-4610-8583-9513D4B364E1\r\n"
+ "TRIGGER:-PT20M\r\n"
+ "ATTACH;VALUE=URI:Basso\r\n"
+ "ACTION:AUDIO\r\n"
+ "END:VALARM\r\n"
+ "END:VEVENT\r\n"
+ "END:VCALENDAR\r\n"
+)
+
+
+
+event4notCalDAV_text = (
+ "BEGIN:VCALENDAR\r\n"
+ "VERSION:2.0\r\n"
+ "PRODID:-//Apple Inc.//iCal 4.0.1//EN\r\n"
+ "CALSCALE:GREGORIAN\r\n"
+ "BEGIN:VEVENT\r\n"
+ "CREATED:20100203T013849Z\r\n"
+ "UID:4\r\n"
+ "DTEND;TZID=US/Pacific:20100207T173000\r\n" # TZID without VTIMEZONE
+ "TRANSP:OPAQUE\r\n"
+ "SUMMARY:New Event\r\n"
+ "DTSTART;TZID=US/Pacific:20100207T170000\r\n"
+ "DTSTAMP:20100203T013909Z\r\n"
+ "SEQUENCE:3\r\n"
+ "BEGIN:VALARM\r\n"
+ "X-WR-ALARMUID:1377CCC7-F85C-4610-8583-9513D4B364E1\r\n"
+ "TRIGGER:-PT20M\r\n"
+ "ATTACH;VALUE=URI:Basso\r\n"
+ "ACTION:AUDIO\r\n"
+ "END:VALARM\r\n"
+ "END:VEVENT\r\n"
+ "END:VCALENDAR\r\n"
+)
+
+
+
+event1modified_text = event4_text.replace(
+ "\r\nUID:uid4\r\n",
+ "\r\nUID:uid1\r\n"
+)
+
+
+
+def assertProvides(testCase, interface, provider):
+ """
+ Verify that C{provider} properly provides C{interface}
+
+ @type interface: L{zope.interface.Interface}
+ @type provider: C{provider}
+ """
+ try:
+ verifyObject(interface, provider)
+ except BrokenMethodImplementation, e:
+ testCase.fail(e)
+ except DoesNotImplement, e:
+ testCase.fail("%r does not provide %s.%s" %
+ (provider, interface.__module__, interface.getName()))
+
+
+class CommonTests(object):
+ """
+ Tests for common functionality of interfaces defined in
+ L{txdav.caldav.icalendarstore}.
+ """
+
+ requirements = {
+ "home1": {
+ "calendar_1": {
+ "1.ics": cal1Root.child("1.ics").getContent(),
+ "2.ics": cal1Root.child("2.ics").getContent(),
+ "3.ics": cal1Root.child("3.ics").getContent()
+ },
+ "calendar_2": {},
+ "calendar_empty": {},
+ "not_a_calendar": None
+ },
+ "not_a_home": None
+ }
+
+ def storeUnderTest(self):
+ """
+ Subclasses must override this to return an L{ICommonDataStore} provider
+ which adheres to the structure detailed by L{CommonTests.requirements}.
+ This attribute is a dict of dict of dicts; the outermost layer
+ representing UIDs mapping to calendar homes, then calendar names mapping
+ to calendar collections, and finally calendar object names mapping to
+ calendar object text.
+ """
+ raise NotImplementedError()
+
+
+ lastTransaction = None
+ savedStore = None
+
+ def transactionUnderTest(self):
+ """
+ Create a transaction from C{storeUnderTest} and save it as
+ C[lastTransaction}. Also makes sure to use the same store, saving the
+ value from C{storeUnderTest}.
+ """
+ if self.lastTransaction is not None:
+ return self.lastTransaction
+ if self.savedStore is None:
+ self.savedStore = self.storeUnderTest()
+ txn = self.lastTransaction = self.savedStore.newTransaction(self.id())
+ return txn
+
+
+ def commit(self):
+ """
+ Commit the last transaction created from C{transactionUnderTest}, and
+ clear it.
+ """
+ self.lastTransaction.commit()
+ self.lastTransaction = None
+
+
+ def abort(self):
+ """
+ Abort the last transaction created from C[transactionUnderTest}, and
+ clear it.
+ """
+ self.lastTransaction.abort()
+ self.lastTransaction = None
+
+
+ def setUp(self):
+ self.notifierFactory = StubNotifierFactory()
+
+
+ def tearDown(self):
+ if self.lastTransaction is not None:
+ self.commit()
+
+
+ def homeUnderTest(self):
+ """
+ Get the calendar home detailed by C{requirements['home1']}.
+ """
+ return self.transactionUnderTest().calendarHomeWithUID("home1")
+
+
+ def calendarUnderTest(self):
+ """
+ Get the calendar detailed by C{requirements['home1']['calendar_1']}.
+ """
+ return self.homeUnderTest().calendarWithName("calendar_1")
+
+
+ def calendarObjectUnderTest(self):
+ """
+ Get the calendar detailed by
+ C{requirements['home1']['calendar_1']['1.ics']}.
+ """
+ return self.calendarUnderTest().calendarObjectWithName("1.ics")
+
+
+ assertProvides = assertProvides
+
+ def test_calendarStoreProvides(self):
+ """
+ The calendar store provides L{IDataStore} and its required attributes.
+ """
+ calendarStore = self.storeUnderTest()
+ self.assertProvides(IDataStore, calendarStore)
+
+
+ def test_transactionProvides(self):
+ """
+ The transactions generated by the calendar store provide
+ L{ICommonStoreTransaction}, L{ICalendarTransaction}, and their
+ respectively required attributes.
+ """
+ txn = self.transactionUnderTest()
+ self.assertProvides(ICommonTransaction, txn)
+ self.assertProvides(ICalendarTransaction, txn)
+
+
+ def test_homeProvides(self):
+ """
+ The calendar homes generated by the calendar store provide
+ L{ICalendarHome} and its required attributes.
+ """
+ self.assertProvides(ICalendarHome, self.homeUnderTest())
+
+
+ def test_calendarProvides(self):
+ """
+ The calendars generated by the calendar store provide L{ICalendar} and
+ its required attributes.
+ """
+ self.assertProvides(ICalendar, self.calendarUnderTest())
+
+
+ def test_calendarObjectProvides(self):
+ """
+ The calendar objects generated by the calendar store provide
+ L{ICalendarObject} and its required attributes.
+ """
+ self.assertProvides(ICalendarObject, self.calendarObjectUnderTest())
+
+
+ def notificationUnderTest(self):
+ txn = self.transactionUnderTest()
+ notifications = txn.notificationsWithUID("home1")
+ inviteNotification = InviteNotification()
+ notifications.writeNotificationObject("abc", inviteNotification,
+ inviteNotification.toxml())
+ notificationObject = notifications.notificationObjectWithUID("abc")
+ return notificationObject
+
+
+ def test_notificationObjectProvides(self):
+ """
+ The objects retrieved from the notification home (the object returned
+ from L{notificationsWithUID}) provide L{INotificationObject}.
+ """
+ notificationObject = self.notificationUnderTest()
+ self.assertProvides(INotificationObject, notificationObject)
+
+
+ def test_replaceNotification(self):
+ """
+ L{INotificationCollection.writeNotificationObject} will silently
+ overwrite the notification object.
+ """
+ notifications = self.transactionUnderTest().notificationsWithUID(
+ "home1"
+ )
+ inviteNotification = InviteNotification()
+ notifications.writeNotificationObject("abc", inviteNotification,
+ inviteNotification.toxml())
+ inviteNotification2 = InviteNotification(InviteSummary("a summary"))
+ notifications.writeNotificationObject(
+ "abc", inviteNotification, inviteNotification2.toxml())
+ abc = notifications.notificationObjectWithUID("abc")
+ self.assertEquals(abc.xmldata(), inviteNotification2.toxml())
+
+
+ def test_notificationObjectModified(self):
+ """
+ The objects retrieved from the notification home have a C{modified}
+ method which returns the timestamp of their last modification.
+ """
+ notification = self.notificationUnderTest()
+ self.assertIsInstance(notification.modified(), int)
+
+
+ def test_notificationObjectParent(self):
+ """
+ L{INotificationObject.notificationCollection} returns the
+ L{INotificationCollection} that the object was retrieved from.
+ """
+ txn = self.transactionUnderTest()
+ collection = txn.notificationsWithUID("home1")
+ notification = self.notificationUnderTest()
+ self.assertIdentical(collection, notification.notificationCollection())
+
+
+ def test_notifierID(self):
+ home = self.homeUnderTest()
+ self.assertEquals(home.notifierID(), "home1")
+ calendar = home.calendarWithName("calendar_1")
+ self.assertEquals(calendar.notifierID(), "home1")
+ self.assertEquals(calendar.notifierID(label="collection"), "home1/calendar_1")
+
+
+ def test_calendarHomeWithUID_exists(self):
+ """
+ Finding an existing calendar home by UID results in an object that
+ provides L{ICalendarHome} and has a C{uid()} method that returns the
+ same value that was passed in.
+ """
+ calendarHome = (self.transactionUnderTest()
+ .calendarHomeWithUID("home1"))
+ self.assertEquals(calendarHome.uid(), "home1")
+ self.assertProvides(ICalendarHome, calendarHome)
+
+
+ def test_calendarHomeWithUID_absent(self):
+ """
+ L{ICommonStoreTransaction.calendarHomeWithUID} should return C{None}
+ when asked for a non-existent calendar home.
+ """
+ txn = self.transactionUnderTest()
+ self.assertEquals(txn.calendarHomeWithUID("xyzzy"), None)
+
+
+ def test_calendarWithName_exists(self):
+ """
+ L{ICalendarHome.calendarWithName} returns an L{ICalendar} provider,
+ whose name matches the one passed in.
+ """
+ home = self.homeUnderTest()
+ for name in home1_calendarNames:
+ calendar = home.calendarWithName(name)
+ if calendar is None:
+ self.fail("calendar %r didn't exist" % (name,))
+ self.assertProvides(ICalendar, calendar)
+ self.assertEquals(calendar.name(), name)
+
+
+ def test_calendarRename(self):
+ """
+ L{ICalendar.rename} changes the name of the L{ICalendar}.
+ """
+ home = self.homeUnderTest()
+ calendar = home.calendarWithName("calendar_1")
+ calendar.rename("some_other_name")
+ def positiveAssertions():
+ self.assertEquals(calendar.name(), "some_other_name")
+ self.assertEquals(calendar, home.calendarWithName("some_other_name"))
+ self.assertEquals(None, home.calendarWithName("calendar_1"))
+ positiveAssertions()
+ self.commit()
+ home = self.homeUnderTest()
+ calendar = home.calendarWithName("some_other_name")
+ positiveAssertions()
+ # FIXME: revert
+ # FIXME: test for multiple renames
+ # FIXME: test for conflicting renames (a->b, c->a in the same txn)
+
+
+ def test_calendarWithName_absent(self):
+ """
+ L{ICalendarHome.calendarWithName} returns C{None} for calendars which
+ do not exist.
+ """
+ self.assertEquals(self.homeUnderTest().calendarWithName("xyzzy"),
+ None)
+
+
+ def test_createCalendarWithName_absent(self):
+ """
+ L{ICalendarHome.createCalendarWithName} creates a new L{ICalendar} that
+ can be retrieved with L{ICalendarHome.calendarWithName}.
+ """
+ home = self.homeUnderTest()
+ name = "new"
+ self.assertIdentical(home.calendarWithName(name), None)
+ home.createCalendarWithName(name)
+ self.assertNotIdentical(home.calendarWithName(name), None)
+ def checkProperties():
+ calendarProperties = home.calendarWithName(name).properties()
+ self.assertEquals(
+ calendarProperties[
+ PropertyName.fromString(davxml.ResourceType.sname())
+ ],
+ davxml.ResourceType.calendar #@UndefinedVariable
+ )
+ checkProperties()
+
+ self.commit()
+
+ # Make sure notification fired after commit
+ self.assertEquals(self.notifierFactory.history, [("update", "home1")])
+
+ # Make sure it's available in a new transaction; i.e. test the commit.
+ home = self.homeUnderTest()
+ self.assertNotIdentical(home.calendarWithName(name), None)
+
+ # Sanity check: are the properties actually persisted? Check in
+ # subsequent transaction.
+ checkProperties()
+
+ # FIXME: no independent testing of the property store's persistence
+ # right now
+
+
+ def test_createCalendarWithName_exists(self):
+ """
+ L{ICalendarHome.createCalendarWithName} raises
+ L{CalendarAlreadyExistsError} when the name conflicts with an already-
+ existing
+ """
+ for name in home1_calendarNames:
+ self.assertRaises(
+ HomeChildNameAlreadyExistsError,
+ self.homeUnderTest().createCalendarWithName, name
+ )
+
+
+ def test_removeCalendarWithName_exists(self):
+ """
+ L{ICalendarHome.removeCalendarWithName} removes a calendar that already
+ exists.
+ """
+ home = self.homeUnderTest()
+
+ # FIXME: test transactions
+ for name in home1_calendarNames:
+ self.assertNotIdentical(home.calendarWithName(name), None)
+ home.removeCalendarWithName(name)
+ self.assertEquals(home.calendarWithName(name), None)
+
+ self.commit()
+
+ # Make sure notification fired after commit
+ self.assertEquals(
+ self.notifierFactory.history,
+ [("update", "home1"), ("update", "home1"), ("update", "home1")]
+ )
+
+
+ def test_removeCalendarWithName_absent(self):
+ """
+ Attempt to remove an non-existing calendar should raise.
+ """
+ home = self.homeUnderTest()
+ self.assertRaises(NoSuchHomeChildError,
+ home.removeCalendarWithName, "xyzzy")
+
+
+ def test_calendarObjects(self):
+ """
+ L{ICalendar.calendarObjects} will enumerate the calendar objects present
+ in the filesystem, in name order, but skip those with hidden names.
+ """
+ calendar1 = self.calendarUnderTest()
+ calendarObjects = list(calendar1.calendarObjects())
+
+ for calendarObject in calendarObjects:
+ self.assertProvides(ICalendarObject, calendarObject)
+ self.assertEquals(
+ calendar1.calendarObjectWithName(calendarObject.name()),
+ calendarObject
+ )
+
+ self.assertEquals(
+ set(list(o.name() for o in calendarObjects)),
+ set(calendar1_objectNames)
+ )
+
+
+ def test_calendarObjectsWithRemovedObject(self):
+ """
+ L{ICalendar.calendarObjects} skips those objects which have been
+ removed by L{Calendar.removeCalendarObjectWithName} in the same
+ transaction, even if it has not yet been committed.
+ """
+ calendar1 = self.calendarUnderTest()
+ calendar1.removeCalendarObjectWithName("2.ics")
+ calendarObjects = list(calendar1.calendarObjects())
+ self.assertEquals(set(o.name() for o in calendarObjects),
+ set(calendar1_objectNames) - set(["2.ics"]))
+
+
+ def test_ownerCalendarHome(self):
+ """
+ L{ICalendar.ownerCalendarHome} should match the home UID.
+ """
+ self.assertEquals(
+ self.calendarUnderTest().ownerCalendarHome().uid(),
+ self.homeUnderTest().uid()
+ )
+
+
+ def test_calendarObjectWithName_exists(self):
+ """
+ L{ICalendar.calendarObjectWithName} returns an L{ICalendarObject}
+ provider for calendars which already exist.
+ """
+ calendar1 = self.calendarUnderTest()
+ for name in calendar1_objectNames:
+ calendarObject = calendar1.calendarObjectWithName(name)
+ self.assertProvides(ICalendarObject, calendarObject)
+ self.assertEquals(calendarObject.name(), name)
+ # FIXME: add more tests based on CommonTests.requirements
+
+
+ def test_calendarObjectWithName_absent(self):
+ """
+ L{ICalendar.calendarObjectWithName} returns C{None} for calendars which
+ don't exist.
+ """
+ calendar1 = self.calendarUnderTest()
+ self.assertEquals(calendar1.calendarObjectWithName("xyzzy"), None)
+
+
+ def test_removeCalendarObjectWithUID_exists(self):
+ """
+ Remove an existing calendar object.
+ """
+ calendar = self.calendarUnderTest()
+ for name in calendar1_objectNames:
+ uid = (u'uid' + name.rstrip(".ics"))
+ self.assertNotIdentical(calendar.calendarObjectWithUID(uid),
+ None)
+ calendar.removeCalendarObjectWithUID(uid)
+ self.assertEquals(
+ calendar.calendarObjectWithUID(uid),
+ None
+ )
+ self.assertEquals(
+ calendar.calendarObjectWithName(name),
+ None
+ )
+
+ # Make sure notifications are fired after commit
+ self.commit()
+ self.assertEquals(
+ self.notifierFactory.history,
+ [
+ ("update", "home1"),
+ ("update", "home1/calendar_1"),
+ ("update", "home1"),
+ ("update", "home1/calendar_1"),
+ ("update", "home1"),
+ ("update", "home1/calendar_1"),
+ ]
+ )
+
+ def test_removeCalendarObjectWithName_exists(self):
+ """
+ Remove an existing calendar object.
+ """
+ calendar = self.calendarUnderTest()
+ for name in calendar1_objectNames:
+ self.assertNotIdentical(
+ calendar.calendarObjectWithName(name), None
+ )
+ calendar.removeCalendarObjectWithName(name)
+ self.assertIdentical(
+ calendar.calendarObjectWithName(name), None
+ )
+
+
+ def test_removeCalendarObjectWithName_absent(self):
+ """
+ Attempt to remove an non-existing calendar object should raise.
+ """
+ calendar = self.calendarUnderTest()
+ self.assertRaises(
+ NoSuchObjectResourceError,
+ calendar.removeCalendarObjectWithName, "xyzzy"
+ )
+
+
+ def test_calendarName(self):
+ """
+ L{Calendar.name} reflects the name of the calendar.
+ """
+ self.assertEquals(self.calendarUnderTest().name(), "calendar_1")
+
+
+ def test_calendarObjectName(self):
+ """
+ L{ICalendarObject.name} reflects the name of the calendar object.
+ """
+ self.assertEquals(self.calendarObjectUnderTest().name(), "1.ics")
+
+
+ def test_component(self):
+ """
+ L{ICalendarObject.component} returns a L{VComponent} describing the
+ calendar data underlying that calendar object.
+ """
+ component = self.calendarObjectUnderTest().component()
+
+ self.failUnless(
+ isinstance(component, VComponent),
+ component
+ )
+
+ self.assertEquals(component.name(), "VCALENDAR")
+ self.assertEquals(component.mainType(), "VEVENT")
+ self.assertEquals(component.resourceUID(), "uid1")
+
+
+ def test_iCalendarText(self):
+ """
+ L{ICalendarObject.iCalendarText} returns a C{str} describing the same
+ data provided by L{ICalendarObject.component}.
+ """
+ text = self.calendarObjectUnderTest().iCalendarText()
+ self.assertIsInstance(text, str)
+ self.failUnless(text.startswith("BEGIN:VCALENDAR\r\n"))
+ self.assertIn("\r\nUID:uid1\r\n", text)
+ self.failUnless(text.endswith("\r\nEND:VCALENDAR\r\n"))
+
+
+ def test_calendarObjectUID(self):
+ """
+ L{ICalendarObject.uid} returns a C{str} describing the C{UID} property
+ of the calendar object's component.
+ """
+ self.assertEquals(self.calendarObjectUnderTest().uid(), "uid1")
+
+
+ def test_organizer(self):
+ """
+ L{ICalendarObject.organizer} returns a C{str} describing the calendar
+ user address of the C{ORGANIZER} property of the calendar object's
+ component.
+ """
+ self.assertEquals(
+ self.calendarObjectUnderTest().organizer(),
+ "mailto:wsanchez at apple.com"
+ )
+
+
+ def test_calendarObjectWithUID_absent(self):
+ """
+ L{ICalendar.calendarObjectWithUID} returns C{None} for calendars which
+ don't exist.
+ """
+ calendar1 = self.calendarUnderTest()
+ self.assertEquals(calendar1.calendarObjectWithUID("xyzzy"), None)
+
+
+ def test_calendars(self):
+ """
+ L{ICalendarHome.calendars} returns an iterable of L{ICalendar}
+ providers, which are consistent with the results from
+ L{ICalendar.calendarWithName}.
+ """
+ # Add a dot directory to make sure we don't find it
+ # self.home1._path.child(".foo").createDirectory()
+ home = self.homeUnderTest()
+ calendars = list(home.calendars())
+
+ for calendar in calendars:
+ self.assertProvides(ICalendar, calendar)
+ self.assertEquals(calendar,
+ home.calendarWithName(calendar.name()))
+
+ self.assertEquals(
+ set(c.name() for c in calendars),
+ set(home1_calendarNames)
+ )
+
+
+ def test_calendarsAfterAddCalendar(self):
+ """
+ L{ICalendarHome.calendars} includes calendars recently added with
+ L{ICalendarHome.createCalendarWithName}.
+ """
+ home = self.homeUnderTest()
+ before = set(x.name() for x in home.calendars())
+ home.createCalendarWithName("new-name")
+ after = set(x.name() for x in home.calendars())
+ self.assertEquals(before | set(['new-name']), after)
+
+
+ def test_createCalendarObjectWithName_absent(self):
+ """
+ L{ICalendar.createCalendarObjectWithName} creates a new
+ L{ICalendarObject}.
+ """
+ calendar1 = self.calendarUnderTest()
+ name = "4.ics"
+ self.assertIdentical(calendar1.calendarObjectWithName(name), None)
+ component = VComponent.fromString(event4_text)
+ calendar1.createCalendarObjectWithName(name, component)
+
+ calendarObject = calendar1.calendarObjectWithName(name)
+ self.assertEquals(calendarObject.component(), component)
+
+ self.commit()
+
+ # Make sure notifications fire after commit
+ self.assertEquals(
+ self.notifierFactory.history,
+ [
+ ("update", "home1"),
+ ("update", "home1/calendar_1"),
+ ]
+ )
+
+
+ def test_createCalendarObjectWithName_exists(self):
+ """
+ L{ICalendar.createCalendarObjectWithName} raises
+ L{CalendarObjectNameAlreadyExistsError} if a calendar object with the
+ given name already exists in that calendar.
+ """
+ cal = self.calendarUnderTest()
+ comp = VComponent.fromString(event4_text)
+ self.assertRaises(
+ ObjectResourceNameAlreadyExistsError,
+ cal.createCalendarObjectWithName,
+ "1.ics", comp
+ )
+
+
+ def test_createCalendarObjectWithName_invalid(self):
+ """
+ L{ICalendar.createCalendarObjectWithName} raises
+ L{InvalidCalendarComponentError} if presented with invalid iCalendar
+ text.
+ """
+ self.assertRaises(
+ InvalidObjectResourceError,
+ self.calendarUnderTest().createCalendarObjectWithName,
+ "new", VComponent.fromString(event4notCalDAV_text)
+ )
+
+
+ def test_setComponent_invalid(self):
+ """
+ L{ICalendarObject.setComponent} raises L{InvalidICalendarDataError} if
+ presented with invalid iCalendar text.
+ """
+ calendarObject = self.calendarObjectUnderTest()
+ self.assertRaises(
+ InvalidObjectResourceError,
+ calendarObject.setComponent,
+ VComponent.fromString(event4notCalDAV_text)
+ )
+
+
+ def test_setComponent_uidchanged(self):
+ """
+ L{ICalendarObject.setComponent} raises L{InvalidCalendarComponentError}
+ when given a L{VComponent} whose UID does not match its existing UID.
+ """
+ calendar1 = self.calendarUnderTest()
+ component = VComponent.fromString(event4_text)
+ calendarObject = calendar1.calendarObjectWithName("1.ics")
+ self.assertRaises(
+ InvalidObjectResourceError,
+ calendarObject.setComponent, component
+ )
+
+
+ def test_calendarHomeWithUID_create(self):
+ """
+ L{ICommonStoreTransaction.calendarHomeWithUID} with C{create=True}
+ will create a calendar home that doesn't exist yet.
+ """
+ txn = self.transactionUnderTest()
+ noHomeUID = "xyzzy"
+ calendarHome = txn.calendarHomeWithUID(
+ noHomeUID,
+ create=True
+ )
+ def readOtherTxn():
+ otherTxn = self.savedStore.newTransaction(self.id() + "other txn")
+ self.addCleanup(otherTxn.commit)
+ return otherTxn.calendarHomeWithUID(noHomeUID)
+ self.assertProvides(ICalendarHome, calendarHome)
+ # Default calendar should be automatically created.
+ self.assertProvides(ICalendar,
+ calendarHome.calendarWithName("calendar"))
+ # A concurrent transaction shouldn't be able to read it yet:
+ self.assertIdentical(readOtherTxn(), None)
+ self.commit()
+ # But once it's committed, other transactions should see it.
+ self.assertProvides(ICalendarHome, readOtherTxn())
+
+
+ def test_setComponent(self):
+ """
+ L{CalendarObject.setComponent} changes the result of
+ L{CalendarObject.component} within the same transaction.
+ """
+ component = VComponent.fromString(event1modified_text)
+
+ calendar1 = self.calendarUnderTest()
+ calendarObject = calendar1.calendarObjectWithName("1.ics")
+ oldComponent = calendarObject.component()
+ self.assertNotEqual(component, oldComponent)
+ calendarObject.setComponent(component)
+ self.assertEquals(calendarObject.component(), component)
+
+ # Also check a new instance
+ calendarObject = calendar1.calendarObjectWithName("1.ics")
+ self.assertEquals(calendarObject.component(), component)
+
+ self.commit()
+
+ # Make sure notification fired after commit
+ self.assertEquals(
+ self.notifierFactory.history,
+ [
+ ("update", "home1"),
+ ("update", "home1/calendar_1"),
+ ]
+ )
+
+
+ def checkPropertiesMethod(self, thunk):
+ """
+ Verify that the given object has a properties method that returns an
+ L{IPropertyStore}.
+ """
+ properties = thunk.properties()
+ self.assertProvides(IPropertyStore, properties)
+
+
+ def test_homeProperties(self):
+ """
+ L{ICalendarHome.properties} returns a property store.
+ """
+ self.checkPropertiesMethod(self.homeUnderTest())
+
+
+ def test_calendarProperties(self):
+ """
+ L{ICalendar.properties} returns a property store.
+ """
+ self.checkPropertiesMethod(self.calendarUnderTest())
+
+
+ def test_calendarObjectProperties(self):
+ """
+ L{ICalendarObject.properties} returns a property store.
+ """
+ self.checkPropertiesMethod(self.calendarObjectUnderTest())
+
+
+ def test_newCalendarObjectProperties(self):
+ """
+ L{ICalendarObject.properties} returns an empty property store for a
+ calendar object which has been created but not committed.
+ """
+ calendar = self.calendarUnderTest()
+ calendar.createCalendarObjectWithName(
+ "4.ics", VComponent.fromString(event4_text)
+ )
+ newEvent = calendar.calendarObjectWithName("4.ics")
+ self.assertEquals(newEvent.properties().items(), [])
+
+
+ def test_setComponentPreservesProperties(self):
+ """
+ L{ICalendarObject.setComponent} preserves properties.
+
+ (Some implementations must go to extra trouble to provide this
+ behavior; for example, file storage must copy extended attributes from
+ the existing file to the temporary file replacing it.)
+ """
+ propertyName = PropertyName("http://example.com/ns", "example")
+ propertyContent = WebDAVUnknownElement("sample content")
+ propertyContent.name = propertyName.name
+ propertyContent.namespace = propertyName.namespace
+
+ self.calendarObjectUnderTest().properties()[
+ propertyName] = propertyContent
+ self.commit()
+ # Sanity check; are properties even readable in a separate transaction?
+ # Should probably be a separate test.
+ self.assertEquals(
+ self.calendarObjectUnderTest().properties()[propertyName],
+ propertyContent)
+ obj = self.calendarObjectUnderTest()
+ event1_text = obj.iCalendarText()
+ event1_text_withDifferentSubject = event1_text.replace(
+ "SUMMARY:CalDAV protocol updates",
+ "SUMMARY:Changed"
+ )
+ # Sanity check; make sure the test has the right idea of the subject.
+ self.assertNotEquals(event1_text, event1_text_withDifferentSubject)
+ newComponent = VComponent.fromString(event1_text_withDifferentSubject)
+ obj.setComponent(newComponent)
+
+ # Putting everything into a separate transaction to account for any
+ # caching that may take place.
+ self.commit()
+ self.assertEquals(
+ self.calendarObjectUnderTest().properties()[propertyName],
+ propertyContent
+ )
+
+
+ eventWithDropbox = "\r\n".join("""
+BEGIN:VCALENDAR
+CALSCALE:GREGORIAN
+PRODID:-//Example Inc.//Example Calendar//EN
+VERSION:2.0
+BEGIN:VTIMEZONE
+LAST-MODIFIED:20040110T032845Z
+TZID:US/Eastern
+BEGIN:DAYLIGHT
+DTSTART:20000404T020000
+RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4
+TZNAME:EDT
+TZOFFSETFROM:-0500
+TZOFFSETTO:-0400
+END:DAYLIGHT
+BEGIN:STANDARD
+DTSTART:20001026T020000
+RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10
+TZNAME:EST
+TZOFFSETFROM:-0400
+TZOFFSETTO:-0500
+END:STANDARD
+END:VTIMEZONE
+BEGIN:VEVENT
+DTSTAMP:20051222T205953Z
+CREATED:20060101T150000Z
+DTSTART;TZID=US/Eastern:20060101T100000
+DURATION:PT1H
+SUMMARY:event 1
+UID:event1 at ninevah.local
+ORGANIZER:user01
+ATTENDEE;PARTSTAT=ACCEPTED:user01
+ATTACH;VALUE=URI:/calendars/users/home1/some-dropbox-id/some-dropbox-id/caldavd.plist
+X-APPLE-DROPBOX:/calendars/users/home1/dropbox/some-dropbox-id
+END:VEVENT
+END:VCALENDAR
+ """.strip().split("\n"))
+
+ def test_dropboxID(self):
+ """
+ L{ICalendarObject.dropboxID} should synthesize its dropbox from the X
+ -APPLE-DROPBOX property, if available.
+ """
+ cal = self.calendarUnderTest()
+ cal.createCalendarObjectWithName("drop.ics", VComponent.fromString(
+ self.eventWithDropbox
+ )
+ )
+ obj = cal.calendarObjectWithName("drop.ics")
+ self.assertEquals(obj.dropboxID(), "some-dropbox-id")
+
+
+ def test_indexByDropboxProperty(self):
+ """
+ L{ICalendarHome.calendarObjectWithDropboxID} will return a calendar
+ object in the calendar home with the given final segment in its C{X
+ -APPLE-DROPBOX} property URI.
+ """
+ objName = "with-dropbox.ics"
+ cal = self.calendarUnderTest()
+ cal.createCalendarObjectWithName(
+ objName, VComponent.fromString(
+ self.eventWithDropbox
+ )
+ )
+ self.commit()
+ home = self.homeUnderTest()
+ cal = self.calendarUnderTest()
+ fromName = cal.calendarObjectWithName(objName)
+ fromDropbox = home.calendarObjectWithDropboxID("some-dropbox-id")
+ self.assertEquals(fromName, fromDropbox)
+
+
+ @inlineCallbacks
+ def createAttachmentTest(self, refresh):
+ """
+ Common logic for attachment-creation tests.
+ """
+ obj = self.calendarObjectUnderTest()
+ t = obj.createAttachmentWithName("new.attachment", MimeType("text", "x-fixture"))
+ t.write("new attachment")
+ t.write(" text")
+ t.loseConnection()
+ obj = refresh(obj)
+ class CaptureProtocol(Protocol):
+ buf = ''
+ def dataReceived(self, data):
+ self.buf += data
+ def connectionLost(self, reason):
+ self.deferred.callback(self.buf)
+ capture = CaptureProtocol()
+ capture.deferred = Deferred()
+ attachment = obj.attachmentWithName("new.attachment")
+ self.assertProvides(IAttachment, attachment)
+ attachment.retrieve(capture)
+ data = yield capture.deferred
+ self.assertEquals(data, "new attachment text")
+ contentType = attachment.contentType()
+ self.assertIsInstance(contentType, MimeType)
+ self.assertEquals(contentType, MimeType("text", "x-fixture"))
+ self.assertEquals(attachment.md5(), '50a9f27aeed9247a0833f30a631f1858')
+ self.assertEquals(
+ [attachment.name() for attachment in obj.attachments()],
+ ['new.attachment']
+ )
+
+
+ def test_createAttachment(self):
+ """
+ L{ICalendarObject.createAttachmentWithName} will store an
+ L{IAttachment} object that can be retrieved by
+ L{ICalendarObject.attachmentWithName}.
+ """
+ return self.createAttachmentTest(lambda x: x)
+
+
+ def test_createAttachmentCommit(self):
+ """
+ L{ICalendarObject.createAttachmentWithName} will store an
+ L{IAttachment} object that can be retrieved by
+ L{ICalendarObject.attachmentWithName} in subsequent transactions.
+ """
+ def refresh(obj):
+ self.commit()
+ return self.calendarObjectUnderTest()
+ return self.createAttachmentTest(refresh)
+
+
+ def test_removeAttachmentWithName(self, refresh=lambda x:x):
+ """
+ L{ICalendarObject.removeAttachmentWithName} will remove the calendar
+ object with the given name.
+ """
+ def deleteIt(ignored):
+ obj = self.calendarObjectUnderTest()
+ obj.removeAttachmentWithName("new.attachment")
+ obj = refresh(obj)
+ self.assertIdentical(
+ None, obj.attachmentWithName("new.attachment")
+ )
+ self.assertEquals(list(obj.attachments()), [])
+ return self.test_createAttachmentCommit().addCallback(deleteIt)
+
+
+ def test_removeAttachmentWithNameCommit(self):
+ """
+ L{ICalendarObject.removeAttachmentWithName} will remove the calendar
+ object with the given name. (After commit, it will still be gone.)
+ """
+ def refresh(obj):
+ self.commit()
+ return self.calendarObjectUnderTest()
+ return self.test_removeAttachmentWithName(refresh)
+
+
+ def test_noDropboxCalendar(self):
+ """
+ L{ICalendarObject.createAttachmentWithName} may create a directory
+ named 'dropbox', but this should not be seen as a calendar by
+ L{ICalendarHome.calendarWithName} or L{ICalendarHome.calendars}.
+ """
+ obj = self.calendarObjectUnderTest()
+ t = obj.createAttachmentWithName("new.attachment", MimeType("text", "plain"))
+ t.write("new attachment text")
+ t.loseConnection()
+ self.commit()
+ self.assertEquals(self.homeUnderTest().calendarWithName("dropbox"),
+ None)
+ self.assertEquals(
+ set([n.name() for n in self.homeUnderTest().calendars()]),
+ set(home1_calendarNames))
+
+
+ def test_finishedOnCommit(self):
+ """
+ Calling L{ITransaction.abort} or L{ITransaction.commit} after
+ L{ITransaction.commit} has already been called raises an
+ L{AlreadyFinishedError}.
+ """
+ self.calendarObjectUnderTest()
+ txn = self.lastTransaction
+ self.commit()
+ self.assertRaises(AlreadyFinishedError, txn.commit)
+ self.assertRaises(AlreadyFinishedError, txn.abort)
+
+
+ def test_dontLeakCalendars(self):
+ """
+ Calendars in one user's calendar home should not show up in another
+ user's calendar home.
+ """
+ home2 = self.transactionUnderTest().calendarHomeWithUID(
+ "home2", create=True)
+ self.assertIdentical(home2.calendarWithName("calendar_1"), None)
+
+
+ def test_dontLeakObjects(self):
+ """
+ Calendar objects in one user's calendar should not show up in another
+ user's via uid or name queries.
+ """
+ home1 = self.homeUnderTest()
+ home2 = self.transactionUnderTest().calendarHomeWithUID(
+ "home2", create=True)
+ calendar1 = home1.calendarWithName("calendar_1")
+ calendar2 = home2.calendarWithName("calendar")
+ objects = list(home2.calendarWithName("calendar").calendarObjects())
+ self.assertEquals(objects, [])
+ for resourceName in self.requirements['home1']['calendar_1'].keys():
+ obj = calendar1.calendarObjectWithName(resourceName)
+ self.assertIdentical(
+ calendar2.calendarObjectWithName(resourceName), None)
+ self.assertIdentical(
+ calendar2.calendarObjectWithUID(obj.uid()), None)
+
+
+
+class StubNotifierFactory(object):
+
+ """ For testing push notifications without an XMPP server """
+
+ def __init__(self):
+ self.reset()
+
+ def newNotifier(self, label="default", id=None):
+ return Notifier(self, label=label, id=id)
+
+ def send(self, op, id):
+ self.history.append((op, id))
+
+ def reset(self):
+ self.history = []
Deleted: CalendarServer/trunk/txdav/caldav/datastore/test/test_file.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/test_file.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/test_file.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,470 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-File calendar store tests.
-"""
-
-# FIXME: all test cases in this file aside from FileStorageTests should be
-# deleted and replaced with either implementation-specific methods on
-# FileStorageTests, or implementation-agnostic methods on CommonTests.
-
-from twext.python.filepath import CachingFilePath as FilePath
-from twisted.trial import unittest
-
-from twext.python.vcomponent import VComponent
-
-from txdav.common.icommondatastore import HomeChildNameNotAllowedError
-from txdav.common.icommondatastore import ObjectResourceNameNotAllowedError
-from txdav.common.icommondatastore import ObjectResourceUIDAlreadyExistsError
-from txdav.common.icommondatastore import NoSuchHomeChildError
-from txdav.common.icommondatastore import NoSuchObjectResourceError
-
-from txdav.caldav.datastore.file import CalendarStore, CalendarHome
-from txdav.caldav.datastore.file import Calendar, CalendarObject
-
-from txdav.caldav.datastore.test.common import (
- CommonTests, event4_text, event1modified_text)
-
-storePath = FilePath(__file__).parent().child("calendar_store")
-
-def _todo(f, why):
- f.todo = why
- return f
-
-
-
-featureUnimplemented = lambda f: _todo(f, "Feature unimplemented")
-testUnimplemented = lambda f: _todo(f, "Test unimplemented")
-todo = lambda why: lambda f: _todo(f, why)
-
-
-
-def setUpCalendarStore(test):
- test.root = FilePath(test.mktemp())
- test.root.createDirectory()
-
- storeRootPath = test.storeRootPath = test.root.child("store")
- calendarPath = storeRootPath.child("calendars").child("__uids__")
- calendarPath.parent().makedirs()
- storePath.copyTo(calendarPath)
-
- test.calendarStore = CalendarStore(storeRootPath, test.notifierFactory)
- test.txn = test.calendarStore.newTransaction()
- assert test.calendarStore is not None, "No calendar store?"
-
-
-
-def setUpHome1(test):
- setUpCalendarStore(test)
- test.home1 = test.txn.calendarHomeWithUID("home1")
- assert test.home1 is not None, "No calendar home?"
-
-
-
-def setUpCalendar1(test):
- setUpHome1(test)
- test.calendar1 = test.home1.calendarWithName("calendar_1")
- assert test.calendar1 is not None, "No calendar?"
-
-
-
-class CalendarStoreTest(unittest.TestCase):
- """
- Test cases for L{CalendarStore}.
- """
-
- notifierFactory = None
-
- def setUp(self):
- setUpCalendarStore(self)
-
-
- def test_calendarHomeWithUID_dot(self):
- """
- Filenames starting with "." are reserved by this
- implementation, so no UIDs may start with ".".
- """
- self.assertEquals(
- self.calendarStore.newTransaction().calendarHomeWithUID("xyzzy"),
- None
- )
-
-
-
-class CalendarHomeTest(unittest.TestCase):
-
- notifierFactory = None
- def setUp(self):
- setUpHome1(self)
-
-
- def test_init(self):
- """
- L{CalendarHome} has C{_path} and L{_calendarStore} attributes,
- indicating its location on disk and parent store, respectively.
- """
- self.failUnless(
- isinstance(self.home1._path, FilePath),
- self.home1._path
- )
- self.assertEquals(
- self.home1._calendarStore,
- self.calendarStore
- )
-
-
- def test_calendarWithName_dot(self):
- """
- Filenames starting with "." are reserved by this
- implementation, so no calendar names may start with ".".
- """
- name = ".foo"
- self.home1._path.child(name).createDirectory()
- self.assertEquals(self.home1.calendarWithName(name), None)
-
-
- def test_createCalendarWithName_dot(self):
- """
- Filenames starting with "." are reserved by this
- implementation, so no calendar names may start with ".".
- """
- self.assertRaises(
- HomeChildNameNotAllowedError,
- self.home1.createCalendarWithName, ".foo"
- )
-
-
- def test_removeCalendarWithName_dot(self):
- """
- Filenames starting with "." are reserved by this
- implementation, so no calendar names may start with ".".
- """
- name = ".foo"
- self.home1._path.child(name).createDirectory()
- self.assertRaises(
- NoSuchHomeChildError,
- self.home1.removeCalendarWithName, name
- )
-
-
-
-class CalendarTest(unittest.TestCase):
-
- notifierFactory = None
-
- def setUp(self):
- setUpCalendar1(self)
-
-
- def test_init(self):
- """
- L{Calendar.__init__} sets private attributes to reflect its constructor
- arguments.
- """
- self.failUnless(
- isinstance(self.calendar1._path, FilePath),
- self.calendar1
- )
- self.failUnless(
- isinstance(self.calendar1._calendarHome, CalendarHome),
- self.calendar1._calendarHome
- )
-
-
- def test_useIndexImmediately(self):
- """
- L{Calendar._index} is usable in the same transaction it is created, with
- a temporary filename.
- """
- self.home1.createCalendarWithName("calendar2")
- calendar = self.home1.calendarWithName("calendar2")
- index = calendar._index
- self.assertEquals(set(index.calendarObjects()),
- set(calendar.calendarObjects()))
- self.txn.commit()
- self.txn = self.calendarStore.newTransaction()
- self.home1 = self.txn.calendarHomeWithUID("home1")
- calendar = self.home1.calendarWithName("calendar2")
- # FIXME: we should be curating our own index here, but in order to fix
- # that the code in the old implicit scheduler needs to change. This
- # test would be more effective if there were actually some objects in
- # this list.
- index = calendar._index
- self.assertEquals(set(index.calendarObjects()),
- set(calendar.calendarObjects()))
-
-
- def test_calendarObjectWithName_dot(self):
- """
- Filenames starting with "." are reserved by this
- implementation, so no calendar object names may start with
- ".".
- """
- name = ".foo.ics"
- self.home1._path.child(name).touch()
- self.assertEquals(self.calendar1.calendarObjectWithName(name), None)
-
-
- @featureUnimplemented
- def test_calendarObjectWithUID_exists(self):
- """
- Find existing calendar object by name.
- """
- calendarObject = self.calendar1.calendarObjectWithUID("1")
- self.failUnless(
- isinstance(calendarObject, CalendarObject),
- calendarObject
- )
- self.assertEquals(
- calendarObject.component(),
- self.calendar1.calendarObjectWithName("1.ics").component()
- )
-
-
- def test_createCalendarObjectWithName_dot(self):
- """
- Filenames starting with "." are reserved by this
- implementation, so no calendar object names may start with
- ".".
- """
- self.assertRaises(
- ObjectResourceNameNotAllowedError,
- self.calendar1.createCalendarObjectWithName,
- ".foo", VComponent.fromString(event4_text)
- )
-
-
- @featureUnimplemented
- def test_createCalendarObjectWithName_uidconflict(self):
- """
- Attempt to create a calendar object with a conflicting UID
- should raise.
- """
- name = "foo.ics"
- assert self.calendar1.calendarObjectWithName(name) is None
- component = VComponent.fromString(event1modified_text)
- self.assertRaises(
- ObjectResourceUIDAlreadyExistsError,
- self.calendar1.createCalendarObjectWithName,
- name, component
- )
-
-
- def test_removeCalendarObject_delayedEffect(self):
- """
- Removing a calendar object should not immediately remove the underlying
- file; it should only be removed upon commit() of the transaction.
- """
- self.calendar1.removeCalendarObjectWithName("2.ics")
- self.failUnless(self.calendar1._path.child("2.ics").exists())
- self.txn.commit()
- self.failIf(self.calendar1._path.child("2.ics").exists())
-
-
- def test_removeCalendarObjectWithName_dot(self):
- """
- Filenames starting with "." are reserved by this
- implementation, so no calendar object names may start with
- ".".
- """
- name = ".foo"
- self.calendar1._path.child(name).touch()
- self.assertRaises(
- NoSuchObjectResourceError,
- self.calendar1.removeCalendarObjectWithName, name
- )
-
-
- def _refresh(self):
- """
- Re-read the (committed) home1 and calendar1 objects in a new
- transaction.
- """
- self.txn = self.calendarStore.newTransaction()
- self.home1 = self.txn.calendarHomeWithUID("home1")
- self.calendar1 = self.home1.calendarWithName("calendar_1")
-
-
- def test_undoCreateCalendarObject(self):
- """
- If a calendar object is created as part of a transaction, it will be
- removed if that transaction has to be aborted.
- """
- # Make sure that the calendar home is actually committed; rolling back
- # calendar home creation will remove the whole directory.
- self.txn.commit()
- self._refresh()
- self.calendar1.createCalendarObjectWithName(
- "sample.ics",
- VComponent.fromString(event4_text)
- )
- self._refresh()
- self.assertIdentical(
- self.calendar1.calendarObjectWithName("sample.ics"),
- None
- )
-
-
- def doThenUndo(self):
- """
- Commit the current transaction, but add an operation that will cause it
- to fail at the end. Finally, refresh all attributes with a new
- transaction so that further operations can be performed in a valid
- context.
- """
- def fail():
- raise RuntimeError("oops")
- self.txn.addOperation(fail, "dummy failing operation")
- self.assertRaises(RuntimeError, self.txn.commit)
- self._refresh()
-
-
- def test_undoModifyCalendarObject(self):
- """
- If an existing calendar object is modified as part of a transaction, it
- should be restored to its previous status if the transaction aborts.
- """
- originalComponent = self.calendar1.calendarObjectWithName(
- "1.ics").component()
- self.calendar1.calendarObjectWithName("1.ics").setComponent(
- VComponent.fromString(event1modified_text)
- )
- # Sanity check.
- self.assertEquals(
- self.calendar1.calendarObjectWithName("1.ics").component(),
- VComponent.fromString(event1modified_text)
- )
- self.doThenUndo()
- self.assertEquals(
- self.calendar1.calendarObjectWithName("1.ics").component(),
- originalComponent
- )
-
-
- def test_modifyCalendarObjectCaches(self):
- """
- Modifying a calendar object should cache the modified component in
- memory, to avoid unnecessary parsing round-trips.
- """
- modifiedComponent = VComponent.fromString(event1modified_text)
- self.calendar1.calendarObjectWithName("1.ics").setComponent(
- modifiedComponent
- )
- self.assertIdentical(
- modifiedComponent,
- self.calendar1.calendarObjectWithName("1.ics").component()
- )
-
-
- @featureUnimplemented
- def test_removeCalendarObjectWithUID_absent(self):
- """
- Attempt to remove an non-existing calendar object should raise.
- """
- self.assertRaises(
- NoSuchObjectResourceError,
- self.calendar1.removeCalendarObjectWithUID, "xyzzy"
- )
-
-
- @testUnimplemented
- def test_syncToken(self):
- """
- Sync token is correct.
- """
- raise NotImplementedError()
-
-
- @testUnimplemented
- def test_calendarObjectsInTimeRange(self):
- """
- Find calendar objects occuring in a given time range.
- """
- raise NotImplementedError()
-
-
- @testUnimplemented
- def test_calendarObjectsSinceToken(self):
- """
- Find calendar objects that have been modified since a given
- sync token.
- """
- raise NotImplementedError()
-
-
-
-class CalendarObjectTest(unittest.TestCase):
- notifierFactory = None
-
- def setUp(self):
- setUpCalendar1(self)
- self.object1 = self.calendar1.calendarObjectWithName("1.ics")
-
-
- def test_init(self):
- """
- L{CalendarObject} has instance attributes, C{_path} and C{_calendar},
- which refer to its position in the filesystem and the calendar in which
- it is contained, respectively.
- """
- self.failUnless(
- isinstance(self.object1._path, FilePath),
- self.object1._path
- )
- self.failUnless(
- isinstance(self.object1._calendar, Calendar),
- self.object1._calendar
- )
-
-
- def test_componentType(self):
- """
- Component type is correct.
- """
- self.assertEquals(self.object1.componentType(), "VEVENT")
-
-
-
-class FileStorageTests(CommonTests, unittest.TestCase):
- """
- File storage tests.
- """
-
- def storeUnderTest(self):
- """
- Create and return a L{CalendarStore} for testing.
- """
- setUpCalendarStore(self)
- return self.calendarStore
-
-
- def test_init(self):
- """
- L{CalendarStore} has a C{_path} attribute which refers to its
- constructor argument.
- """
- self.assertEquals(self.storeUnderTest()._path,
- self.storeRootPath)
-
-
- def test_calendarObjectsWithDotFile(self):
- """
- Adding a dotfile to the calendar home should not increase
- """
- self.homeUnderTest()._path.child(".foo").createDirectory()
- self.test_calendarObjects()
-
Copied: CalendarServer/trunk/txdav/caldav/datastore/test/test_file.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/test_file.py)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/test/test_file.py (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/test_file.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,470 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+File calendar store tests.
+"""
+
+# FIXME: all test cases in this file aside from FileStorageTests should be
+# deleted and replaced with either implementation-specific methods on
+# FileStorageTests, or implementation-agnostic methods on CommonTests.
+
+from twext.python.filepath import CachingFilePath as FilePath
+from twisted.trial import unittest
+
+from twext.python.vcomponent import VComponent
+
+from txdav.common.icommondatastore import HomeChildNameNotAllowedError
+from txdav.common.icommondatastore import ObjectResourceNameNotAllowedError
+from txdav.common.icommondatastore import ObjectResourceUIDAlreadyExistsError
+from txdav.common.icommondatastore import NoSuchHomeChildError
+from txdav.common.icommondatastore import NoSuchObjectResourceError
+
+from txdav.caldav.datastore.file import CalendarStore, CalendarHome
+from txdav.caldav.datastore.file import Calendar, CalendarObject
+
+from txdav.caldav.datastore.test.common import (
+ CommonTests, event4_text, event1modified_text)
+
+storePath = FilePath(__file__).parent().child("calendar_store")
+
+def _todo(f, why):
+ f.todo = why
+ return f
+
+
+
+featureUnimplemented = lambda f: _todo(f, "Feature unimplemented")
+testUnimplemented = lambda f: _todo(f, "Test unimplemented")
+todo = lambda why: lambda f: _todo(f, why)
+
+
+
+def setUpCalendarStore(test):
+ test.root = FilePath(test.mktemp())
+ test.root.createDirectory()
+
+ storeRootPath = test.storeRootPath = test.root.child("store")
+ calendarPath = storeRootPath.child("calendars").child("__uids__")
+ calendarPath.parent().makedirs()
+ storePath.copyTo(calendarPath)
+
+ test.calendarStore = CalendarStore(storeRootPath, test.notifierFactory)
+ test.txn = test.calendarStore.newTransaction()
+ assert test.calendarStore is not None, "No calendar store?"
+
+
+
+def setUpHome1(test):
+ setUpCalendarStore(test)
+ test.home1 = test.txn.calendarHomeWithUID("home1")
+ assert test.home1 is not None, "No calendar home?"
+
+
+
+def setUpCalendar1(test):
+ setUpHome1(test)
+ test.calendar1 = test.home1.calendarWithName("calendar_1")
+ assert test.calendar1 is not None, "No calendar?"
+
+
+
+class CalendarStoreTest(unittest.TestCase):
+ """
+ Test cases for L{CalendarStore}.
+ """
+
+ notifierFactory = None
+
+ def setUp(self):
+ setUpCalendarStore(self)
+
+
+ def test_calendarHomeWithUID_dot(self):
+ """
+ Filenames starting with "." are reserved by this
+ implementation, so no UIDs may start with ".".
+ """
+ self.assertEquals(
+ self.calendarStore.newTransaction().calendarHomeWithUID("xyzzy"),
+ None
+ )
+
+
+
+class CalendarHomeTest(unittest.TestCase):
+
+ notifierFactory = None
+ def setUp(self):
+ setUpHome1(self)
+
+
+ def test_init(self):
+ """
+ L{CalendarHome} has C{_path} and L{_calendarStore} attributes,
+ indicating its location on disk and parent store, respectively.
+ """
+ self.failUnless(
+ isinstance(self.home1._path, FilePath),
+ self.home1._path
+ )
+ self.assertEquals(
+ self.home1._calendarStore,
+ self.calendarStore
+ )
+
+
+ def test_calendarWithName_dot(self):
+ """
+ Filenames starting with "." are reserved by this
+ implementation, so no calendar names may start with ".".
+ """
+ name = ".foo"
+ self.home1._path.child(name).createDirectory()
+ self.assertEquals(self.home1.calendarWithName(name), None)
+
+
+ def test_createCalendarWithName_dot(self):
+ """
+ Filenames starting with "." are reserved by this
+ implementation, so no calendar names may start with ".".
+ """
+ self.assertRaises(
+ HomeChildNameNotAllowedError,
+ self.home1.createCalendarWithName, ".foo"
+ )
+
+
+ def test_removeCalendarWithName_dot(self):
+ """
+ Filenames starting with "." are reserved by this
+ implementation, so no calendar names may start with ".".
+ """
+ name = ".foo"
+ self.home1._path.child(name).createDirectory()
+ self.assertRaises(
+ NoSuchHomeChildError,
+ self.home1.removeCalendarWithName, name
+ )
+
+
+
+class CalendarTest(unittest.TestCase):
+
+ notifierFactory = None
+
+ def setUp(self):
+ setUpCalendar1(self)
+
+
+ def test_init(self):
+ """
+ L{Calendar.__init__} sets private attributes to reflect its constructor
+ arguments.
+ """
+ self.failUnless(
+ isinstance(self.calendar1._path, FilePath),
+ self.calendar1
+ )
+ self.failUnless(
+ isinstance(self.calendar1._calendarHome, CalendarHome),
+ self.calendar1._calendarHome
+ )
+
+
+ def test_useIndexImmediately(self):
+ """
+ L{Calendar._index} is usable in the same transaction it is created, with
+ a temporary filename.
+ """
+ self.home1.createCalendarWithName("calendar2")
+ calendar = self.home1.calendarWithName("calendar2")
+ index = calendar._index
+ self.assertEquals(set(index.calendarObjects()),
+ set(calendar.calendarObjects()))
+ self.txn.commit()
+ self.txn = self.calendarStore.newTransaction()
+ self.home1 = self.txn.calendarHomeWithUID("home1")
+ calendar = self.home1.calendarWithName("calendar2")
+ # FIXME: we should be curating our own index here, but in order to fix
+ # that the code in the old implicit scheduler needs to change. This
+ # test would be more effective if there were actually some objects in
+ # this list.
+ index = calendar._index
+ self.assertEquals(set(index.calendarObjects()),
+ set(calendar.calendarObjects()))
+
+
+ def test_calendarObjectWithName_dot(self):
+ """
+ Filenames starting with "." are reserved by this
+ implementation, so no calendar object names may start with
+ ".".
+ """
+ name = ".foo.ics"
+ self.home1._path.child(name).touch()
+ self.assertEquals(self.calendar1.calendarObjectWithName(name), None)
+
+
+ @featureUnimplemented
+ def test_calendarObjectWithUID_exists(self):
+ """
+ Find existing calendar object by name.
+ """
+ calendarObject = self.calendar1.calendarObjectWithUID("1")
+ self.failUnless(
+ isinstance(calendarObject, CalendarObject),
+ calendarObject
+ )
+ self.assertEquals(
+ calendarObject.component(),
+ self.calendar1.calendarObjectWithName("1.ics").component()
+ )
+
+
+ def test_createCalendarObjectWithName_dot(self):
+ """
+ Filenames starting with "." are reserved by this
+ implementation, so no calendar object names may start with
+ ".".
+ """
+ self.assertRaises(
+ ObjectResourceNameNotAllowedError,
+ self.calendar1.createCalendarObjectWithName,
+ ".foo", VComponent.fromString(event4_text)
+ )
+
+
+ @featureUnimplemented
+ def test_createCalendarObjectWithName_uidconflict(self):
+ """
+ Attempt to create a calendar object with a conflicting UID
+ should raise.
+ """
+ name = "foo.ics"
+ assert self.calendar1.calendarObjectWithName(name) is None
+ component = VComponent.fromString(event1modified_text)
+ self.assertRaises(
+ ObjectResourceUIDAlreadyExistsError,
+ self.calendar1.createCalendarObjectWithName,
+ name, component
+ )
+
+
+ def test_removeCalendarObject_delayedEffect(self):
+ """
+ Removing a calendar object should not immediately remove the underlying
+ file; it should only be removed upon commit() of the transaction.
+ """
+ self.calendar1.removeCalendarObjectWithName("2.ics")
+ self.failUnless(self.calendar1._path.child("2.ics").exists())
+ self.txn.commit()
+ self.failIf(self.calendar1._path.child("2.ics").exists())
+
+
+ def test_removeCalendarObjectWithName_dot(self):
+ """
+ Filenames starting with "." are reserved by this
+ implementation, so no calendar object names may start with
+ ".".
+ """
+ name = ".foo"
+ self.calendar1._path.child(name).touch()
+ self.assertRaises(
+ NoSuchObjectResourceError,
+ self.calendar1.removeCalendarObjectWithName, name
+ )
+
+
+ def _refresh(self):
+ """
+ Re-read the (committed) home1 and calendar1 objects in a new
+ transaction.
+ """
+ self.txn = self.calendarStore.newTransaction()
+ self.home1 = self.txn.calendarHomeWithUID("home1")
+ self.calendar1 = self.home1.calendarWithName("calendar_1")
+
+
+ def test_undoCreateCalendarObject(self):
+ """
+ If a calendar object is created as part of a transaction, it will be
+ removed if that transaction has to be aborted.
+ """
+ # Make sure that the calendar home is actually committed; rolling back
+ # calendar home creation will remove the whole directory.
+ self.txn.commit()
+ self._refresh()
+ self.calendar1.createCalendarObjectWithName(
+ "sample.ics",
+ VComponent.fromString(event4_text)
+ )
+ self._refresh()
+ self.assertIdentical(
+ self.calendar1.calendarObjectWithName("sample.ics"),
+ None
+ )
+
+
+ def doThenUndo(self):
+ """
+ Commit the current transaction, but add an operation that will cause it
+ to fail at the end. Finally, refresh all attributes with a new
+ transaction so that further operations can be performed in a valid
+ context.
+ """
+ def fail():
+ raise RuntimeError("oops")
+ self.txn.addOperation(fail, "dummy failing operation")
+ self.assertRaises(RuntimeError, self.txn.commit)
+ self._refresh()
+
+
+ def test_undoModifyCalendarObject(self):
+ """
+ If an existing calendar object is modified as part of a transaction, it
+ should be restored to its previous status if the transaction aborts.
+ """
+ originalComponent = self.calendar1.calendarObjectWithName(
+ "1.ics").component()
+ self.calendar1.calendarObjectWithName("1.ics").setComponent(
+ VComponent.fromString(event1modified_text)
+ )
+ # Sanity check.
+ self.assertEquals(
+ self.calendar1.calendarObjectWithName("1.ics").component(),
+ VComponent.fromString(event1modified_text)
+ )
+ self.doThenUndo()
+ self.assertEquals(
+ self.calendar1.calendarObjectWithName("1.ics").component(),
+ originalComponent
+ )
+
+
+ def test_modifyCalendarObjectCaches(self):
+ """
+ Modifying a calendar object should cache the modified component in
+ memory, to avoid unnecessary parsing round-trips.
+ """
+ modifiedComponent = VComponent.fromString(event1modified_text)
+ self.calendar1.calendarObjectWithName("1.ics").setComponent(
+ modifiedComponent
+ )
+ self.assertIdentical(
+ modifiedComponent,
+ self.calendar1.calendarObjectWithName("1.ics").component()
+ )
+
+
+ @featureUnimplemented
+ def test_removeCalendarObjectWithUID_absent(self):
+ """
+ Attempt to remove an non-existing calendar object should raise.
+ """
+ self.assertRaises(
+ NoSuchObjectResourceError,
+ self.calendar1.removeCalendarObjectWithUID, "xyzzy"
+ )
+
+
+ @testUnimplemented
+ def test_syncToken(self):
+ """
+ Sync token is correct.
+ """
+ raise NotImplementedError()
+
+
+ @testUnimplemented
+ def test_calendarObjectsInTimeRange(self):
+ """
+ Find calendar objects occuring in a given time range.
+ """
+ raise NotImplementedError()
+
+
+ @testUnimplemented
+ def test_calendarObjectsSinceToken(self):
+ """
+ Find calendar objects that have been modified since a given
+ sync token.
+ """
+ raise NotImplementedError()
+
+
+
+class CalendarObjectTest(unittest.TestCase):
+ notifierFactory = None
+
+ def setUp(self):
+ setUpCalendar1(self)
+ self.object1 = self.calendar1.calendarObjectWithName("1.ics")
+
+
+ def test_init(self):
+ """
+ L{CalendarObject} has instance attributes, C{_path} and C{_calendar},
+ which refer to its position in the filesystem and the calendar in which
+ it is contained, respectively.
+ """
+ self.failUnless(
+ isinstance(self.object1._path, FilePath),
+ self.object1._path
+ )
+ self.failUnless(
+ isinstance(self.object1._calendar, Calendar),
+ self.object1._calendar
+ )
+
+
+ def test_componentType(self):
+ """
+ Component type is correct.
+ """
+ self.assertEquals(self.object1.componentType(), "VEVENT")
+
+
+
+class FileStorageTests(CommonTests, unittest.TestCase):
+ """
+ File storage tests.
+ """
+
+ def storeUnderTest(self):
+ """
+ Create and return a L{CalendarStore} for testing.
+ """
+ setUpCalendarStore(self)
+ return self.calendarStore
+
+
+ def test_init(self):
+ """
+ L{CalendarStore} has a C{_path} attribute which refers to its
+ constructor argument.
+ """
+ self.assertEquals(self.storeUnderTest()._path,
+ self.storeRootPath)
+
+
+ def test_calendarObjectsWithDotFile(self):
+ """
+ Adding a dotfile to the calendar home should not increase
+ """
+ self.homeUnderTest()._path.child(".foo").createDirectory()
+ self.test_calendarObjects()
+
Deleted: CalendarServer/trunk/txdav/caldav/datastore/test/test_scheduling.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/test_scheduling.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/test_scheduling.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,48 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Tests for L{txdav.caldav.datastore.scheduling}.
-"""
-
-from twisted.trial.unittest import TestCase
-from txdav.caldav.datastore.test.common import CommonTests
-from txdav.caldav.datastore.test.test_file import setUpCalendarStore
-from txdav.caldav.datastore.scheduling import ImplicitStore
-
-simpleEvent = """BEGIN:VCALENDAR
-VERSION:2.0
-PRODID:-//CALENDARSERVER.ORG//NONSGML Version 1//EN
-BEGIN:VEVENT
-UID:12345-67890
-DTSTART:20080601T120000Z
-DTEND:20080601T130000Z
-ORGANIZER:mailto:user1 at example.com
-ATTENDEE:mailto:user1 at example.com
-ATTENDEE:mailto:user2 at example.com
-END:VEVENT
-END:VCALENDAR
-"""
-
-class ImplicitStoreTests(CommonTests, TestCase):
- """
- Tests for L{ImplicitSchedulingStore}.
- """
-
- def storeUnderTest(self):
- setUpCalendarStore(self)
- self.implicitStore = ImplicitStore(self.calendarStore)
- return self.implicitStore
Copied: CalendarServer/trunk/txdav/caldav/datastore/test/test_scheduling.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/test_scheduling.py)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/test/test_scheduling.py (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/test_scheduling.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,48 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Tests for L{txdav.caldav.datastore.scheduling}.
+"""
+
+from twisted.trial.unittest import TestCase
+from txdav.caldav.datastore.test.common import CommonTests
+from txdav.caldav.datastore.test.test_file import setUpCalendarStore
+from txdav.caldav.datastore.scheduling import ImplicitStore
+
+simpleEvent = """BEGIN:VCALENDAR
+VERSION:2.0
+PRODID:-//CALENDARSERVER.ORG//NONSGML Version 1//EN
+BEGIN:VEVENT
+UID:12345-67890
+DTSTART:20080601T120000Z
+DTEND:20080601T130000Z
+ORGANIZER:mailto:user1 at example.com
+ATTENDEE:mailto:user1 at example.com
+ATTENDEE:mailto:user2 at example.com
+END:VEVENT
+END:VCALENDAR
+"""
+
+class ImplicitStoreTests(CommonTests, TestCase):
+ """
+ Tests for L{ImplicitSchedulingStore}.
+ """
+
+ def storeUnderTest(self):
+ setUpCalendarStore(self)
+ self.implicitStore = ImplicitStore(self.calendarStore)
+ return self.implicitStore
Deleted: CalendarServer/trunk/txdav/caldav/datastore/test/test_sql.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/test_sql.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/test_sql.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,78 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Tests for txdav.caldav.datastore.postgres, mostly based on
-L{txdav.caldav.datastore.test.common}.
-"""
-
-from txdav.caldav.datastore.test.common import CommonTests as CalendarCommonTests
-
-from txdav.common.datastore.test.util import SQLStoreBuilder
-from txdav.common.icommondatastore import NoSuchHomeChildError
-
-from twisted.trial import unittest
-from twisted.internet.defer import inlineCallbacks
-from twext.python.vcomponent import VComponent
-
-
-theStoreBuilder = SQLStoreBuilder()
-buildStore = theStoreBuilder.buildStore
-
-class CalendarSQLStorageTests(CalendarCommonTests, unittest.TestCase):
- """
- Calendar SQL storage tests.
- """
-
- @inlineCallbacks
- def setUp(self):
- super(CalendarSQLStorageTests, self).setUp()
- self.calendarStore = yield buildStore(self, self.notifierFactory)
- self.populate()
-
-
- def populate(self):
- populateTxn = self.calendarStore.newTransaction()
- for homeUID in self.requirements:
- calendars = self.requirements[homeUID]
- if calendars is not None:
- home = populateTxn.calendarHomeWithUID(homeUID, True)
- # We don't want the default calendar or inbox to appear unless it's
- # explicitly listed.
- try:
- home.removeCalendarWithName("calendar")
- home.removeCalendarWithName("inbox")
- except NoSuchHomeChildError:
- pass
- for calendarName in calendars:
- calendarObjNames = calendars[calendarName]
- if calendarObjNames is not None:
- home.createCalendarWithName(calendarName)
- calendar = home.calendarWithName(calendarName)
- for objectName in calendarObjNames:
- objData = calendarObjNames[objectName]
- calendar.createCalendarObjectWithName(
- objectName, VComponent.fromString(objData)
- )
- populateTxn.commit()
- self.notifierFactory.reset()
-
-
- def storeUnderTest(self):
- """
- Create and return a L{CalendarStore} for testing.
- """
- return self.calendarStore
Copied: CalendarServer/trunk/txdav/caldav/datastore/test/test_sql.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/test/test_sql.py)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/test/test_sql.py (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/test/test_sql.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,78 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Tests for txdav.caldav.datastore.postgres, mostly based on
+L{txdav.caldav.datastore.test.common}.
+"""
+
+from txdav.caldav.datastore.test.common import CommonTests as CalendarCommonTests
+
+from txdav.common.datastore.test.util import SQLStoreBuilder
+from txdav.common.icommondatastore import NoSuchHomeChildError
+
+from twisted.trial import unittest
+from twisted.internet.defer import inlineCallbacks
+from twext.python.vcomponent import VComponent
+
+
+theStoreBuilder = SQLStoreBuilder()
+buildStore = theStoreBuilder.buildStore
+
+class CalendarSQLStorageTests(CalendarCommonTests, unittest.TestCase):
+ """
+ Calendar SQL storage tests.
+ """
+
+ @inlineCallbacks
+ def setUp(self):
+ super(CalendarSQLStorageTests, self).setUp()
+ self.calendarStore = yield buildStore(self, self.notifierFactory)
+ self.populate()
+
+
+ def populate(self):
+ populateTxn = self.calendarStore.newTransaction()
+ for homeUID in self.requirements:
+ calendars = self.requirements[homeUID]
+ if calendars is not None:
+ home = populateTxn.calendarHomeWithUID(homeUID, True)
+ # We don't want the default calendar or inbox to appear unless it's
+ # explicitly listed.
+ try:
+ home.removeCalendarWithName("calendar")
+ home.removeCalendarWithName("inbox")
+ except NoSuchHomeChildError:
+ pass
+ for calendarName in calendars:
+ calendarObjNames = calendars[calendarName]
+ if calendarObjNames is not None:
+ home.createCalendarWithName(calendarName)
+ calendar = home.calendarWithName(calendarName)
+ for objectName in calendarObjNames:
+ objData = calendarObjNames[objectName]
+ calendar.createCalendarObjectWithName(
+ objectName, VComponent.fromString(objData)
+ )
+ populateTxn.commit()
+ self.notifierFactory.reset()
+
+
+ def storeUnderTest(self):
+ """
+ Create and return a L{CalendarStore} for testing.
+ """
+ return self.calendarStore
Deleted: CalendarServer/trunk/txdav/caldav/datastore/util.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/util.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/datastore/util.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,88 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-"""
-Utility logic common to multiple backend implementations.
-"""
-
-from twext.python.vcomponent import InvalidICalendarDataError
-from twext.python.vcomponent import VComponent
-
-from txdav.common.icommondatastore import InvalidObjectResourceError,\
- NoSuchObjectResourceError
-
-
-def validateCalendarComponent(calendarObject, calendar, component, inserting):
- """
- Validate a calendar component for a particular calendar.
-
- @param calendarObject: The calendar object whose component will be replaced.
- @type calendarObject: L{ICalendarObject}
-
- @param calendar: The calendar which the L{ICalendarObject} is present in.
- @type calendar: L{ICalendar}
-
- @param component: The VComponent to be validated.
- @type component: L{VComponent}
- """
-
- if not isinstance(component, VComponent):
- raise TypeError(type(component))
-
- try:
- if not inserting and component.resourceUID() != calendarObject.uid():
- raise InvalidObjectResourceError(
- "UID may not change (%s != %s)" % (
- component.resourceUID(), calendarObject.uid()
- )
- )
- except NoSuchObjectResourceError:
- pass
-
- try:
- # FIXME: This is a bad way to do this test, there should be a
- # Calendar-level API for it.
- if calendar.name() == 'inbox':
- component.validateComponentsForCalDAV(True)
- else:
- component.validateForCalDAV()
- except InvalidICalendarDataError, e:
- raise InvalidObjectResourceError(e)
-
-
-def dropboxIDFromCalendarObject(calendarObject):
- """
- Helper to implement L{ICalendarObject.dropboxID}.
-
- @param calendarObject: The calendar object to retrieve a dropbox ID for.
- @type calendarObject: L{ICalendarObject}
- """
- dropboxProperty = calendarObject.component(
- ).getFirstPropertyInAnyComponent("X-APPLE-DROPBOX")
- if dropboxProperty is not None:
- componentDropboxID = dropboxProperty.value().split("/")[-1]
- return componentDropboxID
- attachProperty = calendarObject.component().getFirstPropertyInAnyComponent("ATTACH")
- if attachProperty is not None:
- # Make sure the value type is URI
- valueType = attachProperty.params().get("VALUE", ("TEXT",))
- if valueType[0] == "URI":
- # FIXME: more aggressive checking to see if this URI is really the
- # 'right' URI. Maybe needs to happen in the front end.
- attachPath = attachProperty.value().split("/")[-2]
- return attachPath
-
- return calendarObject.uid() + ".dropbox"
-
\ No newline at end of file
Copied: CalendarServer/trunk/txdav/caldav/datastore/util.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/datastore/util.py)
===================================================================
--- CalendarServer/trunk/txdav/caldav/datastore/util.py (rev 0)
+++ CalendarServer/trunk/txdav/caldav/datastore/util.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,88 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+"""
+Utility logic common to multiple backend implementations.
+"""
+
+from twext.python.vcomponent import InvalidICalendarDataError
+from twext.python.vcomponent import VComponent
+
+from txdav.common.icommondatastore import InvalidObjectResourceError,\
+ NoSuchObjectResourceError
+
+
+def validateCalendarComponent(calendarObject, calendar, component, inserting):
+ """
+ Validate a calendar component for a particular calendar.
+
+ @param calendarObject: The calendar object whose component will be replaced.
+ @type calendarObject: L{ICalendarObject}
+
+ @param calendar: The calendar which the L{ICalendarObject} is present in.
+ @type calendar: L{ICalendar}
+
+ @param component: The VComponent to be validated.
+ @type component: L{VComponent}
+ """
+
+ if not isinstance(component, VComponent):
+ raise TypeError(type(component))
+
+ try:
+ if not inserting and component.resourceUID() != calendarObject.uid():
+ raise InvalidObjectResourceError(
+ "UID may not change (%s != %s)" % (
+ component.resourceUID(), calendarObject.uid()
+ )
+ )
+ except NoSuchObjectResourceError:
+ pass
+
+ try:
+ # FIXME: This is a bad way to do this test, there should be a
+ # Calendar-level API for it.
+ if calendar.name() == 'inbox':
+ component.validateComponentsForCalDAV(True)
+ else:
+ component.validateForCalDAV()
+ except InvalidICalendarDataError, e:
+ raise InvalidObjectResourceError(e)
+
+
+def dropboxIDFromCalendarObject(calendarObject):
+ """
+ Helper to implement L{ICalendarObject.dropboxID}.
+
+ @param calendarObject: The calendar object to retrieve a dropbox ID for.
+ @type calendarObject: L{ICalendarObject}
+ """
+ dropboxProperty = calendarObject.component(
+ ).getFirstPropertyInAnyComponent("X-APPLE-DROPBOX")
+ if dropboxProperty is not None:
+ componentDropboxID = dropboxProperty.value().split("/")[-1]
+ return componentDropboxID
+ attachProperty = calendarObject.component().getFirstPropertyInAnyComponent("ATTACH")
+ if attachProperty is not None:
+ # Make sure the value type is URI
+ valueType = attachProperty.params().get("VALUE", ("TEXT",))
+ if valueType[0] == "URI":
+ # FIXME: more aggressive checking to see if this URI is really the
+ # 'right' URI. Maybe needs to happen in the front end.
+ attachPath = attachProperty.value().split("/")[-2]
+ return attachPath
+
+ return calendarObject.uid() + ".dropbox"
+
\ No newline at end of file
Deleted: CalendarServer/trunk/txdav/caldav/icalendarstore.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/icalendarstore.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/icalendarstore.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,428 +0,0 @@
-# -*- test-case-name: txdav.caldav.datastore -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Calendar store interfaces
-"""
-
-from txdav.common.icommondatastore import ICommonTransaction,\
- IShareableCollection
-from txdav.idav import IDataStoreResource
-
-from txdav.idav import INotifier
-
-
-__all__ = [
- # Classes
- "ICalendarTransaction",
- "ICalendarHome",
- "ICalendar",
- "ICalendarObject",
-]
-
-
-# The following imports are used by the L{} links below, but shouldn't actually
-# be imported.as they're not really needed.
-
-# from datetime import datetime, date, tzinfo
-
-# from twext.python.vcomponent import VComponent
-
-# from txdav.idav import IPropertyStore
-# from txdav.idav import ITransaction
-
-class ICalendarTransaction(ICommonTransaction):
- """
- Transaction functionality required to be implemented by calendar stores.
- """
-
- def calendarHomeWithUID(uid, create=False):
- """
- Retrieve the calendar home for the principal with the given C{uid}.
-
- If C{create} is C{True}, create the calendar home if it doesn't
- already exist.
-
- @return: an L{ICalendarHome} or C{None} if no such calendar
- home exists.
- """
-
-
-#
-# Interfaces
-#
-
-class ICalendarHome(INotifier, IDataStoreResource):
- """
- An L{ICalendarHome} is a collection of calendars which belongs to a
- specific principal and contains the calendars which that principal has
- direct access to. This includes both calendars owned by the principal as
- well as calendars that have been shared with and accepts by the principal.
- """
-
- def uid():
- """
- Retrieve the unique identifier for this calendar home.
-
- @return: a string.
- """
-
- def calendars():
- """
- Retrieve calendars contained in this calendar home.
-
- @return: an iterable of L{ICalendar}s.
- """
-
- def calendarWithName(name):
- """
- Retrieve the calendar with the given C{name} contained in this
- calendar home.
-
- @param name: a string.
- @return: an L{ICalendar} or C{None} if no such calendar
- exists.
- """
-
-
- def calendarObjectWithDropboxID(dropboxID):
- """
- Retrieve an L{ICalendarObject} by looking up its attachment collection
- ID.
-
- @param dropboxID: The name of the collection in a dropbox corresponding
- to a collection in the user's dropbox.
-
- @type dropboxID: C{str}
-
- @return: the calendar object identified by the given dropbox.
-
- @rtype: L{ICalendarObject}
- """
-
-
- def createCalendarWithName(name):
- """
- Create a calendar with the given C{name} in this calendar
- home.
-
- @param name: a string.
- @raise CalendarAlreadyExistsError: if a calendar with the
- given C{name} already exists.
- """
-
- def removeCalendarWithName(name):
- """
- Remove the calendar with the given C{name} from this calendar
- home. If this calendar home owns the calendar, also remove
- the calendar from all calendar homes.
-
- @param name: a string.
- @raise NoSuchCalendarObjectError: if no such calendar exists.
-
- @return: an L{IPropertyStore}.
- """
-
-
-class ICalendar(INotifier, IShareableCollection, IDataStoreResource):
- """
- Calendar
-
- A calendar is a container for calendar objects (events, to-dos,
- etc.). A calendar belongs to a specific principal but may be
- shared with other principals, granting them read-only or
- read/write access.
- """
-
- def rename(name):
- """
- Change the name of this calendar.
- """
-
- def ownerCalendarHome():
- """
- Retrieve the calendar home for the owner of this calendar.
- Calendars may be shared from one (the owner's) calendar home
- to other (the sharee's) calendar homes.
-
- @return: an L{ICalendarHome}.
- """
-
- def calendarObjects():
- """
- Retrieve the calendar objects contained in this calendar.
-
- @return: an iterable of L{ICalendarObject}s.
- """
-
- def calendarObjectWithName(name):
- """
- Retrieve the calendar object with the given C{name} contained
- in this calendar.
-
- @param name: a string.
- @return: an L{ICalendarObject} or C{None} if no such calendar
- object exists.
- """
-
- def calendarObjectWithUID(uid):
- """
- Retrieve the calendar object with the given C{uid} contained
- in this calendar.
-
- @param uid: a string.
- @return: an L{ICalendarObject} or C{None} if no such calendar
- object exists.
- """
-
- def createCalendarObjectWithName(name, component):
- """
- Create a calendar component with the given C{name} in this
- calendar from the given C{component}.
-
- @param name: a string.
- @param component: a C{VCALENDAR} L{Component}
- @raise ObjectResourceNameAlreadyExistsError: if a calendar
- object with the given C{name} already exists.
- @raise CalendarObjectUIDAlreadyExistsError: if a calendar
- object with the same UID as the given C{component} already
- exists.
- @raise InvalidCalendarComponentError: if the given
- C{component} is not a valid C{VCALENDAR} L{VComponent} for
- a calendar object.
- """
-
- def removeCalendarObjectWithName(name):
- """
- Remove the calendar object with the given C{name} from this
- calendar.
-
- @param name: a string.
- @raise NoSuchCalendarObjectError: if no such calendar object
- exists.
- """
-
- def removeCalendarObjectWithUID(uid):
- """
- Remove the calendar object with the given C{uid} from this
- calendar.
-
- @param uid: a string.
- @raise NoSuchCalendarObjectError: if the calendar object does
- not exist.
- """
-
- def syncToken():
- """
- Retrieve the current sync token for this calendar.
-
- @return: a string containing a sync token.
- """
-
- def calendarObjectsInTimeRange(start, end, timeZone):
- """
- Retrieve all calendar objects in this calendar which have
- instances that occur within the time range that begins at
- C{start} and ends at C{end}.
-
- @param start: a L{datetime} or L{date}.
- @param end: a L{datetime} or L{date}.
- @param timeZone: a L{tzinfo}.
- @return: an iterable of L{ICalendarObject}s.
- """
-
- def calendarObjectsSinceToken(token):
- """
- Retrieve all calendar objects in this calendar that have
- changed since the given C{token} was last valid.
-
- @param token: a sync token.
- @return: a 3-tuple containing an iterable of
- L{ICalendarObject}s that have changed, an iterable of uids
- that have been removed, and the current sync token.
- """
-
-
-class ICalendarObject(IDataStoreResource):
- """
- Calendar object
-
- A calendar object describes an event, to-do, or other iCalendar
- object.
- """
-
- def calendar():
- """
- @return: The calendar which this calendar object is a part of.
- @rtype: L{ICalendar}
- """
-
- def setComponent(component):
- """
- Rewrite this calendar object to match the given C{component}.
- C{component} must have the same UID and be of the same
- component type as this calendar object.
-
- @param component: a C{VCALENDAR} L{VComponent}.
- @raise InvalidCalendarComponentError: if the given
- C{component} is not a valid C{VCALENDAR} L{VComponent} for
- a calendar object.
- """
-
- def component():
- """
- Retrieve the calendar component for this calendar object.
-
- @return: a C{VCALENDAR} L{VComponent}.
- """
-
- def iCalendarText():
- """
- Retrieve the iCalendar text data for this calendar object.
-
- @return: a string containing iCalendar data for a single
- calendar object.
- """
-
- def uid():
- """
- Retrieve the UID for this calendar object.
-
- @return: a string containing a UID.
- """
-
- def componentType():
- """
- Retrieve the iCalendar component type for the main component
- in this calendar object.
-
- @return: a string containing the component type.
- """
-
- def organizer():
- # FIXME: Ideally should return a URI object
- """
- Retrieve the organizer's calendar user address for this
- calendar object.
-
- @return: a URI string.
- """
-
- def dropboxID():
- """
- An identifier, unique to the calendar home, that specifies a location
- where attachments are to be stored for this object.
-
- @return: the value of the last segment of the C{X-APPLE-DROPBOX}
- property.
-
- @rtype: C{string}
- """
-
-
- def createAttachmentWithName(name, contentType):
- """
- Add an attachment to this calendar object.
-
- @param name: An identifier, unique to this L{ICalendarObject}, which
- names the attachment for future retrieval.
-
- @type name: C{str}
-
- @param contentType: a slash-separated content type.
-
- @type contentType: C{str}
-
- @return: the same type as L{IAttachment.store} returns.
- """
-
-
- def attachmentWithName(name):
- """
- Retrieve an attachment from this calendar object.
-
- @param name: An identifier, unique to this L{ICalendarObject}, which
- names the attachment for future retrieval.
-
- @type name: C{str}
- """
- # FIXME: MIME-type?
-
-
- def attachments():
- """
- List all attachments on this calendar object.
-
- @return: an iterable of L{IAttachment}s
- """
-
-
- def removeAttachmentWithName(name):
- """
- Delete an attachment with the given name.
-
- @param name: The basename of the attachment (i.e. the last segment of
- its URI) as given to L{attachmentWithName}.
- @type name: C{str}
- """
-
-
- def attendeesCanManageAttachments():
- """
- Are attendees allowed to manage attachments?
-
- @return: C{True} if they can, C{False} if they can't.
- """
-
-
-
-class IAttachment(IDataStoreResource):
- """
- Information associated with an attachment to a calendar object.
- """
-
- def store(contentType):
- """
- @param contentType: The content type of the data which will be stored.
- @type contentType: C{str}
-
- @return: An L{ITransport}/L{IConsumer} provider that will store the
- bytes passed to its 'write' method.
-
- The caller of C{store} must call C{loseConnection} on its result to
- indicate that the attachment upload was successfully completed. If
- the transaction associated with this upload is committed or aborted
- before C{loseConnection} is called, the upload will be presumed to
- have failed, and no attachment data will be stored.
- """
- # If you do a big write()/loseConnection(), how do you tell when the
- # data has actually been written? you don't: commit() ought to return
- # a deferred anyway, and any un-flushed attachment data needs to be
- # dealt with by that too.
-
-
- def retrieve(protocol):
- """
- Retrieve the content of this attachment into a protocol instance.
-
- @param protocol: A protocol which will receive the contents of the
- attachment to its C{dataReceived} method, and then a notification
- that the stream is complete to its C{connectionLost} method.
- @type protocol: L{IProtocol}
- """
-
-
Copied: CalendarServer/trunk/txdav/caldav/icalendarstore.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/icalendarstore.py)
===================================================================
--- CalendarServer/trunk/txdav/caldav/icalendarstore.py (rev 0)
+++ CalendarServer/trunk/txdav/caldav/icalendarstore.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,428 @@
+# -*- test-case-name: txdav.caldav.datastore -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Calendar store interfaces
+"""
+
+from txdav.common.icommondatastore import ICommonTransaction,\
+ IShareableCollection
+from txdav.idav import IDataStoreResource
+
+from txdav.idav import INotifier
+
+
+__all__ = [
+ # Classes
+ "ICalendarTransaction",
+ "ICalendarHome",
+ "ICalendar",
+ "ICalendarObject",
+]
+
+
+# The following imports are used by the L{} links below, but shouldn't actually
+# be imported.as they're not really needed.
+
+# from datetime import datetime, date, tzinfo
+
+# from twext.python.vcomponent import VComponent
+
+# from txdav.idav import IPropertyStore
+# from txdav.idav import ITransaction
+
+class ICalendarTransaction(ICommonTransaction):
+ """
+ Transaction functionality required to be implemented by calendar stores.
+ """
+
+ def calendarHomeWithUID(uid, create=False):
+ """
+ Retrieve the calendar home for the principal with the given C{uid}.
+
+ If C{create} is C{True}, create the calendar home if it doesn't
+ already exist.
+
+ @return: an L{ICalendarHome} or C{None} if no such calendar
+ home exists.
+ """
+
+
+#
+# Interfaces
+#
+
+class ICalendarHome(INotifier, IDataStoreResource):
+ """
+ An L{ICalendarHome} is a collection of calendars which belongs to a
+ specific principal and contains the calendars which that principal has
+ direct access to. This includes both calendars owned by the principal as
+ well as calendars that have been shared with and accepts by the principal.
+ """
+
+ def uid():
+ """
+ Retrieve the unique identifier for this calendar home.
+
+ @return: a string.
+ """
+
+ def calendars():
+ """
+ Retrieve calendars contained in this calendar home.
+
+ @return: an iterable of L{ICalendar}s.
+ """
+
+ def calendarWithName(name):
+ """
+ Retrieve the calendar with the given C{name} contained in this
+ calendar home.
+
+ @param name: a string.
+ @return: an L{ICalendar} or C{None} if no such calendar
+ exists.
+ """
+
+
+ def calendarObjectWithDropboxID(dropboxID):
+ """
+ Retrieve an L{ICalendarObject} by looking up its attachment collection
+ ID.
+
+ @param dropboxID: The name of the collection in a dropbox corresponding
+ to a collection in the user's dropbox.
+
+ @type dropboxID: C{str}
+
+ @return: the calendar object identified by the given dropbox.
+
+ @rtype: L{ICalendarObject}
+ """
+
+
+ def createCalendarWithName(name):
+ """
+ Create a calendar with the given C{name} in this calendar
+ home.
+
+ @param name: a string.
+ @raise CalendarAlreadyExistsError: if a calendar with the
+ given C{name} already exists.
+ """
+
+ def removeCalendarWithName(name):
+ """
+ Remove the calendar with the given C{name} from this calendar
+ home. If this calendar home owns the calendar, also remove
+ the calendar from all calendar homes.
+
+ @param name: a string.
+ @raise NoSuchCalendarObjectError: if no such calendar exists.
+
+ @return: an L{IPropertyStore}.
+ """
+
+
+class ICalendar(INotifier, IShareableCollection, IDataStoreResource):
+ """
+ Calendar
+
+ A calendar is a container for calendar objects (events, to-dos,
+ etc.). A calendar belongs to a specific principal but may be
+ shared with other principals, granting them read-only or
+ read/write access.
+ """
+
+ def rename(name):
+ """
+ Change the name of this calendar.
+ """
+
+ def ownerCalendarHome():
+ """
+ Retrieve the calendar home for the owner of this calendar.
+ Calendars may be shared from one (the owner's) calendar home
+ to other (the sharee's) calendar homes.
+
+ @return: an L{ICalendarHome}.
+ """
+
+ def calendarObjects():
+ """
+ Retrieve the calendar objects contained in this calendar.
+
+ @return: an iterable of L{ICalendarObject}s.
+ """
+
+ def calendarObjectWithName(name):
+ """
+ Retrieve the calendar object with the given C{name} contained
+ in this calendar.
+
+ @param name: a string.
+ @return: an L{ICalendarObject} or C{None} if no such calendar
+ object exists.
+ """
+
+ def calendarObjectWithUID(uid):
+ """
+ Retrieve the calendar object with the given C{uid} contained
+ in this calendar.
+
+ @param uid: a string.
+ @return: an L{ICalendarObject} or C{None} if no such calendar
+ object exists.
+ """
+
+ def createCalendarObjectWithName(name, component):
+ """
+ Create a calendar component with the given C{name} in this
+ calendar from the given C{component}.
+
+ @param name: a string.
+ @param component: a C{VCALENDAR} L{Component}
+ @raise ObjectResourceNameAlreadyExistsError: if a calendar
+ object with the given C{name} already exists.
+ @raise CalendarObjectUIDAlreadyExistsError: if a calendar
+ object with the same UID as the given C{component} already
+ exists.
+ @raise InvalidCalendarComponentError: if the given
+ C{component} is not a valid C{VCALENDAR} L{VComponent} for
+ a calendar object.
+ """
+
+ def removeCalendarObjectWithName(name):
+ """
+ Remove the calendar object with the given C{name} from this
+ calendar.
+
+ @param name: a string.
+ @raise NoSuchCalendarObjectError: if no such calendar object
+ exists.
+ """
+
+ def removeCalendarObjectWithUID(uid):
+ """
+ Remove the calendar object with the given C{uid} from this
+ calendar.
+
+ @param uid: a string.
+ @raise NoSuchCalendarObjectError: if the calendar object does
+ not exist.
+ """
+
+ def syncToken():
+ """
+ Retrieve the current sync token for this calendar.
+
+ @return: a string containing a sync token.
+ """
+
+ def calendarObjectsInTimeRange(start, end, timeZone):
+ """
+ Retrieve all calendar objects in this calendar which have
+ instances that occur within the time range that begins at
+ C{start} and ends at C{end}.
+
+ @param start: a L{datetime} or L{date}.
+ @param end: a L{datetime} or L{date}.
+ @param timeZone: a L{tzinfo}.
+ @return: an iterable of L{ICalendarObject}s.
+ """
+
+ def calendarObjectsSinceToken(token):
+ """
+ Retrieve all calendar objects in this calendar that have
+ changed since the given C{token} was last valid.
+
+ @param token: a sync token.
+ @return: a 3-tuple containing an iterable of
+ L{ICalendarObject}s that have changed, an iterable of uids
+ that have been removed, and the current sync token.
+ """
+
+
+class ICalendarObject(IDataStoreResource):
+ """
+ Calendar object
+
+ A calendar object describes an event, to-do, or other iCalendar
+ object.
+ """
+
+ def calendar():
+ """
+ @return: The calendar which this calendar object is a part of.
+ @rtype: L{ICalendar}
+ """
+
+ def setComponent(component):
+ """
+ Rewrite this calendar object to match the given C{component}.
+ C{component} must have the same UID and be of the same
+ component type as this calendar object.
+
+ @param component: a C{VCALENDAR} L{VComponent}.
+ @raise InvalidCalendarComponentError: if the given
+ C{component} is not a valid C{VCALENDAR} L{VComponent} for
+ a calendar object.
+ """
+
+ def component():
+ """
+ Retrieve the calendar component for this calendar object.
+
+ @return: a C{VCALENDAR} L{VComponent}.
+ """
+
+ def iCalendarText():
+ """
+ Retrieve the iCalendar text data for this calendar object.
+
+ @return: a string containing iCalendar data for a single
+ calendar object.
+ """
+
+ def uid():
+ """
+ Retrieve the UID for this calendar object.
+
+ @return: a string containing a UID.
+ """
+
+ def componentType():
+ """
+ Retrieve the iCalendar component type for the main component
+ in this calendar object.
+
+ @return: a string containing the component type.
+ """
+
+ def organizer():
+ # FIXME: Ideally should return a URI object
+ """
+ Retrieve the organizer's calendar user address for this
+ calendar object.
+
+ @return: a URI string.
+ """
+
+ def dropboxID():
+ """
+ An identifier, unique to the calendar home, that specifies a location
+ where attachments are to be stored for this object.
+
+ @return: the value of the last segment of the C{X-APPLE-DROPBOX}
+ property.
+
+ @rtype: C{string}
+ """
+
+
+ def createAttachmentWithName(name, contentType):
+ """
+ Add an attachment to this calendar object.
+
+ @param name: An identifier, unique to this L{ICalendarObject}, which
+ names the attachment for future retrieval.
+
+ @type name: C{str}
+
+ @param contentType: a slash-separated content type.
+
+ @type contentType: C{str}
+
+ @return: the same type as L{IAttachment.store} returns.
+ """
+
+
+ def attachmentWithName(name):
+ """
+ Retrieve an attachment from this calendar object.
+
+ @param name: An identifier, unique to this L{ICalendarObject}, which
+ names the attachment for future retrieval.
+
+ @type name: C{str}
+ """
+ # FIXME: MIME-type?
+
+
+ def attachments():
+ """
+ List all attachments on this calendar object.
+
+ @return: an iterable of L{IAttachment}s
+ """
+
+
+ def removeAttachmentWithName(name):
+ """
+ Delete an attachment with the given name.
+
+ @param name: The basename of the attachment (i.e. the last segment of
+ its URI) as given to L{attachmentWithName}.
+ @type name: C{str}
+ """
+
+
+ def attendeesCanManageAttachments():
+ """
+ Are attendees allowed to manage attachments?
+
+ @return: C{True} if they can, C{False} if they can't.
+ """
+
+
+
+class IAttachment(IDataStoreResource):
+ """
+ Information associated with an attachment to a calendar object.
+ """
+
+ def store(contentType):
+ """
+ @param contentType: The content type of the data which will be stored.
+ @type contentType: C{str}
+
+ @return: An L{ITransport}/L{IConsumer} provider that will store the
+ bytes passed to its 'write' method.
+
+ The caller of C{store} must call C{loseConnection} on its result to
+ indicate that the attachment upload was successfully completed. If
+ the transaction associated with this upload is committed or aborted
+ before C{loseConnection} is called, the upload will be presumed to
+ have failed, and no attachment data will be stored.
+ """
+ # If you do a big write()/loseConnection(), how do you tell when the
+ # data has actually been written? you don't: commit() ought to return
+ # a deferred anyway, and any un-flushed attachment data needs to be
+ # dealt with by that too.
+
+
+ def retrieve(protocol):
+ """
+ Retrieve the content of this attachment into a protocol instance.
+
+ @param protocol: A protocol which will receive the contents of the
+ attachment to its C{dataReceived} method, and then a notification
+ that the stream is complete to its C{connectionLost} method.
+ @type protocol: L{IProtocol}
+ """
+
+
Deleted: CalendarServer/trunk/txdav/caldav/resource.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/caldav/resource.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/caldav/resource.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,156 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-CalDAV resources.
-"""
-
-__all__ = [
- "CalDAVResource",
- "CalendarHomeResource",
- "CalendarCollectionResource",
- "CalendarObjectResource",
- "ScheduleInboxResource",
- "ScheduleOutboxResource",
-]
-
-
-import urllib
-
-from twext.python.log import LoggingMixIn
-from twext.web2.dav.element.base import dav_namespace
-from twext.web2.http_headers import MimeType
-from twext.web2.http import RedirectResponse, Response
-from twext.web2.stream import MemoryStream
-
-from twistedcaldav import caldavxml
-from twistedcaldav.caldavxml import caldav_namespace
-from twistedcaldav.config import config
-from twistedcaldav.extensions import DAVResource
-
-
-class CalDAVResource(DAVResource, LoggingMixIn):
- """
- CalDAV resource.
- """
- def davComplianceClasses(self):
- return (
- tuple(super(CalDAVResource, self).davComplianceClasses())
- + config.CalDAVComplianceClasses
- )
-
- supportedCalendarComponentSet = caldavxml.SupportedCalendarComponentSet(
- *[caldavxml.CalendarComponent(name=item) for item in allowedComponents]
- )
-
-
-class CalendarHomeResource(CalDAVResource):
- """
- Calendar home resource.
-
- This resource is backed by an L{ICalendarHome} implementation.
- """
-
-
-class CalendarCollectionResource(CalDAVResource):
- """
- Calendar collection resource.
-
- This resource is backed by an L{ICalendar} implementation.
- """
- #
- # HTTP
- #
-
- def render(self, request):
- if config.EnableMonolithicCalendars:
- #
- # Send listing instead of iCalendar data to HTML agents
- # This is mostly useful for debugging...
- #
- # FIXME: Add a self-link to the dirlist with a query string so
- # users can still download the actual iCalendar data?
- #
- # FIXME: Are there better ways to detect this than hacking in
- # user agents?
- #
- # FIXME: In the meantime, make this a configurable regex list?
- #
- agent = request.headers.getHeader("user-agent")
- if agent is not None and (
- agent.startswith("Mozilla/") and agent.find("Gecko") != -1
- ):
- renderAsHTML = True
- else:
- renderAsHTML = False
- else:
- renderAsHTML = True
-
- if not renderAsHTML:
- # Render a monolithic iCalendar file
- if request.path[-1] != "/":
- # Redirect to include trailing '/' in URI
- return RedirectResponse(request.unparseURL(path=urllib.quote(urllib.unquote(request.path), safe=':/')+'/'))
-
- def _defer(data):
- response = Response()
- response.stream = MemoryStream(str(data))
- response.headers.setHeader("content-type", MimeType.fromString("text/calendar"))
- return response
-
- d = self.iCalendarRolledup(request)
- d.addCallback(_defer)
- return d
-
- return super(CalDAVResource, self).render(request)
-
- #
- # WebDAV
- #
-
- def liveProperties(self):
-
- return super(CalendarCollectionResource, self).liveProperties() + (
- (dav_namespace, "owner"), # Private Events needs this but it is also OK to return empty
- (caldav_namespace, "supported-calendar-component-set"),
- (caldav_namespace, "supported-calendar-data" ),
- )
-
-
-
-
-class CalendarObjectResource(CalDAVResource):
- """
- Calendar object resource.
-
- This resource is backed by an L{ICalendarObject} implementation.
- """
-
-
-class ScheduleInboxResource(CalDAVResource):
- """
- Schedule inbox resource.
-
- This resource is backed by an XXXXXXX implementation.
- """
-
-
-class ScheduleOutboxResource(CalDAVResource):
- """
- Schedule outbox resource.
-
- This resource is backed by an XXXXXXX implementation.
- """
Copied: CalendarServer/trunk/txdav/caldav/resource.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/caldav/resource.py)
===================================================================
--- CalendarServer/trunk/txdav/caldav/resource.py (rev 0)
+++ CalendarServer/trunk/txdav/caldav/resource.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,156 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+CalDAV resources.
+"""
+
+__all__ = [
+ "CalDAVResource",
+ "CalendarHomeResource",
+ "CalendarCollectionResource",
+ "CalendarObjectResource",
+ "ScheduleInboxResource",
+ "ScheduleOutboxResource",
+]
+
+
+import urllib
+
+from twext.python.log import LoggingMixIn
+from twext.web2.dav.element.base import dav_namespace
+from twext.web2.http_headers import MimeType
+from twext.web2.http import RedirectResponse, Response
+from twext.web2.stream import MemoryStream
+
+from twistedcaldav import caldavxml
+from twistedcaldav.caldavxml import caldav_namespace
+from twistedcaldav.config import config
+from twistedcaldav.extensions import DAVResource
+
+
+class CalDAVResource(DAVResource, LoggingMixIn):
+ """
+ CalDAV resource.
+ """
+ def davComplianceClasses(self):
+ return (
+ tuple(super(CalDAVResource, self).davComplianceClasses())
+ + config.CalDAVComplianceClasses
+ )
+
+ supportedCalendarComponentSet = caldavxml.SupportedCalendarComponentSet(
+ *[caldavxml.CalendarComponent(name=item) for item in allowedComponents]
+ )
+
+
+class CalendarHomeResource(CalDAVResource):
+ """
+ Calendar home resource.
+
+ This resource is backed by an L{ICalendarHome} implementation.
+ """
+
+
+class CalendarCollectionResource(CalDAVResource):
+ """
+ Calendar collection resource.
+
+ This resource is backed by an L{ICalendar} implementation.
+ """
+ #
+ # HTTP
+ #
+
+ def render(self, request):
+ if config.EnableMonolithicCalendars:
+ #
+ # Send listing instead of iCalendar data to HTML agents
+ # This is mostly useful for debugging...
+ #
+ # FIXME: Add a self-link to the dirlist with a query string so
+ # users can still download the actual iCalendar data?
+ #
+ # FIXME: Are there better ways to detect this than hacking in
+ # user agents?
+ #
+ # FIXME: In the meantime, make this a configurable regex list?
+ #
+ agent = request.headers.getHeader("user-agent")
+ if agent is not None and (
+ agent.startswith("Mozilla/") and agent.find("Gecko") != -1
+ ):
+ renderAsHTML = True
+ else:
+ renderAsHTML = False
+ else:
+ renderAsHTML = True
+
+ if not renderAsHTML:
+ # Render a monolithic iCalendar file
+ if request.path[-1] != "/":
+ # Redirect to include trailing '/' in URI
+ return RedirectResponse(request.unparseURL(path=urllib.quote(urllib.unquote(request.path), safe=':/')+'/'))
+
+ def _defer(data):
+ response = Response()
+ response.stream = MemoryStream(str(data))
+ response.headers.setHeader("content-type", MimeType.fromString("text/calendar"))
+ return response
+
+ d = self.iCalendarRolledup(request)
+ d.addCallback(_defer)
+ return d
+
+ return super(CalDAVResource, self).render(request)
+
+ #
+ # WebDAV
+ #
+
+ def liveProperties(self):
+
+ return super(CalendarCollectionResource, self).liveProperties() + (
+ (dav_namespace, "owner"), # Private Events needs this but it is also OK to return empty
+ (caldav_namespace, "supported-calendar-component-set"),
+ (caldav_namespace, "supported-calendar-data" ),
+ )
+
+
+
+
+class CalendarObjectResource(CalDAVResource):
+ """
+ Calendar object resource.
+
+ This resource is backed by an L{ICalendarObject} implementation.
+ """
+
+
+class ScheduleInboxResource(CalDAVResource):
+ """
+ Schedule inbox resource.
+
+ This resource is backed by an XXXXXXX implementation.
+ """
+
+
+class ScheduleOutboxResource(CalDAVResource):
+ """
+ Schedule outbox resource.
+
+ This resource is backed by an XXXXXXX implementation.
+ """
Deleted: CalendarServer/trunk/txdav/carddav/__init__.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/__init__.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,19 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-CardDAV support for Twisted.
-"""
Copied: CalendarServer/trunk/txdav/carddav/__init__.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/__init__.py)
===================================================================
--- CalendarServer/trunk/txdav/carddav/__init__.py (rev 0)
+++ CalendarServer/trunk/txdav/carddav/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,19 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+CardDAV support for Twisted.
+"""
Deleted: CalendarServer/trunk/txdav/carddav/datastore/__init__.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/__init__.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,19 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Addressbook stores.
-"""
Copied: CalendarServer/trunk/txdav/carddav/datastore/__init__.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/__init__.py)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/__init__.py (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,19 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Addressbook stores.
+"""
Deleted: CalendarServer/trunk/txdav/carddav/datastore/file.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/file.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/file.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,306 +0,0 @@
-# -*- test-case-name: txdav.carddav.datastore.test.test_file -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-File addressbook store.
-"""
-
-__all__ = [
- "AddressBookStore",
- "AddressBookStoreTransaction",
- "AddressBookHome",
- "AddressBook",
- "AddressBookObject",
-]
-
-from errno import ENOENT
-
-from twext.web2.dav.element.rfc2518 import ResourceType
-
-from twistedcaldav.sharing import InvitesDatabase
-from twistedcaldav.vcard import Component as VComponent, InvalidVCardDataError
-from twistedcaldav.vcardindex import AddressBookIndex as OldIndex
-
-from txdav.carddav.datastore.util import validateAddressBookComponent
-from txdav.carddav.iaddressbookstore import IAddressBook, IAddressBookObject
-from txdav.carddav.iaddressbookstore import IAddressBookHome
-
-from txdav.common.datastore.file import CommonDataStore, CommonHome,\
- CommonStoreTransaction, CommonHomeChild, CommonObjectResource,\
- CommonStubResource
-from txdav.common.icommondatastore import NoSuchObjectResourceError, InternalDataStoreError
-from txdav.base.datastore.file import hidden, writeOperation
-from txdav.base.propertystore.base import PropertyName
-
-from twistedcaldav import customxml, carddavxml
-
-from zope.interface import implements
-
-AddressBookStore = CommonDataStore
-
-AddressBookStoreTransaction = CommonStoreTransaction
-
-class AddressBookHome(CommonHome):
-
- implements(IAddressBookHome)
-
- def __init__(self, uid, path, addressbookStore, transaction, notifier):
- super(AddressBookHome, self).__init__(uid, path, addressbookStore, transaction, notifier)
-
- self._childClass = AddressBook
-
- addressbooks = CommonHome.children
- listAddressbooks = CommonHome.listChildren
- addressbookWithName = CommonHome.childWithName
- createAddressBookWithName = CommonHome.createChildWithName
- removeAddressBookWithName = CommonHome.removeChildWithName
-
- @property
- def _addressbookStore(self):
- return self._dataStore
-
- def createdHome(self):
- self.createAddressBookWithName("addressbook")
-
-class AddressBook(CommonHomeChild):
- """
- File-based implementation of L{IAddressBook}.
- """
- implements(IAddressBook)
-
- def __init__(self, name, addressbookHome, notifier, realName=None):
- """
- Initialize an addressbook pointing at a path on disk.
-
- @param name: the subdirectory of addressbookHome where this addressbook
- resides.
- @type name: C{str}
-
- @param addressbookHome: the home containing this addressbook.
- @type addressbookHome: L{AddressBookHome}
-
- @param realName: If this addressbook was just created, the name which it
- will eventually have on disk.
- @type realName: C{str}
- """
-
- super(AddressBook, self).__init__(name, addressbookHome, notifier,
- realName=realName)
-
- self._index = Index(self)
- self._invites = Invites(self)
- self._objectResourceClass = AddressBookObject
-
- @property
- def _addressbookHome(self):
- return self._home
-
- def resourceType(self):
- return ResourceType.addressbook #@UndefinedVariable
-
- ownerAddressBookHome = CommonHomeChild.ownerHome
- addressbookObjects = CommonHomeChild.objectResources
- listAddressbookObjects = CommonHomeChild.listObjectResources
- addressbookObjectWithName = CommonHomeChild.objectResourceWithName
- addressbookObjectWithUID = CommonHomeChild.objectResourceWithUID
- createAddressBookObjectWithName = CommonHomeChild.createObjectResourceWithName
- removeAddressBookObjectWithName = CommonHomeChild.removeObjectResourceWithName
- removeAddressBookObjectWithUID = CommonHomeChild.removeObjectResourceWithUID
- addressbookObjectsSinceToken = CommonHomeChild.objectResourcesSinceToken
-
-
- def initPropertyStore(self, props):
- # Setup peruser special properties
- props.setSpecialProperties(
- (
- PropertyName.fromElement(carddavxml.AddressBookDescription),
- ),
- (
- PropertyName.fromElement(customxml.GETCTag),
- ),
- )
-
- def _doValidate(self, component):
- component.validForCardDAV()
-
-
-class AddressBookObject(CommonObjectResource):
- """
- """
- implements(IAddressBookObject)
-
- def __init__(self, name, addressbook):
-
- super(AddressBookObject, self).__init__(name, addressbook)
-
-
- @property
- def _addressbook(self):
- return self._parentCollection
-
-
- def addressbook(self):
- return self._addressbook
-
-
- @writeOperation
- def setComponent(self, component, inserting=False):
- validateAddressBookComponent(self, self._addressbook, component, inserting)
-
- self._addressbook.retrieveOldIndex().addResource(
- self.name(), component
- )
-
- self._component = component
- # FIXME: needs to clear text cache
-
- def do():
- # Mark all properties as dirty, so they can be added back
- # to the newly updated file.
- self.properties().update(self.properties())
-
- backup = None
- if self._path.exists():
- backup = hidden(self._path.temporarySibling())
- self._path.moveTo(backup)
- fh = self._path.open("w")
- try:
- # FIXME: concurrency problem; if this write is interrupted
- # halfway through, the underlying file will be corrupt.
- fh.write(str(component))
- finally:
- fh.close()
-
- # Now re-write the original properties on the updated file
- self.properties().flush()
-
- def undo():
- if backup:
- backup.moveTo(self._path)
- else:
- self._path.remove()
- return undo
- self._transaction.addOperation(do, "set addressbook component %r" % (self.name(),))
- if self._addressbook._notifier:
- self._transaction.postCommit(self._addressbook._notifier.notify)
-
-
-
- def component(self):
- if self._component is not None:
- return self._component
- text = self.text()
-
- try:
- component = VComponent.fromString(text)
- except InvalidVCardDataError, e:
- raise InternalDataStoreError(
- "File corruption detected (%s) in file: %s"
- % (e, self._path.path)
- )
- return component
-
-
- def text(self):
- if self._component is not None:
- return str(self._component)
- try:
- fh = self._path.open()
- except IOError, e:
- if e[0] == ENOENT:
- raise NoSuchObjectResourceError(self)
- else:
- raise
-
- try:
- text = fh.read()
- finally:
- fh.close()
-
- if not (
- text.startswith("BEGIN:VCARD\r\n") or
- text.endswith("\r\nEND:VCARD\r\n")
- ):
- raise InternalDataStoreError(
- "File corruption detected (improper start) in file: %s"
- % (self._path.path,)
- )
- return text
-
- vCardText = text
-
- def uid(self):
- if not hasattr(self, "_uid"):
- self._uid = self.component().resourceUID()
- return self._uid
-
-
-class AddressBookStubResource(CommonStubResource):
- """
- Just enough resource to keep the addressbook's sql DB classes going.
- """
-
- def isAddressBookCollection(self):
- return True
-
- def getChild(self, name):
- addressbookObject = self.resource.addressbookObjectWithName(name)
- if addressbookObject:
- class ChildResource(object):
- def __init__(self, addressbookObject):
- self.addressbookObject = addressbookObject
-
- def iAddressBook(self):
- return self.addressbookObject.component()
-
- return ChildResource(addressbookObject)
- else:
- return None
-
-
-class Index(object):
- #
- # OK, here's where we get ugly.
- # The index code needs to be rewritten also, but in the meantime...
- #
- def __init__(self, addressbook):
- self.addressbook = addressbook
- stubResource = AddressBookStubResource(addressbook)
- self._oldIndex = OldIndex(stubResource)
-
-
- def addressbookObjects(self):
- addressbook = self.addressbook
- for name, uid, componentType in self._oldIndex.bruteForceSearch():
- addressbookObject = addressbook.addressbookObjectWithName(name)
-
- # Precache what we found in the index
- addressbookObject._uid = uid
- addressbookObject._componentType = componentType
-
- yield addressbookObject
-
-
-class Invites(object):
- #
- # OK, here's where we get ugly.
- # The index code needs to be rewritten also, but in the meantime...
- #
- def __init__(self, addressbook):
- self.addressbook = addressbook
- stubResource = AddressBookStubResource(addressbook)
- self._oldInvites = InvitesDatabase(stubResource)
Copied: CalendarServer/trunk/txdav/carddav/datastore/file.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/file.py)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/file.py (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/file.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,306 @@
+# -*- test-case-name: txdav.carddav.datastore.test.test_file -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+File addressbook store.
+"""
+
+__all__ = [
+ "AddressBookStore",
+ "AddressBookStoreTransaction",
+ "AddressBookHome",
+ "AddressBook",
+ "AddressBookObject",
+]
+
+from errno import ENOENT
+
+from twext.web2.dav.element.rfc2518 import ResourceType
+
+from twistedcaldav.sharing import InvitesDatabase
+from twistedcaldav.vcard import Component as VComponent, InvalidVCardDataError
+from twistedcaldav.vcardindex import AddressBookIndex as OldIndex
+
+from txdav.carddav.datastore.util import validateAddressBookComponent
+from txdav.carddav.iaddressbookstore import IAddressBook, IAddressBookObject
+from txdav.carddav.iaddressbookstore import IAddressBookHome
+
+from txdav.common.datastore.file import CommonDataStore, CommonHome,\
+ CommonStoreTransaction, CommonHomeChild, CommonObjectResource,\
+ CommonStubResource
+from txdav.common.icommondatastore import NoSuchObjectResourceError, InternalDataStoreError
+from txdav.base.datastore.file import hidden, writeOperation
+from txdav.base.propertystore.base import PropertyName
+
+from twistedcaldav import customxml, carddavxml
+
+from zope.interface import implements
+
+AddressBookStore = CommonDataStore
+
+AddressBookStoreTransaction = CommonStoreTransaction
+
+class AddressBookHome(CommonHome):
+
+ implements(IAddressBookHome)
+
+ def __init__(self, uid, path, addressbookStore, transaction, notifier):
+ super(AddressBookHome, self).__init__(uid, path, addressbookStore, transaction, notifier)
+
+ self._childClass = AddressBook
+
+ addressbooks = CommonHome.children
+ listAddressbooks = CommonHome.listChildren
+ addressbookWithName = CommonHome.childWithName
+ createAddressBookWithName = CommonHome.createChildWithName
+ removeAddressBookWithName = CommonHome.removeChildWithName
+
+ @property
+ def _addressbookStore(self):
+ return self._dataStore
+
+ def createdHome(self):
+ self.createAddressBookWithName("addressbook")
+
+class AddressBook(CommonHomeChild):
+ """
+ File-based implementation of L{IAddressBook}.
+ """
+ implements(IAddressBook)
+
+ def __init__(self, name, addressbookHome, notifier, realName=None):
+ """
+ Initialize an addressbook pointing at a path on disk.
+
+ @param name: the subdirectory of addressbookHome where this addressbook
+ resides.
+ @type name: C{str}
+
+ @param addressbookHome: the home containing this addressbook.
+ @type addressbookHome: L{AddressBookHome}
+
+ @param realName: If this addressbook was just created, the name which it
+ will eventually have on disk.
+ @type realName: C{str}
+ """
+
+ super(AddressBook, self).__init__(name, addressbookHome, notifier,
+ realName=realName)
+
+ self._index = Index(self)
+ self._invites = Invites(self)
+ self._objectResourceClass = AddressBookObject
+
+ @property
+ def _addressbookHome(self):
+ return self._home
+
+ def resourceType(self):
+ return ResourceType.addressbook #@UndefinedVariable
+
+ ownerAddressBookHome = CommonHomeChild.ownerHome
+ addressbookObjects = CommonHomeChild.objectResources
+ listAddressbookObjects = CommonHomeChild.listObjectResources
+ addressbookObjectWithName = CommonHomeChild.objectResourceWithName
+ addressbookObjectWithUID = CommonHomeChild.objectResourceWithUID
+ createAddressBookObjectWithName = CommonHomeChild.createObjectResourceWithName
+ removeAddressBookObjectWithName = CommonHomeChild.removeObjectResourceWithName
+ removeAddressBookObjectWithUID = CommonHomeChild.removeObjectResourceWithUID
+ addressbookObjectsSinceToken = CommonHomeChild.objectResourcesSinceToken
+
+
+ def initPropertyStore(self, props):
+ # Setup peruser special properties
+ props.setSpecialProperties(
+ (
+ PropertyName.fromElement(carddavxml.AddressBookDescription),
+ ),
+ (
+ PropertyName.fromElement(customxml.GETCTag),
+ ),
+ )
+
+ def _doValidate(self, component):
+ component.validForCardDAV()
+
+
+class AddressBookObject(CommonObjectResource):
+ """
+ """
+ implements(IAddressBookObject)
+
+ def __init__(self, name, addressbook):
+
+ super(AddressBookObject, self).__init__(name, addressbook)
+
+
+ @property
+ def _addressbook(self):
+ return self._parentCollection
+
+
+ def addressbook(self):
+ return self._addressbook
+
+
+ @writeOperation
+ def setComponent(self, component, inserting=False):
+ validateAddressBookComponent(self, self._addressbook, component, inserting)
+
+ self._addressbook.retrieveOldIndex().addResource(
+ self.name(), component
+ )
+
+ self._component = component
+ # FIXME: needs to clear text cache
+
+ def do():
+ # Mark all properties as dirty, so they can be added back
+ # to the newly updated file.
+ self.properties().update(self.properties())
+
+ backup = None
+ if self._path.exists():
+ backup = hidden(self._path.temporarySibling())
+ self._path.moveTo(backup)
+ fh = self._path.open("w")
+ try:
+ # FIXME: concurrency problem; if this write is interrupted
+ # halfway through, the underlying file will be corrupt.
+ fh.write(str(component))
+ finally:
+ fh.close()
+
+ # Now re-write the original properties on the updated file
+ self.properties().flush()
+
+ def undo():
+ if backup:
+ backup.moveTo(self._path)
+ else:
+ self._path.remove()
+ return undo
+ self._transaction.addOperation(do, "set addressbook component %r" % (self.name(),))
+ if self._addressbook._notifier:
+ self._transaction.postCommit(self._addressbook._notifier.notify)
+
+
+
+ def component(self):
+ if self._component is not None:
+ return self._component
+ text = self.text()
+
+ try:
+ component = VComponent.fromString(text)
+ except InvalidVCardDataError, e:
+ raise InternalDataStoreError(
+ "File corruption detected (%s) in file: %s"
+ % (e, self._path.path)
+ )
+ return component
+
+
+ def text(self):
+ if self._component is not None:
+ return str(self._component)
+ try:
+ fh = self._path.open()
+ except IOError, e:
+ if e[0] == ENOENT:
+ raise NoSuchObjectResourceError(self)
+ else:
+ raise
+
+ try:
+ text = fh.read()
+ finally:
+ fh.close()
+
+ if not (
+ text.startswith("BEGIN:VCARD\r\n") or
+ text.endswith("\r\nEND:VCARD\r\n")
+ ):
+ raise InternalDataStoreError(
+ "File corruption detected (improper start) in file: %s"
+ % (self._path.path,)
+ )
+ return text
+
+ vCardText = text
+
+ def uid(self):
+ if not hasattr(self, "_uid"):
+ self._uid = self.component().resourceUID()
+ return self._uid
+
+
+class AddressBookStubResource(CommonStubResource):
+ """
+ Just enough resource to keep the addressbook's sql DB classes going.
+ """
+
+ def isAddressBookCollection(self):
+ return True
+
+ def getChild(self, name):
+ addressbookObject = self.resource.addressbookObjectWithName(name)
+ if addressbookObject:
+ class ChildResource(object):
+ def __init__(self, addressbookObject):
+ self.addressbookObject = addressbookObject
+
+ def iAddressBook(self):
+ return self.addressbookObject.component()
+
+ return ChildResource(addressbookObject)
+ else:
+ return None
+
+
+class Index(object):
+ #
+ # OK, here's where we get ugly.
+ # The index code needs to be rewritten also, but in the meantime...
+ #
+ def __init__(self, addressbook):
+ self.addressbook = addressbook
+ stubResource = AddressBookStubResource(addressbook)
+ self._oldIndex = OldIndex(stubResource)
+
+
+ def addressbookObjects(self):
+ addressbook = self.addressbook
+ for name, uid, componentType in self._oldIndex.bruteForceSearch():
+ addressbookObject = addressbook.addressbookObjectWithName(name)
+
+ # Precache what we found in the index
+ addressbookObject._uid = uid
+ addressbookObject._componentType = componentType
+
+ yield addressbookObject
+
+
+class Invites(object):
+ #
+ # OK, here's where we get ugly.
+ # The index code needs to be rewritten also, but in the meantime...
+ #
+ def __init__(self, addressbook):
+ self.addressbook = addressbook
+ stubResource = AddressBookStubResource(addressbook)
+ self._oldInvites = InvitesDatabase(stubResource)
Deleted: CalendarServer/trunk/txdav/carddav/datastore/sql.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/sql.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/sql.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,241 +0,0 @@
-# -*- test-case-name: txdav.carddav.datastore.test.test_sql -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-__all__ = [
- "AddressBookHome",
- "AddressBook",
- "AddressBookObject",
-]
-
-from twext.web2.dav.element.rfc2518 import ResourceType
-from twext.web2.http_headers import MimeType
-
-from twistedcaldav import carddavxml, customxml
-from twistedcaldav.vcard import Component as VCard
-
-from txdav.common.datastore.sql_legacy import \
- PostgresLegacyABIndexEmulator, PostgresLegacyABInvitesEmulator,\
- PostgresLegacyABSharesEmulator
-
-from txdav.carddav.datastore.util import validateAddressBookComponent
-from txdav.carddav.iaddressbookstore import IAddressBookHome, IAddressBook,\
- IAddressBookObject
-
-from txdav.common.datastore.sql import CommonHome, CommonHomeChild,\
- CommonObjectResource
-from txdav.common.datastore.sql_tables import ADDRESSBOOK_TABLE,\
- ADDRESSBOOK_BIND_TABLE, ADDRESSBOOK_OBJECT_REVISIONS_TABLE,\
- ADDRESSBOOK_OBJECT_TABLE
-from txdav.base.propertystore.base import PropertyName
-
-from zope.interface.declarations import implements
-
-class AddressBookHome(CommonHome):
-
- implements(IAddressBookHome)
-
- def __init__(self, transaction, ownerUID, resourceID, notifier):
- super(AddressBookHome, self).__init__(transaction, ownerUID, resourceID, notifier)
-
- self._shares = PostgresLegacyABSharesEmulator(self)
- self._childClass = AddressBook
- self._childTable = ADDRESSBOOK_TABLE
- self._bindTable = ADDRESSBOOK_BIND_TABLE
-
- addressbooks = CommonHome.children
- listAddressbooks = CommonHome.listChildren
- addressbookWithName = CommonHome.childWithName
- createAddressBookWithName = CommonHome.createChildWithName
- removeAddressBookWithName = CommonHome.removeChildWithName
-
- def createdHome(self):
- self.createAddressBookWithName("addressbook")
-
-class AddressBook(CommonHomeChild):
- """
- File-based implementation of L{IAddressBook}.
- """
- implements(IAddressBook)
-
- def __init__(self, home, name, resourceID, notifier):
- """
- Initialize an addressbook pointing at a path on disk.
-
- @param name: the subdirectory of addressbookHome where this addressbook
- resides.
- @type name: C{str}
-
- @param addressbookHome: the home containing this addressbook.
- @type addressbookHome: L{AddressBookHome}
-
- @param realName: If this addressbook was just created, the name which it
- will eventually have on disk.
- @type realName: C{str}
- """
-
- super(AddressBook, self).__init__(home, name, resourceID, notifier)
-
- self._index = PostgresLegacyABIndexEmulator(self)
- self._invites = PostgresLegacyABInvitesEmulator(self)
- self._objectResourceClass = AddressBookObject
- self._bindTable = ADDRESSBOOK_BIND_TABLE
- self._homeChildTable = ADDRESSBOOK_TABLE
- self._revisionsTable = ADDRESSBOOK_OBJECT_REVISIONS_TABLE
- self._objectTable = ADDRESSBOOK_OBJECT_TABLE
-
- @property
- def _addressbookHome(self):
- return self._home
-
- def resourceType(self):
- return ResourceType.addressbook #@UndefinedVariable
-
- ownerAddressBookHome = CommonHomeChild.ownerHome
- addressbookObjects = CommonHomeChild.objectResources
- listAddressbookObjects = CommonHomeChild.listObjectResources
- addressbookObjectWithName = CommonHomeChild.objectResourceWithName
- addressbookObjectWithUID = CommonHomeChild.objectResourceWithUID
- createAddressBookObjectWithName = CommonHomeChild.createObjectResourceWithName
- removeAddressBookObjectWithName = CommonHomeChild.removeObjectResourceWithName
- removeAddressBookObjectWithUID = CommonHomeChild.removeObjectResourceWithUID
- addressbookObjectsSinceToken = CommonHomeChild.objectResourcesSinceToken
-
-
- def initPropertyStore(self, props):
- # Setup peruser special properties
- props.setSpecialProperties(
- (
- PropertyName.fromElement(carddavxml.AddressBookDescription),
- ),
- (
- PropertyName.fromElement(customxml.GETCTag),
- ),
- )
-
- def _doValidate(self, component):
- component.validForCardDAV()
-
- def contentType(self):
- """
- The content type of Addresbook objects is text/vcard.
- """
- return MimeType.fromString("text/vcard; charset=utf-8")
-
-class AddressBookObject(CommonObjectResource):
-
- implements(IAddressBookObject)
-
- def __init__(self, name, addressbook, resid):
-
- super(AddressBookObject, self).__init__(name, addressbook, resid)
-
- self._objectTable = ADDRESSBOOK_OBJECT_TABLE
-
- @property
- def _addressbook(self):
- return self._parentCollection
-
- def addressbook(self):
- return self._addressbook
-
- def setComponent(self, component, inserting=False):
- validateAddressBookComponent(self, self._addressbook, component, inserting)
-
- self.updateDatabase(component, inserting=inserting)
- if inserting:
- self._addressbook._insertRevision(self._name)
- else:
- self._addressbook._updateRevision(self._name)
-
- if self._addressbook._notifier:
- self._addressbook._home._txn.postCommit(self._addressbook._notifier.notify)
-
- def updateDatabase(self, component, expand_until=None, reCreate=False, inserting=False):
- """
- Update the database tables for the new data being written.
-
- @param component: addressbook data to store
- @type component: L{Component}
- """
-
- componentText = str(component)
- self._objectText = componentText
-
- # ADDRESSBOOK_OBJECT table update
- if inserting:
- self._resourceID = self._txn.execSQL(
- """
- insert into ADDRESSBOOK_OBJECT
- (ADDRESSBOOK_RESOURCE_ID, RESOURCE_NAME, VCARD_TEXT, VCARD_UID)
- values
- (%s, %s, %s, %s)
- returning RESOURCE_ID
- """,
- [
- self._addressbook._resourceID,
- self._name,
- componentText,
- component.resourceUID(),
- ]
- )[0][0]
- else:
- self._txn.execSQL(
- """
- update ADDRESSBOOK_OBJECT set
- (VCARD_TEXT, VCARD_UID, MODIFIED)
- =
- (%s, %s, timezone('UTC', CURRENT_TIMESTAMP))
- where RESOURCE_ID = %s
- """,
- [
- componentText,
- component.resourceUID(),
- self._resourceID
- ]
- )
-
- def component(self):
- return VCard.fromString(self.vCardText())
-
- def text(self):
- if self._objectText is None:
- text = self._txn.execSQL(
- "select VCARD_TEXT from ADDRESSBOOK_OBJECT where "
- "RESOURCE_ID = %s", [self._resourceID]
- )[0][0]
- self._objectText = text
- return text
- else:
- return self._objectText
-
- vCardText = text
-
- def uid(self):
- return self.component().resourceUID()
-
- def name(self):
- return self._name
-
- def componentType(self):
- return self.component().mainType()
-
- # IDataStoreResource
- def contentType(self):
- """
- The content type of Addressbook objects is text/x-vcard.
- """
- return MimeType.fromString("text/vcard; charset=utf-8")
Copied: CalendarServer/trunk/txdav/carddav/datastore/sql.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/sql.py)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/sql.py (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/sql.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,241 @@
+# -*- test-case-name: txdav.carddav.datastore.test.test_sql -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+__all__ = [
+ "AddressBookHome",
+ "AddressBook",
+ "AddressBookObject",
+]
+
+from twext.web2.dav.element.rfc2518 import ResourceType
+from twext.web2.http_headers import MimeType
+
+from twistedcaldav import carddavxml, customxml
+from twistedcaldav.vcard import Component as VCard
+
+from txdav.common.datastore.sql_legacy import \
+ PostgresLegacyABIndexEmulator, PostgresLegacyABInvitesEmulator,\
+ PostgresLegacyABSharesEmulator
+
+from txdav.carddav.datastore.util import validateAddressBookComponent
+from txdav.carddav.iaddressbookstore import IAddressBookHome, IAddressBook,\
+ IAddressBookObject
+
+from txdav.common.datastore.sql import CommonHome, CommonHomeChild,\
+ CommonObjectResource
+from txdav.common.datastore.sql_tables import ADDRESSBOOK_TABLE,\
+ ADDRESSBOOK_BIND_TABLE, ADDRESSBOOK_OBJECT_REVISIONS_TABLE,\
+ ADDRESSBOOK_OBJECT_TABLE
+from txdav.base.propertystore.base import PropertyName
+
+from zope.interface.declarations import implements
+
+class AddressBookHome(CommonHome):
+
+ implements(IAddressBookHome)
+
+ def __init__(self, transaction, ownerUID, resourceID, notifier):
+ super(AddressBookHome, self).__init__(transaction, ownerUID, resourceID, notifier)
+
+ self._shares = PostgresLegacyABSharesEmulator(self)
+ self._childClass = AddressBook
+ self._childTable = ADDRESSBOOK_TABLE
+ self._bindTable = ADDRESSBOOK_BIND_TABLE
+
+ addressbooks = CommonHome.children
+ listAddressbooks = CommonHome.listChildren
+ addressbookWithName = CommonHome.childWithName
+ createAddressBookWithName = CommonHome.createChildWithName
+ removeAddressBookWithName = CommonHome.removeChildWithName
+
+ def createdHome(self):
+ self.createAddressBookWithName("addressbook")
+
+class AddressBook(CommonHomeChild):
+ """
+ File-based implementation of L{IAddressBook}.
+ """
+ implements(IAddressBook)
+
+ def __init__(self, home, name, resourceID, notifier):
+ """
+ Initialize an addressbook pointing at a path on disk.
+
+ @param name: the subdirectory of addressbookHome where this addressbook
+ resides.
+ @type name: C{str}
+
+ @param addressbookHome: the home containing this addressbook.
+ @type addressbookHome: L{AddressBookHome}
+
+ @param realName: If this addressbook was just created, the name which it
+ will eventually have on disk.
+ @type realName: C{str}
+ """
+
+ super(AddressBook, self).__init__(home, name, resourceID, notifier)
+
+ self._index = PostgresLegacyABIndexEmulator(self)
+ self._invites = PostgresLegacyABInvitesEmulator(self)
+ self._objectResourceClass = AddressBookObject
+ self._bindTable = ADDRESSBOOK_BIND_TABLE
+ self._homeChildTable = ADDRESSBOOK_TABLE
+ self._revisionsTable = ADDRESSBOOK_OBJECT_REVISIONS_TABLE
+ self._objectTable = ADDRESSBOOK_OBJECT_TABLE
+
+ @property
+ def _addressbookHome(self):
+ return self._home
+
+ def resourceType(self):
+ return ResourceType.addressbook #@UndefinedVariable
+
+ ownerAddressBookHome = CommonHomeChild.ownerHome
+ addressbookObjects = CommonHomeChild.objectResources
+ listAddressbookObjects = CommonHomeChild.listObjectResources
+ addressbookObjectWithName = CommonHomeChild.objectResourceWithName
+ addressbookObjectWithUID = CommonHomeChild.objectResourceWithUID
+ createAddressBookObjectWithName = CommonHomeChild.createObjectResourceWithName
+ removeAddressBookObjectWithName = CommonHomeChild.removeObjectResourceWithName
+ removeAddressBookObjectWithUID = CommonHomeChild.removeObjectResourceWithUID
+ addressbookObjectsSinceToken = CommonHomeChild.objectResourcesSinceToken
+
+
+ def initPropertyStore(self, props):
+ # Setup peruser special properties
+ props.setSpecialProperties(
+ (
+ PropertyName.fromElement(carddavxml.AddressBookDescription),
+ ),
+ (
+ PropertyName.fromElement(customxml.GETCTag),
+ ),
+ )
+
+ def _doValidate(self, component):
+ component.validForCardDAV()
+
+ def contentType(self):
+ """
+ The content type of Addresbook objects is text/vcard.
+ """
+ return MimeType.fromString("text/vcard; charset=utf-8")
+
+class AddressBookObject(CommonObjectResource):
+
+ implements(IAddressBookObject)
+
+ def __init__(self, name, addressbook, resid):
+
+ super(AddressBookObject, self).__init__(name, addressbook, resid)
+
+ self._objectTable = ADDRESSBOOK_OBJECT_TABLE
+
+ @property
+ def _addressbook(self):
+ return self._parentCollection
+
+ def addressbook(self):
+ return self._addressbook
+
+ def setComponent(self, component, inserting=False):
+ validateAddressBookComponent(self, self._addressbook, component, inserting)
+
+ self.updateDatabase(component, inserting=inserting)
+ if inserting:
+ self._addressbook._insertRevision(self._name)
+ else:
+ self._addressbook._updateRevision(self._name)
+
+ if self._addressbook._notifier:
+ self._addressbook._home._txn.postCommit(self._addressbook._notifier.notify)
+
+ def updateDatabase(self, component, expand_until=None, reCreate=False, inserting=False):
+ """
+ Update the database tables for the new data being written.
+
+ @param component: addressbook data to store
+ @type component: L{Component}
+ """
+
+ componentText = str(component)
+ self._objectText = componentText
+
+ # ADDRESSBOOK_OBJECT table update
+ if inserting:
+ self._resourceID = self._txn.execSQL(
+ """
+ insert into ADDRESSBOOK_OBJECT
+ (ADDRESSBOOK_RESOURCE_ID, RESOURCE_NAME, VCARD_TEXT, VCARD_UID)
+ values
+ (%s, %s, %s, %s)
+ returning RESOURCE_ID
+ """,
+ [
+ self._addressbook._resourceID,
+ self._name,
+ componentText,
+ component.resourceUID(),
+ ]
+ )[0][0]
+ else:
+ self._txn.execSQL(
+ """
+ update ADDRESSBOOK_OBJECT set
+ (VCARD_TEXT, VCARD_UID, MODIFIED)
+ =
+ (%s, %s, timezone('UTC', CURRENT_TIMESTAMP))
+ where RESOURCE_ID = %s
+ """,
+ [
+ componentText,
+ component.resourceUID(),
+ self._resourceID
+ ]
+ )
+
+ def component(self):
+ return VCard.fromString(self.vCardText())
+
+ def text(self):
+ if self._objectText is None:
+ text = self._txn.execSQL(
+ "select VCARD_TEXT from ADDRESSBOOK_OBJECT where "
+ "RESOURCE_ID = %s", [self._resourceID]
+ )[0][0]
+ self._objectText = text
+ return text
+ else:
+ return self._objectText
+
+ vCardText = text
+
+ def uid(self):
+ return self.component().resourceUID()
+
+ def name(self):
+ return self._name
+
+ def componentType(self):
+ return self.component().mainType()
+
+ # IDataStoreResource
+ def contentType(self):
+ """
+ The content type of Addressbook objects is text/x-vcard.
+ """
+ return MimeType.fromString("text/vcard; charset=utf-8")
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/__init__.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/__init__.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,20 +0,0 @@
-# -*- test-case-name: txdav.carddav.datastore.test -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-AddressBook store tests.
-"""
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/__init__.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/__init__.py)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/__init__.py (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,20 @@
+# -*- test-case-name: txdav.carddav.datastore.test -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+AddressBook store tests.
+"""
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/1.vcf
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/1.vcf 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/1.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,12 +0,0 @@
-BEGIN:VCARD
-VERSION:3.0
-N:Thompson;Default;;;
-FN:Default Thompson
-EMAIL;type=INTERNET;type=WORK;type=pref:lthompson at example.com
-TEL;type=WORK;type=pref:1-555-555-5555
-TEL;type=CELL:1-444-444-4444
-item1.ADR;type=WORK;type=pref:;;1245 Test;Sesame Street;California;11111;USA
-item1.X-ABADR:us
-UID:uid1
-NOTE:CardDAV protocol updates
-END:VCARD
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/1.vcf (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/1.vcf)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/1.vcf (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/1.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,12 @@
+BEGIN:VCARD
+VERSION:3.0
+N:Thompson;Default;;;
+FN:Default Thompson
+EMAIL;type=INTERNET;type=WORK;type=pref:lthompson at example.com
+TEL;type=WORK;type=pref:1-555-555-5555
+TEL;type=CELL:1-444-444-4444
+item1.ADR;type=WORK;type=pref:;;1245 Test;Sesame Street;California;11111;USA
+item1.X-ABADR:us
+UID:uid1
+NOTE:CardDAV protocol updates
+END:VCARD
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/2.vcf
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/2.vcf 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/2.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,17 +0,0 @@
-BEGIN:VCARD
-VERSION:3.0
-N:Contact;Mulberry;;;
-FN:Mulberry Contact
-NICKNAME:mulberry
-ORG:Apple Inc.;
-EMAIL;type=INTERNET;type=WORK;type=pref:mulberry at example.com
-TEL;type=HOME;type=pref:777-777-7777
-TEL;type=WORK:8888888888
-TEL;type=WORK;type=FAX:5555555555
-item1.ADR;type=WORK;type=pref:;;1234 Infinite Circle;Exampletino\, CA 99999;USA;;
-item1.X-ABADR:us
-NOTE:This is a contact created in Mulberry.
-item2.URL;type=pref:http://www.example.com/~magic
-item2.X-ABLabel:_$!<HomePage>!$_
-UID:uid2
-END:VCARD
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/2.vcf (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/2.vcf)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/2.vcf (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/2.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,17 @@
+BEGIN:VCARD
+VERSION:3.0
+N:Contact;Mulberry;;;
+FN:Mulberry Contact
+NICKNAME:mulberry
+ORG:Apple Inc.;
+EMAIL;type=INTERNET;type=WORK;type=pref:mulberry at example.com
+TEL;type=HOME;type=pref:777-777-7777
+TEL;type=WORK:8888888888
+TEL;type=WORK;type=FAX:5555555555
+item1.ADR;type=WORK;type=pref:;;1234 Infinite Circle;Exampletino\, CA 99999;USA;;
+item1.X-ABADR:us
+NOTE:This is a contact created in Mulberry.
+item2.URL;type=pref:http://www.example.com/~magic
+item2.X-ABLabel:_$!<HomePage>!$_
+UID:uid2
+END:VCARD
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/3.vcf
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/3.vcf 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/3.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,12 +0,0 @@
-BEGIN:VCARD
-VERSION:3.0
-N:Kawado;Saeko;;;
-FN:Snow Leopard
-ORG:Snow Leopard;
-EMAIL;type=INTERNET;type=WORK;type=pref:snowleopard at example.com
-TEL;type=WORK;type=pref:777-777-7777
-item1.ADR;type=WORK;type=pref:;;1 Fidel Ave. Suite 100;Mountain Top;CA;99999;USA
-item1.X-ABADR:us
-X-ABShowAs:COMPANY
-UID:uid3
-END:VCARD
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/3.vcf (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/3.vcf)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/3.vcf (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_1/3.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,12 @@
+BEGIN:VCARD
+VERSION:3.0
+N:Kawado;Saeko;;;
+FN:Snow Leopard
+ORG:Snow Leopard;
+EMAIL;type=INTERNET;type=WORK;type=pref:snowleopard at example.com
+TEL;type=WORK;type=pref:777-777-7777
+item1.ADR;type=WORK;type=pref:;;1 Fidel Ave. Suite 100;Mountain Top;CA;99999;USA
+item1.X-ABADR:us
+X-ABShowAs:COMPANY
+UID:uid3
+END:VCARD
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/3765A955-1B96-41EA-994D-335192BEDCCD.vcf
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/3765A955-1B96-41EA-994D-335192BEDCCD.vcf 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/3765A955-1B96-41EA-994D-335192BEDCCD.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,18 +0,0 @@
-BEGIN:VCARD
-VERSION:3.0
-N:InfoIn;All;;;
-FN:All InfoIn
-ORG:allinfo Company;
-EMAIL;type=INTERNET;type=WORK;type=pref:allinfomationin at example.com
-TEL;type=WORK;type=pref:777-777-7777
-TEL;type=CELL:8888888888
-item1.ADR;type=WORK;type=pref:;;1 Gally Street Apt #2;Mountain Top;CA;99999;USA
-item1.X-ABADR:us
-X-YAHOO;type=WORK;type=pref:saeko.where at example.com
-X-YAHOO-ID;type=WORK;type=pref:saeko.where at example.com
-item2.X-ABRELATEDNAMES;type=pref:Mayumi Yan
-item2.X-ABLabel:_$!<Friend>!$_
-item3.X-ABRELATEDNAMES:Shane
-item3.X-ABLabel:_$!<Assistant>!$_
-UID:3765A955-1B96-41EA-994D-335192BEDCCD
-END:VCARD
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/3765A955-1B96-41EA-994D-335192BEDCCD.vcf (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/3765A955-1B96-41EA-994D-335192BEDCCD.vcf)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/3765A955-1B96-41EA-994D-335192BEDCCD.vcf (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/3765A955-1B96-41EA-994D-335192BEDCCD.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,18 @@
+BEGIN:VCARD
+VERSION:3.0
+N:InfoIn;All;;;
+FN:All InfoIn
+ORG:allinfo Company;
+EMAIL;type=INTERNET;type=WORK;type=pref:allinfomationin at example.com
+TEL;type=WORK;type=pref:777-777-7777
+TEL;type=CELL:8888888888
+item1.ADR;type=WORK;type=pref:;;1 Gally Street Apt #2;Mountain Top;CA;99999;USA
+item1.X-ABADR:us
+X-YAHOO;type=WORK;type=pref:saeko.where at example.com
+X-YAHOO-ID;type=WORK;type=pref:saeko.where at example.com
+item2.X-ABRELATEDNAMES;type=pref:Mayumi Yan
+item2.X-ABLabel:_$!<Friend>!$_
+item3.X-ABRELATEDNAMES:Shane
+item3.X-ABLabel:_$!<Assistant>!$_
+UID:3765A955-1B96-41EA-994D-335192BEDCCD
+END:VCARD
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44745975-AE6D-4FB0-80A6-A298427E047A.vcf
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44745975-AE6D-4FB0-80A6-A298427E047A.vcf 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44745975-AE6D-4FB0-80A6-A298427E047A.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,12 +0,0 @@
-BEGIN:VCARD
-VERSION:3.0
-N:Mariotte;WithNote;;;
-FN:WithNote Mariotte
-EMAIL;type=INTERNET;type=WORK;type=pref:withnmariotte at example.com
-TEL;type=WORK;type=pref:1-777-777-7777
-TEL;type=CELL:1-8888888888
-item1.ADR;type=WORK;type=pref:;;1 North Blvd;Cupertino;CA;99999;United States
-item1.X-ABADR:us
-NOTE: Address book server test contact that hsa note field filled in.
-UID:44745975-AE6D-4FB0-80A6-A298427E047A
-END:VCARD
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44745975-AE6D-4FB0-80A6-A298427E047A.vcf (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44745975-AE6D-4FB0-80A6-A298427E047A.vcf)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44745975-AE6D-4FB0-80A6-A298427E047A.vcf (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44745975-AE6D-4FB0-80A6-A298427E047A.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,12 @@
+BEGIN:VCARD
+VERSION:3.0
+N:Mariotte;WithNote;;;
+FN:WithNote Mariotte
+EMAIL;type=INTERNET;type=WORK;type=pref:withnmariotte at example.com
+TEL;type=WORK;type=pref:1-777-777-7777
+TEL;type=CELL:1-8888888888
+item1.ADR;type=WORK;type=pref:;;1 North Blvd;Cupertino;CA;99999;United States
+item1.X-ABADR:us
+NOTE: Address book server test contact that hsa note field filled in.
+UID:44745975-AE6D-4FB0-80A6-A298427E047A
+END:VCARD
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44EE78BF-8814-4471-899C-92280CEFB098.vcf
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44EE78BF-8814-4471-899C-92280CEFB098.vcf 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44EE78BF-8814-4471-899C-92280CEFB098.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,11 +0,0 @@
-BEGIN:VCARD
-VERSION:3.0
-N:Śuterry;HiAscii;;;
-FN:HiAscii Śuterry
-EMAIL;type=INTERNET;type=WORK;type=pref:hiascii at example.com
-TEL;type=WORK;type=pref:777-777-7777
-TEL;type=CELL:8888888888
-item1.ADR;type=WORK;type=pref:;;1 ïlena;Paris;Paris;77777;France
-item1.X-ABADR:us
-UID:44EE78BF-8814-4471-899C-92280CEFB098
-END:VCARD
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44EE78BF-8814-4471-899C-92280CEFB098.vcf (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44EE78BF-8814-4471-899C-92280CEFB098.vcf)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44EE78BF-8814-4471-899C-92280CEFB098.vcf (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/44EE78BF-8814-4471-899C-92280CEFB098.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,11 @@
+BEGIN:VCARD
+VERSION:3.0
+N:Śuterry;HiAscii;;;
+FN:HiAscii Śuterry
+EMAIL;type=INTERNET;type=WORK;type=pref:hiascii at example.com
+TEL;type=WORK;type=pref:777-777-7777
+TEL;type=CELL:8888888888
+item1.ADR;type=WORK;type=pref:;;1 ïlena;Paris;Paris;77777;France
+item1.X-ABADR:us
+UID:44EE78BF-8814-4471-899C-92280CEFB098
+END:VCARD
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/8424B7F0-C878-4722-B522-EBB07CF48AD7.vcf
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/8424B7F0-C878-4722-B522-EBB07CF48AD7.vcf 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/8424B7F0-C878-4722-B522-EBB07CF48AD7.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,12 +0,0 @@
-BEGIN:VCARD
-VERSION:3.0
-N:バイト;ダブル;;;
-FN:ダブル バイト
-EMAIL;type=INTERNET;type=WORK;type=pref:doublebytes at example.com
-TEL;type=WORK;type=pref:777-777-7777
-TEL;type=CELL:8888888888
-item1.ADR;type=WORK;type=pref:;;1-23-4 Irohacho #2;Tokyo;Japan;33-3333;Japan
-item1.X-ABADR:us
-NOTE:日本ですよ。
-UID:8424B7F0-C878-4722-B522-EBB07CF48AD7
-END:VCARD
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/8424B7F0-C878-4722-B522-EBB07CF48AD7.vcf (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/8424B7F0-C878-4722-B522-EBB07CF48AD7.vcf)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/8424B7F0-C878-4722-B522-EBB07CF48AD7.vcf (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/8424B7F0-C878-4722-B522-EBB07CF48AD7.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,12 @@
+BEGIN:VCARD
+VERSION:3.0
+N:バイト;ダブル;;;
+FN:ダブル バイト
+EMAIL;type=INTERNET;type=WORK;type=pref:doublebytes at example.com
+TEL;type=WORK;type=pref:777-777-7777
+TEL;type=CELL:8888888888
+item1.ADR;type=WORK;type=pref:;;1-23-4 Irohacho #2;Tokyo;Japan;33-3333;Japan
+item1.X-ABADR:us
+NOTE:日本ですよ。
+UID:8424B7F0-C878-4722-B522-EBB07CF48AD7
+END:VCARD
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/934731C6-1C95-4C40-BE1F-FA4215B2307B.vcf
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/934731C6-1C95-4C40-BE1F-FA4215B2307B.vcf 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/934731C6-1C95-4C40-BE1F-FA4215B2307B.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,993 +0,0 @@
-BEGIN:VCARD
-VERSION:3.0
-N:Picture;With;;;
-FN:With Picture
-EMAIL;type=INTERNET;type=WORK;type=pref:withpicture at example.com
-TEL;type=WORK;type=pref:777-777-7777
-TEL;type=CELL:8888888888
-item1.ADR;type=WORK;type=pref:;;1234 Golly Street;Sunnyside;CA;99999;USA
-item1.X-ABADR:us
-PHOTO;BASE64:
- /9j/4AAQSkZJRgABAQAAAQABAAD/7QA8UGhvdG9zaG9wIDMuMAA4QklNBAQAAAAAAB8cAVoAAx
- sl RxwCAAACAAIcAhkAC1Bob3RvIEJvb3RoAP/iG6hJQ0NfUFJPRklMRQABAQAAG5hhcHBsAgA
- AAG1u dHJSR0IgWFlaIAfaAAEAEwAJADEABGFjc3BBUFBMAAAAAAAAAAAAAAAAAAAAAAAAAAAA
- AAAAAAD2 1gABAAAAANMtYXBwbFYcEOZVYuhIRg5LwLIi62wAAAAAAAAAAAAAAAAAAAAAAAAAA
- AAAAAAAAAAA AAAAEXJYWVoAAAFQAAAAFGdYWVoAAAFkAAAAFGJYWVoAAAF4AAAAFHd0cHQAAA
- GMAAAAFGNoYWQA AAGgAAAALHJUUkMAAAHMAAAIDGdUUkMAAAnYAAAIDGJUUkMAABHkAAAIDGF
- hcmcAABnwAAAAIGFh Z2cAABoQAAAAIGFhYmcAABowAAAAIHZjZ3QAABpQAAAAMG5kaW4AABqA
- AAAAOGRlc2MAABq4AAAA ZGRzY20AABscAAAALm1tb2QAABtMAAAAKGNwcnQAABt0AAAAJFhZW
- iAAAAAAAAB7vQAAQXsAAAJL WFlaIAAAAAAAAFYqAACp0AAAFF9YWVogAAAAAAAAJO8AABS1AA
- C8glhZWiAAAAAAAADz2AABAAAA ARYIc2YzMgAAAAAAAQu3AAAFlv//81cAAAcpAAD91///+7f
- ///2mAAAD2gAAwPZjdXJ2AAAAAAAA BAAAAAAFAAoADwAUABkAHgAjACgALQAyADcAOwBAAEUA
- SgBPAFQAWQBeAGMAaABtAHIAdwB8AIEA hgCLAJAAlQCaAJ8ApACpAK4AsgC3ALwAwQDGAMsA0
- ADVANoA4ADlAOoA8AD1APsBAQEHAQwBEgEY AR4BJQErATEBOAE+AUUBSwFSAVkBYAFmAW0BdQ
- F8AYMBigGSAZkBoQGoAbABuAHAAcgB0AHYAeAB 6QHxAfoCAgILAhQCHAIlAi4CNwJAAkoCUwJ
- cAmYCcAJ5AoMCjQKXAqECqwK1Ar8CygLUAt8C6gL0 Av8DCgMVAyADKwM3A0IDTQNZA2UDcAN8
- A4gDlAOgA6wDuQPFA9ID3gPrA/gEBAQRBB4ELAQ5BEYE VARhBG8EfASKBJgEpgS0BMIE0QTfB
- O4E/AULBRoFKAU3BUcFVgVlBXQFhAWTBaMFswXDBdMF4wXz BgMGFAYkBjUGRQZWBmcGeAaJBp
- oGqwa9Bs4G4AbyBwMHFQcnBzkHTAdeB3AHgweWB6gHuwfOB+EH 9AgICBsILwhCCFYIagh+CJI
- Ipgi6CM4I4wj3CQwJIQk2CUsJYAl1CYoJoAm1CcsJ4An2CgwKIgo5 Ck8KZQp8CpIKqQrACtcK
- 7gsFCx0LNAtLC2MLewuTC6sLwwvbC/MMDAwkDD0MVgxuDIcMoQy6DNMM 7Q0GDSANOg1UDW4Ni
- A2iDbwN1w3xDgwOJw5CDl0OeA6TDq8Oyg7mDwIPHg86D1YPcg+OD6sPyA/k EAEQHhA7EFgQdh
- CTELEQzhDsEQoRKBFGEWQRgxGhEcAR3xH+Eh0SPBJbEnoSmhK5EtkS+RMZEzkT WRN6E5oTuxP
- bE/wUHRQ+FF8UgRSiFMQU5RUHFSkVSxVtFZAVshXVFfcWGhY9FmAWgxanFsoW7hcS FzUXWRd9
- F6IXxhfqGA8YNBhZGH0YoxjIGO0ZExk4GV4ZhBmqGdAZ9hodGkMaahqQGrca3hsGGy0b VBt8G
- 6MbyxvzHBscQxxsHJQcvRzmHQ4dNx1gHYodsx3dHgYeMB5aHoQerh7YHwMfLR9YH4Mfrh/Z IA
- QgMCBbIIcgsyDeIQohNyFjIY8hvCHpIhUiQiJwIp0iyiL4IyUjUyOBI68j3SQMJDokaSSXJMYk
- 9SUkJVQlgyWzJeImEiZCJnImoybTJwMnNCdlJ5Ynxyf4KCooWyiNKL4o8CkiKVUphym5KewqHy
- pS KoUquCrrKx4rUiuGK7or7iwiLFYsiiy/LPQtKS1eLZMtyC39LjMuaS6eLtQvCy9BL3cvri/
- kMBsw UjCJMMEw+DEwMWcxnzHXMg8ySDKAMrgy8TMqM2MznDPVNA80SDSCNLw09jUwNWo1pTXf
- Nho2VTaQ Nss3BjdCN343uTf1ODE4bTiqOOY5IzlgOZ052joXOlQ6kjrPOw07SzuJO8c8BjxEP
- IM8wj0BPUA9 fz2/Pf4+Pj5+Pr4+/j8/P38/wEAAQEFAgkDEQQVBR0GIQcpCDEJOQpFC00MWQ1
- hDm0PeRCFEZUSo ROxFMEV0RbhF/EZARoVGykcOR1NHmUfeSCNIaUivSPVJO0mBScdKDkpVSpt
- K4ksqS3FLuEwATEhM kEzYTSBNaE2xTfpOQk6MTtVPHk9nT7FP+1BFUI9Q2VEkUW5RuVIEUk9S
- mlLlUzFTfFPIVBRUYFSt VPlVRlWSVd9WLFZ6VsdXFFdiV7BX/lhMWJpY6Vk4WYZZ1VokWnRaw
- 1sTW2NbslwDXFNco1z0XURd lV3mXjdeiV7aXyxffl/QYCJgdGDHYRlhbGG/YhJiZWK5YwxjYG
- O0ZAhkXGSxZQVlWmWvZgRmWWav ZwRnWmewaAZoXGiyaQlpX2m2ag1qZGq8axNra2vDbBtsc2z
- LbSNtfG3Vbi5uh27gbzpvk2/tcEdw oXD7cVZxsHILcmZywXMcc3hz03QvdIt053VDdaB1/HZZ
- drZ3E3dwd854K3iJeOd5RXmjegJ6YHq/ ex57fXvcfDx8m3z7fVt9u34bfnx+3H89f55//4Bgg
- MKBI4GFgeeCSYKrgw6DcIPThDaEmYT8hWCF w4YnhouG74dUh7iIHYiBiOaJTImxihaKfIrii0
- iLrowUjHuM4o1Ija+OF45+juWPTY+1kB2QhZDu kVaRv5IokpGS+pNkk82UN5ShlQuVdZXglkq
- WtZcgl4uX95himM6ZOpmmmhKafprrm1ebxJwxnJ+d DJ15neeeVZ7DnzGfoKAPoH2g7KFbocui
- OqKqoxqjiqP6pGqk26VMpbymLqafpxCngqf0qGWo2KlK qbyqL6qiqxWriKv7rG+s461WrcuuP
- 66zryivnbARsIew/LFxseeyXbLTs0mzv7Q2tK21JLWbthK2 ibcBt3m38bhpuOG5WrnSuku6xL
- s+u7e8MLyqvSS9nr4ZvpO/Dr+JwATAf8D6wXbB8cJtwunDZsPi xF/E3MVZxdbGU8bRx07HzMh
- KyMnJR8nGykXKxMtDy8LMQszBzUHNwc5CzsLPQ8/D0ETQxtFH0cjS StLM007T0NRT1NbVWNXb
- 1l7W4tdl1+nYbdjx2XXZ+tp/2wPbiNwO3JPdGd2e3iTeqt8x37fgPuDF 4Uzh0+Ja4uLjauPy5
- HrlAuWL5hPmnOcl56/oOOjC6Uzp1upg6urrdev/7IrtFu2h7izuuO9E79Dw XPDp8XXyAvKP8x
- zzqvQ39MX1U/Xh9m/2/veM+Bv4qvk5+cn6Wfro+3j8CPyZ/Sn9uv5L/tz/bmN1 cnYAAAAAAAA
- EAAAAAAUACgAPABQAGQAeACMAKAAtADIANwA7AEAARQBKAE8AVABZAF4AYwBoAG0A cgB3AHwA
- gQCGAIsAkACVAJoAnwCkAKkArgCyALcAvADBAMYAywDQANUA2gDgAOUA6gDwAPUA+wEB AQcBD
- AESARgBHgElASsBMQE4AT4BRQFLAVIBWQFgAWYBbQF1AXwBgwGKAZIBmQGhAagBsAG4AcAB yA
- HQAdgB4AHpAfEB+gICAgsCFAIcAiUCLgI3AkACSgJTAlwCZgJwAnkCgwKNApcCoQKrArUCvwLK
- AtQC3wLqAvQC/wMKAxUDIAMrAzcDQgNNA1kDZQNwA3wDiAOUA6ADrAO5A8UD0gPeA+sD+AQEBB
- EE HgQsBDkERgRUBGEEbwR8BIoEmASmBLQEwgTRBN8E7gT8BQsFGgUoBTcFRwVWBWUFdAWEBZM
- FowWz BcMF0wXjBfMGAwYUBiQGNQZFBlYGZwZ4BokGmgarBr0GzgbgBvIHAwcVBycHOQdMB14H
- cAeDB5YH qAe7B84H4Qf0CAgIGwgvCEIIVghqCH4IkgimCLoIzgjjCPcJDAkhCTYJSwlgCXUJi
- gmgCbUJywng CfYKDAoiCjkKTwplCnwKkgqpCsAK1wruCwULHQs0C0sLYwt7C5MLqwvDC9sL8w
- wMDCQMPQxWDG4M hwyhDLoM0wztDQYNIA06DVQNbg2IDaINvA3XDfEODA4nDkIOXQ54DpMOrw7
- KDuYPAg8eDzoPVg9y D44Pqw/ID+QQARAeEDsQWBB2EJMQsRDOEOwRChEoEUYRZBGDEaERwBHf
- Ef4SHRI8ElsSehKaErkS 2RL5ExkTORNZE3oTmhO7E9sT/BQdFD4UXxSBFKIUxBTlFQcVKRVLF
- W0VkBWyFdUV9xYaFj0WYBaD FqcWyhbuFxIXNRdZF30XohfGF+oYDxg0GFkYfRijGMgY7RkTGT
- gZXhmEGaoZ0Bn2Gh0aQxpqGpAa txreGwYbLRtUG3wboxvLG/McGxxDHGwclBy9HOYdDh03HWA
- dih2zHd0eBh4wHloehB6uHtgfAx8t H1gfgx+uH9kgBCAwIFsghyCzIN4hCiE3IWMhjyG8Ieki
- FSJCInAinSLKIvgjJSNTI4EjryPdJAwk OiRpJJckxiT1JSQlVCWDJbMl4iYSJkImciajJtMnA
- yc0J2UnlifHJ/goKihbKI0ovijwKSIpVSmH Kbkp7CofKlIqhSq4KusrHitSK4YruivuLCIsVi
- yKLL8s9C0pLV4tky3ILf0uMy5pLp4u1C8LL0Ev dy+uL+QwGzBSMIkwwTD4MTAxZzGfMdcyDzJ
- IMoAyuDLxMyozYzOcM9U0DzRINII0vDT2NTA1ajWl Nd82GjZVNpA2yzcGN0I3fje5N/U4MTht
- OKo45jkjOWA5nTnaOhc6VDqSOs87DTtLO4k7xzwGPEQ8 gzzCPQE9QD1/Pb89/j4+Pn4+vj7+P
- z8/fz/AQABAQUCCQMRBBUFHQYhBykIMQk5CkULTQxZDWEOb Q95EIURlRKhE7EUwRXRFuEX8Rk
- BGhUbKRw5HU0eZR95II0hpSK9I9Uk7SYFJx0oOSlVKm0riSypL cUu4TABMSEyQTNhNIE1oTbF
- N+k5CToxO1U8eT2dPsU/7UEVQj1DZUSRRblG5UgRST1KaUuVTMVN8 U8hUFFRgVK1U+VVGVZJV
- 31YsVnpWx1cUV2JXsFf+WExYmljpWThZhlnVWiRadFrDWxNbY1uyXANc U1yjXPRdRF2VXeZeN
- 16JXtpfLF9+X9BgImB0YMdhGWFsYb9iEmJlYrljDGNgY7RkCGRcZLFlBWVa Za9mBGZZZq9nBG
- daZ7BoBmhcaLJpCWlfabZqDWpkarxrE2tra8NsG2xzbMttI218bdVuLm6HbuBv Om+Tb+1wR3C
- hcPtxVnGwcgtyZnLBcxxzeHPTdC90i3TndUN1oHX8dll2tncTd3B3zngreIl453lF eaN6Anpg
- er97Hnt9e9x8PHybfPt9W327fht+fH7cfz1/nn//gGCAwoEjgYWB54JJgquDDoNwg9OE NoSZh
- PyFYIXDhieGi4bvh1SHuIgdiIGI5olMibGKFop8iuKLSIuujBSMe4zijUiNr44Xjn6O5Y9N j7
- WQHZCFkO6RVpG/kiiSkZL6k2STzZQ3lKGVC5V1leCWSpa1lyCXi5f3mGKYzpk6maaaEpp+muub
- V5vEnDGcn50MnXmd555VnsOfMZ+goA+gfaDsoVuhy6I6oqqjGqOKo/qkaqTbpUylvKYupp+nEK
- eC p/SoZajYqUqpvKovqqKrFauIq/usb6zjrVaty64/rrOvKK+dsBGwh7D8sXGx57JdstOzSbO
- /tDa0 rbUktZu2EraJtwG3ebfxuGm44blaudK6S7rEuz67t7wwvKq9JL2evhm+k78Ov4nABMB/
- wPrBdsHx wm3C6cNmw+LEX8TcxVnF1sZTxtHHTsfMyErIyclHycbKRcrEy0PLwsxCzMHNQc3Bz
- kLOws9Dz8PQ RNDG0UfRyNJK0szTTtPQ1FPU1tVY1dvWXtbi12XX6dht2PHZddn62n/bA9uI3A
- 7ck90Z3Z7eJN6q 3zHft+A+4MXhTOHT4lri4uNq4/LkeuUC5YvmE+ac5yXnr+g46MLpTOnW6mD
- q6ut16//siu0W7aHu LO6470Tv0PBc8OnxdfIC8o/zHPOq9Df0xfVT9eH2b/b+94z4G/iq+Tn5
- yfpZ+uj7ePwI/Jn9Kf26 /kv+3P9uY3VydgAAAAAAAAQAAAAABQAKAA8AFAAZAB4AIwAoAC0AM
- gA3ADsAQABFAEoATwBUAFkA XgBjAGgAbQByAHcAfACBAIYAiwCQAJUAmgCfAKQAqQCuALIAtw
- C8AMEAxgDLANAA1QDaAOAA5QDq APAA9QD7AQEBBwEMARIBGAEeASUBKwExATgBPgFFAUsBUgF
- ZAWABZgFtAXUBfAGDAYoBkgGZAaEB qAGwAbgBwAHIAdAB2AHgAekB8QH6AgICCwIUAhwCJQIu
- AjcCQAJKAlMCXAJmAnACeQKDAo0ClwKh AqsCtQK/AsoC1ALfAuoC9AL/AwoDFQMgAysDNwNCA
- 00DWQNlA3ADfAOIA5QDoAOsA7kDxQPSA94D 6wP4BAQEEQQeBCwEOQRGBFQEYQRvBHwEigSYBK
- YEtATCBNEE3wTuBPwFCwUaBSgFNwVHBVYFZQV0 BYQFkwWjBbMFwwXTBeMF8wYDBhQGJAY1BkU
- GVgZnBngGiQaaBqsGvQbOBuAG8gcDBxUHJwc5B0wH XgdwB4MHlgeoB7sHzgfhB/QICAgbCC8I
- QghWCGoIfgiSCKYIugjOCOMI9wkMCSEJNglLCWAJdQmK CaAJtQnLCeAJ9goMCiIKOQpPCmUKf
- AqSCqkKwArXCu4LBQsdCzQLSwtjC3sLkwurC8ML2wvzDAwM JAw9DFYMbgyHDKEMugzTDO0NBg
- 0gDToNVA1uDYgNog28DdcN8Q4MDicOQg5dDngOkw6vDsoO5g8C Dx4POg9WD3IPjg+rD8gP5BA
- BEB4QOxBYEHYQkxCxEM4Q7BEKESgRRhFkEYMRoRHAEd8R/hIdEjwS WxJ6EpoSuRLZEvkTGRM5
- E1kTehOaE7sT2xP8FB0UPhRfFIEUohTEFOUVBxUpFUsVbRWQFbIV1RX3 FhoWPRZgFoMWpxbKF
- u4XEhc1F1kXfReiF8YX6hgPGDQYWRh9GKMYyBjtGRMZOBleGYQZqhnQGfYa HRpDGmoakBq3Gt
- 4bBhstG1QbfBujG8sb8xwbHEMcbByUHL0c5h0OHTcdYB2KHbMd3R4GHjAeWh6E Hq4e2B8DHy0
- fWB+DH64f2SAEIDAgWyCHILMg3iEKITchYyGPIbwh6SIVIkIicCKdIsoi+CMlI1Mj gSOvI90k
- DCQ6JGkklyTGJPUlJCVUJYMlsyXiJhImQiZyJqMm0ycDJzQnZSeWJ8cn+CgqKFsojSi+ KPApI
- ilVKYcpuSnsKh8qUiqFKrgq6yseK1Irhiu6K+4sIixWLIosvyz0LSktXi2TLcgt/S4zLmku ni
- 7ULwsvQS93L64v5DAbMFIwiTDBMPgxMDFnMZ8x1zIPMkgygDK4MvEzKjNjM5wz1TQPNEg0gjS8
- NPY1MDVqNaU13zYaNlU2kDbLNwY3Qjd+N7k39TgxOG04qjjmOSM5YDmdOdo6FzpUOpI6zzsNO0
- s7 iTvHPAY8RDyDPMI9AT1APX89vz3+Pj4+fj6+Pv4/Pz9/P8BAAEBBQIJAxEEFQUdBiEHKQgx
- CTkKR QtNDFkNYQ5tD3kQhRGVEqETsRTBFdEW4RfxGQEaFRspHDkdTR5lH3kgjSGlIr0j1STtJ
- gUnHSg5K VUqbSuJLKktxS7hMAExITJBM2E0gTWhNsU36TkJOjE7VTx5PZ0+xT/tQRVCPUNlRJ
- FFuUblSBFJP UppS5VMxU3xTyFQUVGBUrVT5VUZVklXfVixWelbHVxRXYlewV/5YTFiaWOlZOF
- mGWdVaJFp0WsNb E1tjW7JcA1xTXKNc9F1EXZVd5l43Xole2l8sX35f0GAiYHRgx2EZYWxhv2I
- SYmViuWMMY2BjtGQI ZFxksWUFZVplr2YEZllmr2cEZ1pnsGgGaFxosmkJaV9ptmoNamRqvGsT
- a2trw2wbbHNsy20jbXxt 1W4ubodu4G86b5Nv7XBHcKFw+3FWcbByC3JmcsFzHHN4c9N0L3SLd
- Od1Q3Wgdfx2WXa2dxN3cHfO eCt4iXjneUV5o3oCemB6v3see3173Hw8fJt8+31bfbt+G358ft
- x/PX+ef/+AYIDCgSOBhYHngkmC q4MOg3CD04Q2hJmE/IVghcOGJ4aLhu+HVIe4iB2IgYjmiUy
- JsYoWinyK4otIi66MFIx7jOKNSI2v jheOfo7lj02PtZAdkIWQ7pFWkb+SKJKRkvqTZJPNlDeU
- oZULlXWV4JZKlrWXIJeLl/eYYpjOmTqZ ppoSmn6a65tXm8ScMZyfnQydeZ3nnlWew58xn6CgD
- 6B9oOyhW6HLojqiqqMao4qj+qRqpNulTKW8 pi6mn6cQp4Kn9KhlqNipSqm8qi+qoqsVq4ir+6
- xvrOOtVq3Lrj+us68or52wEbCHsPyxcbHnsl2y 07NJs7+0NrSttSS1m7YStom3Abd5t/G4abj
- huVq50rpLusS7Pru3vDC8qr0kvZ6+Gb6Tvw6/icAE wH/A+sF2wfHCbcLpw2bD4sRfxNzFWcXW
- xlPG0cdOx8zISsjJyUfJxspFysTLQ8vCzELMwc1BzcHO Qs7Cz0PPw9BE0MbRR9HI0krSzNNO0
- 9DUU9TW1VjV29Ze1uLXZdfp2G3Y8dl12fraf9sD24jcDtyT 3Rndnt4k3qrfMd+34D7gxeFM4d
- PiWuLi42rj8uR65QLli+YT5pznJeev6DjowulM6dbqYOrq63Xr /+yK7Rbtoe4s7rjvRO/Q8Fz
- w6fF18gLyj/Mc86r0N/TF9VP14fZv9v73jPgb+Kr5OfnJ+ln66Pt4 /Aj8mf0p/br+S/7c/25w
- YXJhAAAAAAADAAAAAmZmAADypwAADVkAABPQAAALA3BhcmEAAAAAAAMA AAACZmYAAPKnAAANW
- QAAE9AAAAsDcGFyYQAAAAAAAwAAAAJmZgAA8qcAAA1ZAAAT0AAACwN2Y2d0 AAAAAAAAAAEAAQ
- AAAAAAAAABAAAAAQAAAAAAAAABAAAAAQAAAAAAAAABAABuZGluAAAAAAAAADAA AKPAAABXwAA
- ASsAAAJ5AAAAlQAAAEwAAAFBAAABUQAACMzMAAjMzAAIzM2Rlc2MAAAAAAAAACkNp bmVtYSBI
- RAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA AAAAA
- AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABtbHVjAAAAAAAAAAEAAAAMZW5VUwAAABIAAAAc AE
- MAaQBuAGUAbQBhACAASABEAABtbW9kAAAAAAAABhAAAJIjAgAqqcBCT4AAAAAAAAAAAAAAAAAA
- AAAAdGV4dAAAAABDb3B5cmlnaHQgQXBwbGUsIEluYy4sIDIwMTAA/+EAQEV4aWYAAE1NACoAAA
- AI AAGHaQAEAAAAAQAAABoAAAAAAAKgAgAEAAAAAQAAAoCgAwAEAAAAAQAAAeAAAAAA/9sAQwA
- CAgIC AgECAgICAgICAwMGBAMDAwMHBQUEBggHCAgIBwgICQoNCwkJDAoICAsPCwwNDg4ODgkL
- EBEPDhEN Dg4O/9sAQwECAgIDAwMGBAQGDgkICQ4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4OD
- g4ODg4ODg4O Dg4ODg4ODg4ODg4ODg4O/8AAEQgB4AKAAwEiAAIRAQMRAf/EAB8AAAEFAQEBAQ
- EBAAAAAAAAAAAB AgMEBQYHCAkKC//EALUQAAIBAwMCBAMFBQQEAAABfQECAwAEEQUSITFBBhN
- RYQcicRQygZGhCCNC scEVUtHwJDNicoIJChYXGBkaJSYnKCkqNDU2Nzg5OkNERUZHSElKU1RV
- VldYWVpjZGVmZ2hpanN0 dXZ3eHl6g4SFhoeIiYqSk5SVlpeYmZqio6Slpqeoqaqys7S1tre4u
- brCw8TFxsfIycrS09TV1tfY 2drh4uPk5ebn6Onq8fLz9PX29/j5+v/EAB8BAAMBAQEBAQEBAQ
- EAAAAAAAABAgMEBQYHCAkKC//E ALURAAIBAgQEAwQHBQQEAAECdwABAgMRBAUhMQYSQVEHYXE
- TIjKBCBRCkaGxwQkjM1LwFWJy0QoW JDThJfEXGBkaJicoKSo1Njc4OTpDREVGR0hJSlNUVVZX
- WFlaY2RlZmdoaWpzdHV2d3h5eoKDhIWG h4iJipKTlJWWl5iZmqKjpKWmp6ipqrKztLW2t7i5u
- sLDxMXGx8jJytLT1NXW19jZ2uLj5OXm5+jp 6vLz9PX29/j5+v/aAAwDAQACEQMRAD8A+To438
- n5HVipGCR1q/Em6JiwOc9PrUEKbY8KCd3SrsUZ UfNk7sZA9ayje2jCbjHboWoh5gIJCsp71PH
- ErQltrDcfXrTQvz4I4zwAeasLE8bcA4HTNNU07u+o oKLd0yaJFEWQCGwCD6CtJBFsOCSx6nNV
- wuxE+Q5HC59KsxxuACNufUjpWjp33ZLm0TwpsdgnO3HG fWrSjEnG7nr+VQRIWYqQ3J4q7GMNn
- OCTilaNxzaT1Y+NWSJQw6Lipo13RM3pgfjUa5RzuO5asxKi kEkYA6VqtI+pLaWhZZRJFGDlyM
- kY71MgUIecgjn61HbguCjoyg9+lSxoVbDHbnkkjpWaj0fQldri g52+4yaFj2y8AqvbmlCls4O
- DnAJ9Ksqylxxz0Fa81mrIt3jHTqKkW/DZ5PJxUgQhmByXBz+lAAUA k9D2qcbAFZTuPcntUOUm
- 7Iq17SYigNGoYbiF7UhGJfQAZz2zUiliduRwdtTkYQhSFXd1NDk927lR Scit85kG3gnHFLhwA
- FHOc/Wp4wNuVHHc1IACxB7D16VE2+a6QX3uiEJlD8pyTkj0p2394QQSQMYF SnOSMhe3Sm7Cjq
- mecck1fNeOpMoprRgFOCOetOcMSrHg9elPAxIN2MHkcdqcRltuDg85rG0m9EFu zI/LRWwAScZ
- Jz1qdYQ6EkYZe1KiFnG0qPXNPwSQpJI7Y705yfRltSe5XdHLbu2enpT1HzorHOBjH tUmxhkkP
- j+VTJGwYFgNp6k9qXncIzTViAxgAjsPU0Kn4euas+WcOx4GaBHk7sL1GM96q65Xd3KjF W0ZFj
- aSQQ425NSrGSARwTkA9s0qRqCSenQVZCrsO0MOelRKSTSSM3q9Csy5Us3LnH51IYyYs5zg8 U4
- JyTtJwanCsEwMHDZxjkD3q4JNle8rscgySOM/xe9O2sMdTjJx9KmSJt7FgASevtT1j3MCucg9/
- SpcnfVGcVGz1K5A3K5545+tChshh1PUmrYjUxlQCy57elBSQPwowDjmm9VuNP3WV+EYEhsZwD7
- 1J 5J8hPmBPU+9WFBEhLAcgHkd6MFQvTI45PQVV9dUOzbuiqy7QY14wc59qV41LYHJz61Oqqck
- 5Ck55 70BAH3N+B9aXMloWm4+RD5fBU8Z70/CGIIRkgCpH+UZYZUd6REClRz64PUVUWum5Ls3e
- 40KOWPJJ ySO1O2ARk7SecjHrUx4G7aCuc1IIt5G3PqwpXfVA1FSRXXPkruO3PApArlxhgFAw1
- WOA4UgLzyT/ ACppUuu3B2selWvd3RDTTd1Yr/OZM54PU+1QqDu5OwDocdauldockfTNQAttBl
- AKdiB1pdW2i3Hm V0tCA9HwDgNjFVHAM24Da2CRn0q80eWPGFIz9DVdo+u75h0yKiN4PRgrRV7
- 3KgDbl24B5xnvUbB9 x2jAH86uBBgsM5XvnpUBDEE7lJ7itLoEkiIKwO5Dgn1HWlbywxDk5HQ+
- 9TE7odqDaucjPUVGw+Ut wxPYCsVBbBy8xG65jcHJycDHeq4j3LuHAA+6fX1q9jeMgDJ6HFQbS
- YsZ6frXRBNJ3JbtuisBiTK8 8jj3pXZyjsOCCOoqfaDKCo+975pxVhHtwAMYJNZ631FzK+xS2h
- gxYcqvr1qIYDbiCMj86uGPDBWH TuO9Q+UCgC/fz1NaXhs2a+0SVkQBR5me5/Wo2CkMGyvHU/W
- rRTM23DZzyewprKwKbgSMdMVlzLWx GiaZWZjgncMdAKh2MZPu4wME5qw6cL8v3RSNteQqnUD5
- qSduhUndaFJ0zDg9R1xUBjDJtwdpXIIq 4yb3L7H9wTTHXbBtTHzdCaq/K77icVoomeYiybHO0
- d6rSJt6EEdiexrTKtxsXLY5Paq7RMJgSAV9 AO9aRloxwhzbsoup2LzkHjFV5EJCg4Cgg5J71o
- vHvZQT8oyenJqJok3spy2RkUmna19yNFKyZmkF Qp77TkfWmYAUKwK854rQ8tGiBLLnuPU01kX
- G/cCc8nHFRey1KbVrWM4oPvAhWB4J7etV2RfkUckN lauunm4JAHJOMVXlj2gEHoM5qKcYq/mV
- Dkdmys20naecZGRVcqQAduATyPStJAB8q7Xz6dxULsm0 gRk49T1pTcl0M5rXRGaYwqAschhgj
- pz2pohVSH5DAYPPerk4Aj2k8KPkJqMqhVsjA+8aykoyV1qy KrbdmcPbIxtyuVDk8fnU6rIGwd
- ofkdO3rTIInVztB4HrV6AMW3Pznnnr9K0Ti/e6lxS2JI0Cpuxu I6Cre1WXBIyM8jtUaBWcFM4
- PcnirO35gvOSMgY61cZrZjcHGWwi7iuCcj+Ee1X9hXaW4GOcnvUSq DtAIYDgEVbRCBubJwcVS
- kk77EX5unyFQnapVjwfWrigjkHDcHJ5qFVIjD4G0jgkd6srGGm5IU4ye a05+idg93axKCN6kr
- vBPIFWU2iR8grn7opIIwTtDAgc4qYRESj5DhelJVOgRkh6K5X5uFHT3NWM5 D56ds0kSkCMEdu
- hqVEEm4ZAb+lNpLV7DhK716CAgAZIXb6ipCSUYfx54I9KcEGdzfMO3vUwUsQdp 9zUOSvciDs7
- iRhfJAI3AEDOeasEqVZQhB3YzTFXksFOMAjPQ1KuWl5GOetOUrlNSXoBQHaOjcfnU ka5U8YHb
- PenlOeDnjipowcruCliMAAVkqivoTKWtyNFwrZHy+3rTRlnYpww9RVlk25yehAxTdv8A dGQD0
- qlO+xoprls9xrMCB8wyf0pCuWHfHenCPMuzGOeuKmSM7kBIZTyfX6VF9GPVOxCvLZPOD27V KE
- bIJypB4PtTyquWCgkcdOtSohG35WbB4FKLk7hLyIwrGU7umck461OqnIBABxxgUoV15QBhyRxU
- 6eYeoyMfLxRe1rpBza3bsQZQjcfu9TUygsrkKewB7U7CBcbcZOOaeyB9wG4ZYdDilLlRLV43aI
- TG uPmJ6cD0p21sBODjgcdqnxmPIU8cD6UuTtw2D6cYzQ46FySk1cjCfu8MMgHjFAQlNoGVz1q
- 4qghS VyT2oC4QYVjk5JHalC9rijOMVpoVkUhyQC4x0FWQB5bYG3d1Y/yp+z+JAct/KpnX5Azj
- bz0NNTaY k4vcr7ZNoyQwHT61NtO44OM/p7U5doVdx5IPSpgCEAGCTxWmluxUdFqyMAhcHO3GK
- cSfJORkAjp6 U9cFgMH/AGs96fszGV2n1xUzXSwKdyIE7GUjjqPWmMMjoTk8+9TjG7ZyzY5FLt
- 3KNwwR29KXw9Ac bNELITCVOGbsQKaAxkI6AdzVkgliFU7R3pdqm4YlSoOMZp82gcyk7EGNyjc
- QwxggU0A+aXCYY+tW lTO3b3PXFKEKJwCcVSfvWsVztIgbe23cPy70/Y2CBnPXr0qVfmBVl4Jy
- MU9IwAR0HbJ6Ucz6IzTa 1kyLYMDncWHTHU0NkAKFxjjOOlP2Ethn4HpSAbnz144BPNNy1CMdd
- dSsWy43jdj26UhH948jsBUp QtIwxtIPT0pWX94GUHO386huI3Ho0UXjV3ZS5Uk5AzTRGFkJOc
- dvarGBvxgk7eD6VHs3RKTuDZwc 96J3ve5bTatcpMSrPgD73PFJJHnHG47cjjvVh4w82GO3NN2
- lmbkbRwvrRGpZak8qbK4RdoPc56U1 Yssc8Lj8an2FWA2kDrmjaSJF7Acgd6qE2nZAoroVzkRh
- MHryR601lLK/RecdKsbVQhs5BOMZppUD LDoWyM/zpuUn5lUuVN6FFDhsgcKecUOQVODhifu1b
- KjcSBt9TUJQqRlcE8Z60m1fUUpq2xAQxG4Y yM5yKiY8jcV6YOBirexgvIyvUkVXKhotuNp7E+
- 9J27ExlddyJkzGqA8g4+opgQ+YTuyoOfc1MQm/ axIIXtSIqHDchgp4zS5ujJcXazKnl43NnIL
- fjTCWBJ+UZzxire0OMgYXnPNRMpZQNpx9KtSuh3Kp QH0BA4Gar/NuJVlK5wAfSr5HAJwQOw6i
- oiBsJI2nODx1pqL6FOS27lRlj3ZUnntmogqtuyCBwTV4 qBDwPusB7moggxgMApOSKz5o3sDir
- XKJiGAD/A3B9cVG4zCCAADkZx61oFi2F2NtyeSKg6wfKowD 0I6+9a03fWRn71ym0YEW3AyOp9
- aq7Au4ckdh6f8A160Xj/dDdnPbNQeSCDuJIz69TVWuro0VmUHC 7AuGG3pVfyQ1xt3YBB4NaDQ
- 7Mt1LHHNRybWK7mULjhhwBUPmbuiZtRdkzLwF2quA5GR9KrvG29lY ckdfStJkJj3BVYLxkVCV
- dUBxuycnik5a3BN82hQ2c/N8317VAY2JZuMY45q+6b0Jyc+npVRl2Abu F6hu2ayjPmlpv6BNy
- scWkR3jy2LAnB9quLjesZZX+nGDTY1jLkYZT6E9asDHnMccqeKelryZTjL1 FjYKFB7cYq2u1S
- u70xn29KYqLlTtPIz06VONznDHCYHUd61cby00RKcXezJgQASANuMgDrU6FimE yVwBmmIMbVy
- B2JI4qwmVOM454GOoqXZIlOUVoTJhjtY8A4/GrCqC+FPze/NV0CEMNxGfSrKAL8oy W65PaiKT
- WzE27WluWgxVtnG4kc46+1Wgrn7o4zwT2qqh+QO3UNjOPWrhP7kBWUELzSu001oRy20S J0VVG
- 7JYHoKkjAAdz8pNNjKsFVVZiF69qVmDBAM5x2rSMeZe87lXT0JFXmMs4GB/KnLkRhgxyR2/ Wl
- 2guoxyeeakj27UGQp64PaovJO6Qk3HbUWP7/Xv8ue9TkOzPtA4PpQEBnDg8Dr6GpSPkLA8Yxij
- nTZq7qwJnYA+SSOwxzUwVlkG44OM/SowMptTjB6mpwoZSTnI7Z7VTb6iWj2H5wDlctjIPpTRyw
- zn 3xT1RSRklcdc1IFCsGzjI9O9ZxUNbAnbWxXCKThSR3GT0zU5XaVXOWPp3pwX93uCk/NyalR
- RyWyp BOM9jROS66E8zvoNU4I+Vl55JqULnc/IYniiMeZHuHQHk9jT1x5mARzzjHIp8umxq3ro
- PXBVQMgj v2qQLu3jBUfWpPLOVw67SvI9akXaRxlm74PWnLTUjpdkIiHnsDkgHIqdFYxFmVcjv
- ilCsMSLxz1N SspDc4xjJFQotyuLV69CGMkPyOnftineWwkIC5FPKDY69SDSKjFzjKgf3jRbRt
- sbgl5Cbdy4YEYA BqTaCjIOdvB9alVG8xsLlcYA9akCYIAOecDNPmu97FJx7kYwD042kcdqVc7
- djfMAOB3p3zbgOPfj rUobLDK9tpx2pRi7aohNX1GgKAoIB9Tj9Kbty6gZHGf/AK1WNoQhscA0
- 9VHILA5746VVkXGViEr8 4VO1WAFADt0xg+1TBQCCRnjOaDGrx/KOSOM9KhN3DmVyodyvlRkg5
- 6cmnsN8gYqykjBx61N5W5lH LEccdTStGQQAhD9GJqrpyDroQfO0ez15/AUpOTt2HINSiLPAzu
- we9PIIBADZI4/xofI1sNMhKYkB /g6/So1VPMU/MCw/vVcyWhIBDY4IqIgAqMHjP+RUwV2xvfc
- RP9Y3QkYwaXG+ZW3Acf8A6qGUlAFw CD940Z287d+ey0cvYjldrpjH+/k/dJzSIcHd/CR8pqZQ
- Cxdl3Hdzjpmoir8jrk4A9PWtFcE00MIA BLN8xP6VEdwZW4wPxqc4Uj5fl7Z7VEyjcSu7BPU9A
- aUtgcrtXehCQxcE460wuRKqEdRgVOR84Bx0 /P3prYAUYDY6kVCk7lOGlyFolCbm5btUBBVs42
- qvT6mpMNwwbryBUwAywyCSfyrR3jZkxauVAoK4 bOM8HNC4yybfmPUipNuUAYZOfzoCsZAOAev
- /ANamo33L+FXW5WZFEhHX0BpxXzHcgDdx19BUzEbi WXBJPHeo0UtJvCsgHBU+lTL4dTOfe5Xd
- B5pZuMHIUd6iYZbvgkVbkG1TznHf0qFkzI46D+92qeZ7 iSRWcZn2nIbODzTCpGSMHB61YZf3b
- ZGAO+eTVd4yA2MnPvQk927DVlKxG6jBJwcnk+lQmMFTjJA7 9KlkBWLnO3pinEfIQpHT8qpNtI
- NU79Cru2odvA6NSMjcBTkY61JtJhDNjlhkAdKeU+UgOFbPJP8A KrejuOL10RVK7IyGI64B9Kj
- 2Ah9wPTKk1O0bFgjnbu6d6btAdhtLc4P+NLRa9yb62bKm3EQDA4xk ketMaM43Y/i6Y5FW2UHG
- Bmo34PAy9KU7i1asirIGKY7kjgVEQSp3HbjjPqKsSKQTj0BNMPzbsYLd 89Kq+qsVfuQk5YFun
- QAjioyp2KQDyc1aZThkG3JHSo/4FTBOD8xH6UlG6egnLQolN2dvQDnNVGBC 4AUg9citd1YLjK
- nJxgcVUkiUQMG4YccU4ya06Cg0tWZ3lLu+X7uM4BqrInzttcH6VpHcG/2T271W aMoF55Kk+9S
- k3rsaK6bszPkiITczAFuQPQVUMSlPmOQvH1rSePMALZLH37VWkiXJ2HaOvNVzWRPN b1OQ4YLt
- VTjrxU8cQMiylcDrnHenIvzljjcTgAd6nXIjYN0B5XuKxTd9URHR3S3ARE5wAARmpo1Q hiQ2R
- j5T1BpQF25DfLwQfapFXDuFBwTz64rW11ZsfM7bCoMHOCTjn61OpUv8yjJ6EdqbyoI5JPPW pl
- GG46EdamMW90Wo6PUkCAN/CBnJOKsKp2fKC7dGqNYgmCT1NXIwPIC9VPP4+lVTcWmupM5K2jJI
- 1O1QBwTnj6VZjj2jcWyccj3qJUwqYJXnvVlVUBSCBk9D2qV7quZubY5RkEDIAHGKlVeMbSpzzU
- aq AQc4GOBVhAH4JwelP3+UpxaW+g+NldSDw+MEU5QWlBAx2xjrTEVftDY5I61Nyr5IJPt0pcj
- TLj7z 0JkY45IBI6YqQr8gHUDpxUaIABtB5PP1q7tGPccYNS0oscFyvQYnzH5cZPTFTKORwRxw
- PWkC5BOM g8DFSrncFPUcim1bUUnfW4g+VMhSc9c1IY8y7ucA9DQY9/GTz3HSplAEOQDknvT3W
- gruyY1FyhLn AHIx61LtBjDf7XNNCtySMZ6gU5cnae/92okubqNJLYkCbVIGMY605OeSAPl60r
- DYM9MYxzTwMvgH APX0ojZ7scZN6MlVCyKXGAv4VKFUSsSQBnK+4qJA7Bg2Qo6L6fWrqrGdxUb
- l449KblyqzYWdvIjV RvwPnDcn8KmweODgccinEYlAwMdqUoTIoLdOabWt2Jt3syHbmTcOucED
- uaVtoVQ3BPGTUxjIz6hs 5qTblG+Xjd+lU5K1xrTV7FYJljlzgHOe1TbeSVVs49e9O4JcsMKT0
- FSKQrjB4789awblbRAoys2Q KSrD5Tk8gnpVhUAKkLuGCTShN69yw6Uqn5ioBU9+atJuzsN2+H
- uC5VyzggehqVETaBtzz69RTkiM krPj5SvQ84qRlOQxI2jsKeknbqNWTsmO2AH3HamfMpXcBnG
- Tx0qTkwED0zn0pu3OCRnjinFa6kNN qzDsc8DI5pwB35Jy3b3p3BTIBA6nNAQl+CM96hrW76By
- WtYjA+YKpw3XB600LIsvIOO4P6VM0Yxu AO49TT8HA7GqUU3ce22hCMkDPDdzSYUjef8AgIqXy
- y2Rnae9DR7EHoOopJR72BRs1Z7kecJxt56i onGEOB8zDtUyoAvBOKQ7BKvXJAzzTh1sCa2K4f
- BVf4R0A/nTF5OC+M9qtMIyWO3B6A1CF3bcA8Lx xThNrQJLaxExHmE9RxgetM25cAErg8ZqfGE
- IYYbFRH5SgK5yOPXNTzeYN3diEjAOBnLcYprIRIVY YGOtT7S67sYYjcaZ5TYJIOc4BpJWepLT
- tbYYRmIKFI3dKhIw6jnuM9quKH/dgjgc5xUG3Mm/rk/d x0q0+bQtVJR0RFyGVSQcDgUMAXkJU
- gk5OP5VPtDOcckHtUcyEoSAVJPA9Kq8X6kc3NIr7cyAMDgj jHtRkrICSAT7VMuPkJyG757Gkb
- PJK5J6YpO70YSd20VHUvHu3AnPboajbCOFOArHBq2w27VAHTp6 UEApjAPf6VU2r2HotEUnU7m
- Kjfnt6VGwcnbt2kDnjp71aIDMVOdxOQe2Ka6kZ+916VLp3KXLzLuZ +Nr84Y+9H7sx4XhzyRVn
- y9z/ACglsY3f1qLYBDjGCT1o91oLK+4xwTGGjAbPUCoMZi+YgOOntUxy nVie4x2pzKu7jJDHG
- aHZLREu97FSSNmdVLDdnAIFMbchCEAsBzVqRS0eSOfT+tRTZMTv6kA4pqVk KMddSmR8wDZwR8
- xqPgP0zxx71ZCKpyQcBe571AB1yRuHFVKSauXBLUjI5BK4Lc4PVaj2neVcbV6H AqVVLMcfNxy
- TTSDuBJ68GpgrPUma7kZCMDjIG7hqgWT5dp5OCOP51aZVdODx6VF5ahSxI3H5sY7V cWr+YrNL
- QiK8DeOT0AqHY2/Zt4znJqwSx9MY4JFRYYHrhRyfrR7NpPQUm7FKdRvHXKjk+tVH+b7o IbsfQ
- VpMv7sBgOhzkVAy/IW4AwO1EZpoqLcUrGbJBjb1YdiOlVJUGG2Ho2AOta7BQW6nB5qjKWZA Rh
- cHJOOvvSjNteRK35n+Jx2zLkFgB1x3qwvlvCSDknuPT3pkKctv+ZmH5VbWFfKP8K+1S3Hlu0VT
- UktBiKcxqMHaMfWpxuMhJGMcYNBVdyYyScZ9qlCjJOfl3Zya0m9UiHT1fMJ/GyhSnOSakRQck5
- JH 8IPSnCIhiSmSGxmpYgysWABOemKlRvKyKUktiVUVlIySOuM81NHGpmj+96gg96EXeylDjHH
- Tp9au Ip3cYP0p8zQRu9ESAHY2SBhh19KeFbaVHzKCOlNwjxkMD8w5we9Tqu3Z8hPfg011E2kt
- h3lg4JyA Dz7VKuQcfLkHPPYUKMyZJ2jqM96kCq4BAZiTyajTVMvVR1YKpbBHIz27VaWMKvJye
- 3vUIiKybRke v+NWgCFBPJxzTlJP4SVGSSdxIwjOWyRnselWE+Y8fNkcUxAxVTtB+g6Va8sYIT
- txnNRdKSTKbYCJ ieMrgjgVKsZYNjIG7IzzRGDu4BG08GpVXMiu+evABpzk2LXdkUauHOCMHqK
- mUbZguCR0JPalYEKw PUdcCnxgNEzEYI9TWU00hWd7oQJ8mCc4HBqQIqKpzu29/Q0oBOFwATzU
- xBCMduOeRUtONrlNydk7 EcQDIykEtnj29qmaPJXAK4680KOFbBJI7elPCbo1wduFJwfSq91dC
- rPo7Eq/eUBflxxUwABOcgf1 pqACFN5UsecipkAMg3Hg88UoyVr2JVm7Dgm9t3UEZFP2hFBJxu
- 55pFB3gKrFgOlPKMV3N3PB7VTv a7GoJvYRow+5lYDP5mlXIZOp4pdjBFUtlRnGKkERBBIbnt3
- FO91ua83LFJ7BtT5jwBnFO2jPsB19 KkCkgLtO3GfrSgEvgkAEdMcmkrcy0FGzW+gxQCoKgtzj
- ilwROARySefapgm04UfNnnFKfvqSvXpU 3voTe70/EWNZd3P44qRwfOAAHPUYpEVmkYDpnjH6i
- pwrB9nU4x9KmWjJu1Ig8vLAFTg9DninKCV2 gd+CalUk8A/MTmpFIbKt1rbm01RTjK9yAAyOfu
- qRwB604ghAuMMcGpU6kDGD3I5FIqfMWJwVOMno aj3t0hTjeRCsZXJbGPrSYPmYP3SMZqcbiow
- uQvf8aUhCVycEH7o61oovqGlyAwhFzye3JpPvNjnb jg1Ic/eLA59+lIMeaG2nG39ahxa3KbaV
- 2RqpPzbhjBycVEUxGoI5PB561YOzgdCOvvTijMSy8qDk AGs3U5CFzXuykw7DnHQe1NHCHjBzz
- U+MlQwwCOPemMm98DGM9a1TuhtQZAdzEuudpOee1K8ZIjA4 GcZqVdzPtBG0j06CkMTAbVy2Dk
- k0ot37F1IrS7IGDrnHQjFOKtt4BJH8qs+WPLKnOexzTFVlQKCc k9x0FJz0uZOloVlGZc5xx0P
- WmlNrLkH1B9qteWeD0JGaULx1DN2HpVRWliZR5VdbFXauQAOvJwaJ NocgDdnpk/pVp4gTkgjn
- gd6Ro2YLhQB6nnmnyor3U1YoFPn5ZeuAaYYzgAjLDg81om3KgMSOpxgV EFBfKc84PPSn7ttAU
- le5RcEsdq/IMc1AEZpGypxnj3960XSQEkqcZwABTGDIuSuG7E96b006im2m upRMYDE49vrTcE
- 5bBzjvVzbwCwJpCpCBQMjHQ1jJNNXHvdGeRj5lznOPlqNVUMhUqX27SDVpk53Y YAVDJEzSDkK
- DVJSW5KtaxTkwBgEE9+KjI4AwTkVZljjyfm+YDGPemmPKKB/31VruXzqBUGCp2ggA bagkTajD
- kAt69/SrTnAcFW4P51WkHAbady9vWkpMKl3a5G2DHs7tzioZFbzNoAHPYfrUrhjhlO1Q e9MZR
- 5rLuJY9MdqqTWmolFpX6EGMZAYDuD60MCVBf0z9aTB3KpHzECjDAEMQCtJQ1LuyMhsZUcdO el
- JsUplWAYDGPUVKcHpjgcj1NLjMK7gFbkkdOKpS+8ht21Kzrh/7xxx6YpOAuOhz0xUuR5YGRkcH
- im7kGSfWhQbQlq7MpFSZNxOBjvUbkBdxGQTzV0puk44P9KqyREbSMYBPXvWnkJK7V9ikwwrDac
- no KoyRjA7FupNarRsAxIJdT+BqnIoHJwxNYqd37o2r2OSVAH4GTjC+9SiIlP4hgYPPWiNd77O
- hJzU6 AeYcnkeveiEWndijKXzGoCHGep5H0qdAAEfHBHBqRFH3mU4fJ4pUA2hdp5Hr0q/djqhR
- ldXGkjyj jOSct7VYgwfmZl4OD6Gh1Uop2MrVaSNBGcDHPU9DRNqMdQ91LUBh5o3UjgnIq0oKH
- LL8nG0+ntUK Bvk3LtYrk8dCKuqSQC/3OpBpJ6aIbTi7t7iCE+XkcEHOB6VNGgx0ZVbqSc0Ko8
- xtpLcYODVlVxDw 2OOQR0qm77j1T1ZGisFYZQgcDAqzHt2lMYx/OnIMrx2qxGAJMbcEc896xem
- w48sVdoYqc8DIIx15 pQgXDEcHg896kMbs8gzhQePpUmzKkcsMnJPrVQlpZDjo9BYlcxsqgZHT
- PcetWUXy4+oJzSxx4hQn IYggCp0T5cBSXHOaiSbIkuXYYiK7cH5icnBqQja4KA4VtuMVKNq4b
- GWyMYqRkPmAg9fmxVySSByf qVwuGJbI78+lPXaTtwMZ/KnbSzgYYnOfrTRw6sw+XoR3xWerRV
- lF3e5MFKjOw4HJz1p+A8oySccq PX60IPlDBWwTzk9KaquFJXnBwPU1Lbej0Jv5jWB8racx5xg
- +gqeMkZx82FI3evtTQvmJgttbpzTx Gylg2CMYAFOyvqbuCfUkiX92rZGO6+lW1TIwQduetQxD
- 5WVvbOfSreAm0cgYzyaOfo2TNxi9NRMv t447fWpduIvusefu55FQ5YMDj3z1Bq0qsTubPsBVq
- KWz0Ks762EjjBwwPc5zUvJcAggAZ3HtQPvk 8BfSpQrllHXNS7bszlve43nb6+9TKp2EgBsDOA
- OaRYz9OM4zV+9vZb65SaZY1YRqg8pNowPp396i bk9LaepajzPRmfg5BKnlue1TMmZcAHJ5APp
- T+NoCdz370pG1x/Hx2pqLbuK1t9xgRkXeCCCecVN5 b+Q/UH1oQMEVWUk54o2s0yqrMuBg5PU1
- pZ6CteWoKo3biMBTyaVTmMhyFOPSgxs0PzffHWpApxyM Y4FTy8q3KlFNakQiJYdQQeT/ACqRB
- hQuQMg4BGakAyoyOB3po5OSmMjr7VTTb1Rnz31vcRQNmRyp 600xqJ+clgORnmpAMoB1Q8ijaD
- 8zHH+16099UNR1uV2QbuhHp/jQFXaC/BwenFWFUFJG5PHY1Hsd WBbgEdetZ7p3ZUrIiCEZC9y
- MsfaoWOZCMhc9fSrbgEsGVjxke9QDbnLAL8vfvRG27VxKz2IlTOD0 471GUUkHIABq0QGTABPP
- H+FIsW4LkHimhQ5lqyExlV29s5yKcFcENwDg9qlGPPK5G0E4HrQYyeit nPXNXSWqvsZ2VyFjn
- eo4bIIzT1jTbvY4PYetTGIh8hRtH51IqEE5TaeM0pRXQ1bKeHIPy5xx0p6R F2Bx1HDdqvqik5
- cbDg/Q0rRhIDg4GOtQ5JWJk/spbmf5Q3Hb3PGaaqDewPYY/Grvl7lDZAyMA4pp UGfnhh7VUUK
- a7lVe7MM7fWqzrlchcAHJxWg0YGdoJWhYSVK4ySMnPtWqilqNNJ7FBl+fAzzVdoix GQSoHb1r
- U8k5Axs5PXtUHlPg8fMvUe9C0d0yYyd7GcIct3z6VE8e5dwyW9PSr5+7u2kjdULJzuCk HJyPW
- p13Zo7bGcVLSsM/MePpUEifeHBx1NX3Tjbzu/vYqCQARAEEsfQ1knF7MUU07FCRG3AhR24x z7
- 1BJGBvYZX6mtArlivO4HNUpPvsCOR3q1JPboHNLmKkoUx7sbiRyAagkGYgeSw4xVvZhRgZbHT2
- qsy7mIH8XQ0Jx7g+V9SEgDkc+tRBSCHHHPGe9Tsq5ywO7HrUHP8AF8wWko3ZCiloiJgGdjkfSg
- nI baARxjingMYCP4euMc0mEIB3gY+8K1lBRHzJ6kIUs2NuHz29aCrdcZI4/OnHGVfJPGBj+dO
- KEqVA IyecnvTSW9ynOzTSGPGViAwCOnHWoXTbg5Hr9atlGVCHXPriq7cI2MZDDnqKE3bUUOaW
- xHtJTBHI NV5CMgEgHkirjcy/eBJ64qPYm8qCG5y2am9tQta6KbqcbsZz1xWdKFXLMpJA4wa0Z
- G/dfM2FJzmq L7TMSGAX0PUirjFv3rCglbU5gI4j4252nGBT448oqgdfzpVJVmABGehNTopfGC
- MjP41Ki0op7E87 WiSBFILqc8HP41Oqh2OGAOMn61AgKguMlu/vVqMAptcYz07YFU6cbrW4NRX
- QkWPFxgkMAD+NTeSS BlwOfTpTWXaVyCX3Yx7VaUOzZUcZwKlT00exMVZjfKZ12hup/MVaUMAd
- zKADjBFLs4I6joAvUmpo 1fG35ST1z0xVJprc0cm9LhkqAFAzj045qZIzu5OSq80KAQC4LKRzg
- 4xU6D98Aepx1pN8q0H7vKHl k7XBwAatJy24gknoaiWMMw56enSrMarGuQ3bvzUWctLi0+YBVz
- xn73Q9c1LHDlgTnGfyqeJN5DYA 559qkCncBjAPT3odrWW5MXK46OPERJIJAyfzqXy2EkpJxkZ
- XHepY9oX58cHp6VJtVwMkjBznPWiD aRUpcysiDYcAYIPoamcr5XzIyYbrUwUljkg5GcntTfvb
- gAGX3qGtSoK2+xERgccseSSOBTNi7Mry QclR71Y45DEc+1Q+VsACFsk561UU+4rx6gpIifeQA
- TwKQ7vOKFW2Zxu9Klk+cfLjrxxSkseA3XkV KvLYPdWohUxlVGNucZqXbk5GB3570iYYDJyOuK
- lUZJwQDnafapd47sFG7HwuGcHIIJqYrtUEjLHr UIj+f3x0FWAm2QJnIA7mhRb1iNQSJljUgFs
- 57c1Y2ttByB/eqCIh5VHXvVlQFOfU80Wa3KkmtEOw fLCLHjA9OvNSiP7rbjkjkUcHAwSMcVYV
- Ay7gu0k8Amna7uhX8hnlqNp5zjBz0zT/AC+eF6DvUywn eRn5QevvTzFzuAI7EUdRwik73KqJw
- OwBxzUm3EXylQcd6mCOFYcD5snPenhVVtxIJJ5HoablO+hT ty2GBGf72MZyGFBDbUyoGfQVaA
- O3gAY60hVyCSFDdv8AGjfcmMYy1IgvGADnPfvUvlZbHY5PFAUF l3blyM9elLsbJJyOMGs5pqV
- 0LlTVrkBQNnnDdMZpWGCB9457DrUoQE4OOBTdjbeBuOeNv86pK8bi Sa0uR45XgAE8CkIBf7wH
- J4x0q0wIAUqOnJxTAozyRhTx70lrrsDepDtG0IWGc9cU0g+bjBbH86nO 3eSRnB4pAp39MN/Oh
- WauKz2ZAc4XbjJ5INQFQxOEJIHGKuH74yVB9cVGw67APbFXDUFHl1KoG+P7 rEg9qeiHYeMHPe
- pwhRctyp7U8gKv3TkcDjrmhtDjdvQhaLMZKDcQc5pyIVZOOCMkGpEVs9CFxyB6 1KxJcFQMdia
- SvYSp3umRffBKr3wMVIsMjElmGSMkY6elShQsauDkk8gdKlCKWGCSV6gd6Iu2liL8 qsVvLDqF
- 79/rR5DNK2WBAGDnpVsR8YwQcdaapYAZIPvjrT1GpaalIoVwB0757U7ZldxXqcVa2jyv xxz3p
- AoVsO6sMZzVSiujLeq0KjRfMQAVPoaaAWHVSPbg1awAz91PWo9i7cdT1wKztzego766FYqd 2W
- bPrxUZRSd24sOTnNWNowzFSOw5poT5cDA7E1qlrvYTSKDoVL7cbcA59feqzLuwWyfpxmr5DZIT
- BUdPpUUiZB2kYbr7VMnfdjaa3RmvtbHynHtVSRFwDtJwOMelakiquFIKkjqaqSg+Ruxls4z2xU
- cz a0KppddTKnD4QqOW6VVmQguA3zZwR71pN0wc5zwKqug2sfmOTkVV7eQNqKM6XIjBHBIxVdg
- Rkg5I 4Aq267gcsCR0A71Ds3SMWXaCRn61T1WoorS7Km1uMjGBTTwhJXBPXjrUroVbuU3du9I+
- Fl6qeMjN Du3ZA4EDDahYnAI6U0xlWOFGcDPvT8HY2c7u2akWMeXnlznrmqlLRGclGG/UrBAXY
- N6jgdqD8vQE 89TUmAXYchup96UHjdkYA446U1d9DW15e9sNyfNAxkE/pUPAOBjGcgYqdchieK
- YynfkjOWolG8bh F2TRXIJyAueM8VAAAzNnDn9assA4OzcSc57Ux1VoY/UDqO1EWuVXFdJGfKo
- LfN2GMVVdQI/ubsdV xzV+VSCwAPzEduaqyZIZsFfqaqN09xt30RzGAJMLjG3gVLGEEMW/qD1F
- RjcAHI/H1qYHLsevoPT1 p662MG4t2JT8+CFIUHjijZnag5GOfpUwLBlxjaVwvFCEHIzyp61nG
- pNPRF35rpIlJbJPbjFWowRG Cx6dPU1WBLI2Msc1OoHkDJI9vSnyXVy46R2LCKwk4X5BjBqym4
- Lk/Xiod6MwyG4461OAw5ByB2NK K0ZEo/zDtuUKg7geRg1ZjU+WeMgcD1piA+YdwGVHHFTRjbt
- ZeMjBz70NN2Vxuy2JwihcNkEdcd81 OBGG+ZgMdSKjjyVxtYe5HWrJVMJ0PsOtTfQbtzWHqFWJ
- eCDmrCK0sakcbT0oVB5TKcbhknNWYubf BOCcEKOopw1iDVtLD0QmIsANucY71Kik4yuec0kSF
- pHboB2p20E919M1K6hyXYIvzcZYHoe30pyx gBU+73P0qZFORtIyD0ppVmlbAANKTb6g1LoyB+
- 5Tpu7imMjYJzyeRgdqtDhcFc4HT0ppHHHzZ6Yq nIlS0sQlSqk8YA/Ko/KbKMCPKUZNWCreWpz
- kEcj3pVLbSqr94c8dKhKUXdMcFf0GgDfgsPy6VIqj YAx9/rS7dibQCQeckU5fvBQBnPU1HNdm
- junoPCrvycBck4FTKrFchhgnqeabG2VZTjgYB9aspGdg A9OnaiLktBx3HJGFB44z1xViNCbgF
- VO0DAz2poBOAAcdqtqTvUMM5AB7Uc0ktyLvcUJ83TcAOcVM qAzEnK46Aninhf3hO3p39akZN8
- vPyj1p01Lc0i9fUaqkZ6lieBUyBgpyj+1SxodqrjBweT3qVIiQ OowepqrX1sZyiQ4byk2jnHc
- U5YlLtvGT2IqYxsYiFOSDg0oRliA4PPPrTXky3tqRIMlh2xT3Rtqc YGOM09FDOF6Lg80/6AkA
- 49aOV7ia1uV9u9tv3mI5pAjE/N0I5qwvysTlVOeQe9GwBiOpxnFG2iJc tdNCHywGIJGDxkd6Z
- tZQWC7WHBH1qyCAjAcjOeabyMbs5I5JrJxaditdSEqdoUDH071GinkjDDIz kdasBVY/KrA4OM
- 96Ty+VxnC9MGr5rKxMo2SZAIyMgDqeSe1OdSBwuQDwKslQfYAdD1pgjxyTwOAK FZopR01Kyod
- 58wbe3P8AOoWAVioBGF5Iq+yhYMN14yfemsmC3HOaIvleocz3K+C2FI4zmnYJjAIz /M1KEIVe
- Rzxn0p6ALJubOBkCqnZvTUGVwu1WDKQM+vNIFO8hmG32FW3BHJwf6UhwULdeelLmsiYu T6EeB
- nIDA+hFCBstg/NiplT5iMb365BqTO2MkDDg5YU4zsu4KWpCBzywGCM5qUL83IBUDjFOdV3D 5h
- 83r2qMKu9s8jtg0KSlqx8rlqgKl5QUX5TyM1HtGWPAOec1OMoMZIA6AdqaQoRjz9aLvoK1tyP5
- JeGwvFQuqqgIP3jx7VPgHAPOB2ppTn5iMKck4qnZE31K7Rgjbn5aidh5RIyGLVaPKjjqQTgVA4
- GT gYPai19Ane5WkwUO3GSvPvVUjEOHDZP3jVzcykEKHGeRUEjP8oIzlc5os27Ar7dDPlUBDyT
- gd6qh Q4DE5GOlaLKpBBbnghu2KpOGbG3AwcYx1pabLc1im42RVZQMAfMQccVRdWBPB4/WtHcC
- zYwMYxkV UcNhgSFPv1NRJyi1clU0tDPkj+QsoxjoRVKUtvywAbPKitBowoIfIBHFQSKu7JUlg
- vBFVfTVaj52 ml0KZA8rHUkVWdNx4OCSO1XnRwm47QPYVHngYUgjgn6UczW24Nte8iu0P7/ZnO
- By3bNI5bdtVtuD gD2qVl3KnUk54B5qEI7H5R8o4IPWq1t7zH5takYBUMAQ3PIApuTsG5SOMmp
- 2iCsSScDrTCcp9OmK 0g7aolct9iONgQoZSM8g4psgCjG8D0OamyA2DgAdD601gwTGwnjk1Upa
- 3JUnFpWKyjM2Mjj0HFQs NpUNkGrMqb4s459qrEMgALDnnkVKSfW7E3e8iGQ5Dbhxk45qkcM+5
- RkYzzVxtxJBxVSRvLj27eB2 xzTUWjVL3fM5bdu5cnGOmasjBb5Qc/xH0qoqhQyuQ4H3CKsJIq
- hVJwT61PNFrTc5X8PmTqHDFj8w xxjvUy/eUkblzVbcywkEgNnj3qdJGdQw2/THH1ocnbUtO9r
- lsMoBKgc9DntU0aYLfOOeTxVdFVnC qdwPX2q2VUZ/hI6AnNSk0xNJK1ydAqbSRxyOKmjznAO5
- umPSqy8sMg7Satp/ruy55zitG0tLjUep MnDHnBJwc1aViCxwuMDjFVwvPykEnnpVmI/MOMA9c
- 9jUTit3qhuS5kmizHubO4AAHGMVJ5W6bI4x 1zTVA8zkkDA6dc1PuYBduGDdRilG0dEaJuOqWp
- biIZUyMnGAR3FWlCrKMc5HNQxJt5bk9QasY3Ho cHr60Kd2Qvi2FRMLnc3XkZqwi55JyAM9OlM
- wwwCNzZxVpcKi5BII5NJspxaIwGCAhSvzCkO4NIcY GR19anZQ0eAxGeQKiAwzNznPSmttUSo6
- 7jDhkI54HNR5CDBU4PerI3BMHBweOKMA4LY9MGlJ8ysy tnsVuSS33jjj0pVA8oZBJ6ilB2DC4
- INOB/eqTzgHJ7VlPRWG4ybsNwwXJztqVI1eQnngc+9NJY8h SwUZwKmjBUHaDu6Gm9tB8slG1h
- 8cYAVlIPHI9KsqpUsRjnqPSkAywBG0Hmp40Y4YDA60r31Y7skT HIPHNWSmTnBOaIugAUk44Pr
- VgKCRnuOT2FF1zaISs2EasEAYc/xfWrKAZGcHmnImX5yVx2qcLjrg 8gCtZVL6Gb3uMjAMmATk
- dasAMFORwMjNKoDBmA4AxkCngcKAwx1JxUX1vsauOl0AjcL1JJ9KFG0Y bIXvkVJltz5I+Xoe2
- KeD8/zY2gdxQrtXHPoQhQWODz2o34l+bgjqcdak48wgKeRUY5yx6elVGCl8 RUrdxMKzNn74pp
- B5ZSCdv+RUu1g7BsEY5I44p4iw5wGOzGMd6TcY7shuKGbRtweh6cUzZuBxk461 OVBUNyAOcUg
- VA+TwGHOegqVKz0E3G2pA0Z8xSMgEYBpmwGIBQwYnAFWvLyowR14B707YWkDKBn17 VV0yeezv
- cg8vJUhcr60YViQcepx2qTBE2F+YL05qdkUpv4B6ips2y29SiYmfaoOR700pubGCSPvE VcyNm
- Rgtn8hQyhWyMHjPHen72w9ehRUMp4AGBzTtpJZccfSrZ2luAWyM8DrUQD4JOeveq3VxWSaZ Bz
- zuXk80pUByGyAeKsnHA3AknnjimOiqAyEhQO/NRe+yFomRB9qrtBye9OC7lc8nJAFLwpC53ev1
- qVscgpgZ7etVDcmWm5HgkHLKoPHIpiqozk5Iz0p2wZY88nIBpM7fw4OKculgcLIUlQvUMM8Uxs
- bQ B0J6UhHXdxj7tHCt8x68jntUR0eg9JdAfaCSDjB/OmMOuDkdzUoG5GJI44qDJDY4xyTQr7d
- Qcbaj PmKEH2PSmO373joD1p5kyDzk9uKiOdhGCxYZAHatNiZJW0KsknynYnUHr2qEhfKcsSTg
- Cp8DhQck dRUDhxk4BBIzipUo27AovRFaR1wV2NtJ49qruy4wePWrxUkglehPJPBqiVLMpdG+b
- 7px6U5W0sSr NWuVSueR8ueSTVaVSJDjGR2PXFWn+8wY8fyqq4ypK5OTyc9qvlu9TXZ3KjKTzz
- z0qu4ZigVSMAZP rVp0G9gAwBxwarMWG0cHA/GsrXbS/EpQbTdyuyHYp2tg9vaoVAOBwT3qSQM
- yr1X09qjbakxIBAzx n9avl7mKg2vQYygTIRkgdcetN2/OCGUDrjH60/c+8k4BPI96coACse+d
- wqm31HV5lYjZSWzncoFQ uA+8N8oB6nip+fLZsgI3AJpjnEm3cuTgn2qU+wlK1m27orMCFUBSx
- x+VNbMkZDvhQw6cVZbAcAq3 3aikAyApHyjn1NUpN6JWDmTXvIrOjAMVO4KePeo5DuAyMjPQda
- mLAqSoJBODz+tVmz5WBx6moa7h K60KrszHgcjrVd0weeHHUE9amcMMAAnjPFR7gG3OTuz3rdt
- pJi5mk9NDjY1Ckclhk4qXc2NyqDt9 qrhpRMWG0fUVKoLBN5Kk4wKiCVtWh6W1ZIrSyFQw9uB6
- 1cVNiYPHTr61WCDBJfkn5cfrVtGVsDna RkUpaPRFqXRLQtRYCAMRz3FTRsr8569/Wq4XbsBwy
- Afw9asqMOjDHoBRrvuQ5R5diYKckndhTxir kYJfBPbA9qpBf3ipuIx6981eiwG69u9DslqSk7
- E8ZXcSf4atxIoB5OOvTvVVQQdv97HarUZ+RgCB kd6Tu9i7K9rFkDKqGJyeQfWrUa4b1I6jv9a
- rK24qehA/KrW3d1ID4AIHem22yox0szSiI81iFBIG M468VYRfnXBHIyQfWqUHAByx46DrV5cG
- 3yMdce9Q6coshLWxajUAZHJx6Ui8rypIJp4+VVLNjIwf rSlHVSgOQeh9Kuj5lJ20FH3nIxnPH
- HQU+OMAgbx170mweuOak2bHBYnJPWsp2u9SddiCQAyqOCQM getRyBW4UFcdfxq2RkhsDHqKjc
- BztIyM8047JNGvw6opjG3aMEr0p467SAwJ6etTCNR04zR067Sf aiatr1BSSV+hCissnB6jlT1
- xVuLcVCYBJ6eppoXBUgdanjC5zv29xU3VtiZrm1QqowYHGOO4zVmM MiNkYAIH1qOMLvwWOSep
- q2MK23IDDgZp8y2aKtZEkZBwRkgdhU678BAvPeo1Q7hhGUnnHtWiqsij GDgUKTWwJxWlhY0Uq
- ufl9asKijhQTzknPFRhTjnk+1Tx8NnaQQO1G+7EpNdAC4fKjqeRTzy7FlAz 0FSMoDhiflJ5FI
- yoZFzk4HUVSSkwbjfmeg4YECqy+5NOIBU4GfWgKwiOG3YNOLED5lJ4OQDUcqWy CPLuiLYdvA2
- +uaCoD9Rn0xUm7jHQ459qYUYliGUn2PJptoq4m1TgnJLdvSpNrA5UbgBg4pyYwO2R wTQEGeG3
- AHHB70tGzPnSViHJdQudqmjywAOSeOD2qUtiYkqRxT02lVyeAMe5qrS3aNZaLyGKpEee G7ChE
- lckMdo7YFTortIG424x0p4HBA7fr7VHMldEvV6WINiqAARuxz7+9KBxlVJI7VYSIPg9Oxpw T5
- SF7HiiTRL1ZXEZZCSoUHpTNjM53AD+oq0uPKAz17UeUcdOn6ZpN2dmVJJPVFOOPapY7d2etMZN
- x6g55NWyCpyB8gqJl3AsMD1HcVUVuwk9SuF8sZUA7h82e1MJAZQG3ADk4q75bmMg7QD047VWZd
- pb ONhPFQnd2QWTYwEmMFgHOeMDt60jH5G2jvnHvTmXLjnGeDjvRsweOctiqtswirDQDu+cgHr
- UDKPM IU85qUgrGAw7kdaXaSm4FQT7d6ak4vUhy1uRKocc5ODUe07sEEkH5cmpTGynIdeOuPWm
- kqOW6ZOM VUXqW72uMDsuflBye3Smg5U9h796kC/IvB55qJicjAx6ZqHHrbUl6sj2YZm3A/3eK
- h+UPnIyM5NT uGB7HsDjrUL8IitsPckU0iZOxXZ/vBRlgM8dSarNllBww4yee9Wn3NI23A4yOK
- idiJFBHUdaFo9E XIqtln28mPqaqPvxkk+w7c1eGwyDaedvIzUMoIySc5I6DpTcrbg5JOyM9lG
- 7aMbcYyahZG3ZwVxV mYMQfmG44NVnL7MZ4GOPem17oc7lqyo4PlhmPJPTvUG0AvjnuAetWJFb
- eCQWAOKrudr5b5cjj6UO zVjRp21e5AyDb5jbunK1FIC8KnGORnI6VZO10DFiD2FRE7l2nqR0q
- rS00MG0upVIyhZM5AOc0pGY 9+04IyamG0M3XpyMd6Yy/uxgNucDPoKTTk7GvM7bkW0AxlsgDj
- BNI4ILYVSfUjpVoRAHghiOlMyN r7sY5FVdOWphKzd2yg24yE5zkYxUbxnJIzx0NWnQhAcgnHG
- B1FNO4xkk4XGF9aG23p0HzRVuUqMC FIGM9yRxWczvgkD25H51fdWWQ5yQx6elQOVKkAAkrkjH
- elGV9C9tTPJJU/KzkH17VCwBRhkjPTNX MAnBBGRyAarSZ2hgQMcEYpxbb1By10RxAR2DHaw2n
- vU4wEDclx0FRZPnbVHB65NSr/q/mwSPSnaa 3WhnGMpOxIkYcqSSM9BVpOAAgyqnGfWokRTJye
- /SrUaFM9CSc1LlZ2RUn/Kx6D5k3E46H1q2QTMv zDnt6UyPY0jZHfr6U9UbILevH0rSTbI5dUi
- wIxuDHd+BqyobrjPNQxbVcjdk+/NTICrBdrgE5JJr K8upspa7k6glsAkHHFXUXDIQM4B/Gq0b
- KNvRmHJx9atBQNrBjgHOParcrWZLm5aNXLShCFYkg55A q6FGdzHn+9VWMANuySAMgetXI+Y8k
- YO4Yz6U7q17ii2mWkAVVbgn1FX0j4Udx6iqaj7gHJ3VeiBc 5OSOq4qJau6HfXUsIw2gn5yemO
- lWUzk4Xgd6gj4QYUdenpVgYJYsTszwB1py2JaSe4w/cGfv/Snq uSpY8gZ5HWpAAyBipCA85PW
- m8Ky7eR79qhtWdhJdLDFUsxIZcdcYoCnBJUkAcU/bmQtwVHXbQwBD Ku7pwM1Tir6s0TuiLAJw
- x6DnFRNwTwc9qsBMQvkEnIwaTaRJkjGBSja9nqJct9AHysw4J7HHFSIm HHTBHFNXAycZOOpNW
- YwrsgByMZIrOTsrlNW+YRhUm5G8Hgn3qxg7iDgcYyajYbQchWXqNvepskjb jt8opxSdmPVsmj
- JyAWB9eOlaKKTlgRgcEVSjGcHaScfSr6qCPQEZNXNx6E3k1ZE4BXBYZHQU7LC4 BYZBPUUiqRG
- AgOc5INToEIGckg4UZ6VKskNNpa7j2GSBjjGQaYNocEhvb3pyZUYLAH3puO+B6del LZgiTIEn
- Q7j2NJ87yMT0xmmgsOh4A5yOTTgrErjOP50k3uUoqIMGRiw+6Rkk01sja5xu64p0mTEQ 3AzTC
- 6tHySCOo9adkyYRfVCnJ6nnPrT14kBBAznioYwc7gc46A1KhJcswBXODijW+5vbSyQ/hijb Xb
- Pp2qVVTzM+vWmqpUZ3LjPSpgqFxjj+9k03boZbxtIMLtAyQc+tSIAYzn5cdz3pCwLYHJ7Yp4AL
- bW4XPPNZuPViau0mRgMsxKsGHoKmCEkYU8ZIHpT9pU7sYXPFNAdZyVILMc/hVKTa3LsmtBCpOF
- AA 9OKWVBv2jPIyTmpI4280biAR2oCKfvE49aUY31bM07S1ICi8ZyQOvNQ7EycffPUE9KsnAfB
- ySTz7 VGUVXbB3MT070WaepSVr3ZWdWCbGJwOgqJjgJ3HHFWnDOw5xyOMUwqMtjHDdT60NpbkS
- baRX24wQ jYJORTXIwoIwR0+vensjEAjPShozgFCpwDmnyxHBJdSAoG+YZb0GajcELjcAMdKsY
- /dsCME4Ix2p N2YBjacdeOarm122G073IQqrGzFWwegJpjBSwYKcE8CrBi/dAksuB0NDLk9cY6
- ij3G7CUnzXuVDu EgBztHTimEDByxBPSrRGXwrDH8xVXYJGAPU84pytYaW5HJneMsCOPzqFhuY
- p1OcD6VaaLGOMAdjT HCBgWIBApSfREN32Kflny8HOcYB71DJEzHjqBz9atOPnznGDnk9KgYP5
- Xy/Mfapemty48zXmUplI A2qzNjGRVd1eNRvUirrABn6g8d6pS7i+SSf5E04663G7oqspI3EHn
- mqzFVG4DGD3qydwbaPSoJSu SvGT2xVySJnLVXKxJLKvzZ5JNV2A8vjax28DFWJOm3BJPTFV8f
- vOeBSslcE7FfgpknbjtSAF5c4y CPpxVna2SdoIz3FRHaGyM5zx70k09A50lfYiKBo/pwDTQCG
- yvOGxx2qQ7NuQrZDcinMP3jHGMnkV TSSVyXKzSRE2M7h1PXHemeWrAF+GI4FSbQhAOPahm4x0
- JPGe1OVraAo2d7laSNsIy8YHQiq0iHLs WAK4wParxU8gsT61CwBWQMM/Lx70SlqiXOy5Uigy7
- pGLbj0JPaqjhWkbI2kdavHeocYx04NUyjA8 nODyfU04xS2ZTs7vqUnwp3AgfhVaQFXzjapGST
- V+URhlGeRnI9OKpH51XJ4C8+1XGTetxKz1OGRH XcDx9asx7gCQBuxk8UkSs7Efe+lWYlZRtI3
- Ang+tSut3sZ36WEVC0W98KCQRjirKFgQMlhu5zTNp Jz2z8q5596sRxEfJyxJyOcmlLk5WaWtH
- XYcrMJNvUHjA7VaUoyqSTlSAfc1GigTjIx681ZQASKcZ BGcinFWjYVknZDgpzvZhjdgECrfBA
- ywztzjHeokUgbsrgcnI71IpJVeV3d+KUWzRt3LUWBGDxuPT /CrKhmlIJGR6DqKgQgkbcZA54q
- ypDMrEgHGc1lZpkJ66Muop2oCQCO/Y1cQfKpPfpz2qpGcBQCNn bPfNaMIUJn5S49appvcce7L
- MKhSGHIJAq3GoTGc/n0qpHxKp6gnIFX41LMGOMHnFJvuOTV7lvomF GQCSakKkrljnIzxUXcbT
- nd2FS4G1T/CB07inTTRCg1G6Q4DCZVlYDOOKcf3sYIHJ75oiXfvDEKM8 U4K2xQqkjtVcqva+o
- S27jUTG49e4pxUKvuRUyjLnbkAg5yKaDlScZGCOnNZ2uaJ63IGTjklh/CV7 077rDb1I5BpzKT
- EqnjIwKbtZVGeDuwM85qVDogatuAWNwDnHHSnLgElTtBPODTiCGXJUIO+KcQuM cHb7VUlbR7C
- +1zE6BSGGak27skZHGefamAEFdoABGcVJEH5LE4PAzUpMaSaumWYlZpc89KuqSpxg A9PpVeKN
- hGfmJz6VOi5JwNpB6k048vQNd2WflKhs8qe3enqh3lhwe5ojAEahirDHHvQFY4YHK45q k7PQN
- 3qSMVyCoJbvzQ5yFDEKew70oAIBIPB4pCSc5HIPU0nFNXQknsMD4O1OWxRhiAd4JHtSqB5i j2
- 60joAcp1+vWqtdGrS5rCgk5JwfrSkArzgE0iqQ5I5AGKcuHwOgrFtqW4SsnoMUjzAVOTnpUyqG
- JwCPX61EmFbI5yOanVVZCyMc5zyf0qpRd7kys3cVIyRyGGeetTlcgEYz1I9qSPlQ3HTgCpQPl4
- wW 6fhQpS26GbkubXoL1YN1xx0p2wALngE8jv8AWm/MHx3qwqq0ZKZLdDROTXU21vZAflwFcNg
- 88dKV hsdWxkbevvSbEaIspJYHOD3FO/e+YMKOnSiLu9EDbuG0GNmw27sD3poG3g5IzkAU8OPL
- Kty9O2bc EDBxnk1V7XQa21GsmSrE4JPP0qDy/wB4SMjHUntU+zd1O0DjOaTYFLMx9MUrPl3Mp
- S7MpucDLDq2 aaykJkcgmp9mVbODgcH2qEAhMA9+/an00NY2IAJGyCpOc5x2oZWBOVJ+Xip1yM
- jIBPrTWLAgsCeM ZqVo7ESqW2RXwfLz91elRONuBjIq0yfNkg8DpTCAxUbWHGDRfUV0lcjHCjq
- fQGmhOehJY1JJuWba vX0IzTdpI3Ecn+VDWgOOt9rkBB5JGDjBqLBRzgEsB1q2R8nIO4+1NbJb
- aCuc8gindsV1LS5TzkkS McgVE8fI+Uk9B6GrLIxkZuFxycimFQYyRnccY5q7NLRjjZ7FNlLFt
- q4GOaruDtUc5A4Aq0V4JAY5 zxUcnyxqXAXAqLaA+XoUXHOTgMB0qjJkSABgcDpjmtGQ5XcF5I
- wapgNuycZHBJq1dxLc5LqUmH38 kBic1E5LNkgA1cypUsu0MQeTVNwBGScnPIqJxu1czTUpabl
- BlIkKkcN055qHC72KqST2PNWySGJ6 ZHGRUJQEHBye+PSteW6syp9rkZHyg7wdvHHeowu5VB+X
- 6npUhIJxyF7+xpm3EuQS6Zx71PK72bJj OVmhNuEO5gR600xksfm3dKlUAsVwCucj3oCBo22sR
- g0c043JipRd0MCoVBAbKnp61CoBlJ6kt0qx 5alzgkY5+tMkTanyDJ3ZOPXFOD08hyV3vqViWD
- MMADtUAAMbnDcNzV4gBFU/ewMmq7HAYdM0SfMy HorWKTJukZ+QO/1qlInPJBbqeKuurByM4H8
- 6glRfMwW249aXO47Mr2fZmcY/nbcpJbuKptxPg4K4 xj3rSk3IWwQwHT2qhIrlCRg554HSrU0x
- p6WbOTRcKu0jp2p4R3dQWGcZwBjFNRW81t3HHIFW/LO1 WBz+FVCavqzBSEKuBlyBsOOPepVGE
- B3HjgGlBA2ZXI6kZ61MFOSI+M8gHtUxkn0L1vqCLgjcCy9R irMbbhnAwTUfV1Gff6VMgJjOSM
- 54IpuF1sXZrccBhsn0xU6hWCtnPHWmqNxIALHpx3qVFAAUgg9R T0tpuVa7J9gRMqu7ng1cjBf
- kId+MYqBcjb0VcdxVpEODIWAHQYqOawrW1ZcjRvKAPQfdJFXIlKx9 MYPaq0R/dBScqOM/yq6u
- 4qQDjP3ie1SnKXkTF9WW4wjELnHynbmrUKu6Zx04OPWqgTJ3dyOFFWEY jOcnHUDvRGBV7rQux
- qI5h1IFSnIQHI46+hqDkhSowD1zUysFbOM9unQ1ST1ZKvfclU7HyTtB7GpT kP8AKcHGc9qiLo
- 8Y5B569Kk+UF9z5AXFZ6PdGkeZvTQAoZSQ3OOMVKoHA+YZ96jXbs4buB0p+AFT HT1zV811a5k
- m0mMdVDKc5YcAf1pz7gABjOeR60qjdIWOGbOOBUgHO0927dai7RbnZLuQqp8ojg5P GR0qRVzk
- cYByPWgrmRuRweacOH9QAMkUpO6Ls2rdRyZaQEHoOKurlmxjKkn8KrxjlvlYgAbWHcVO AQ27J
- UEZxUyV3oQ4q+hOmAo6lhxyasCNkbkbh0z6VXADMVGMjn61bRSF+6SDxgdacUoaopavUkVf lD
- EEAHGKtqPkLbhtHU4qFVygKhgw65OalQ7WOcFcYx6Vo5NqyCK0EB+Qb+mc8Gnkcnn60KMLuJVu
- OlPADkAg5xSaT6FNkZjxFuORnktTGyd+CBg8Va2g5zzjpzUAYBSpPKjAx3rNNvVgm5q1iMbg+5
- iM GnjY0nJxx60vCg7uv8vakCZBCg9eM9vam2xqKd7jesgUY+tSqm0gZKkN/OmquJV5+cE1MQS
- MBWIB 5x3o5ul7FbpxFO8QFRjcDgEDFTAd8980i8QscHr370oYttIwfUY6VUddzLWyViQEuwcL
- x0x71MIg sg3NhvQVAMEDBBqYN+93MpB6VN0ti32RIVO9FXovBNOZk8rvjOajDHL4VmPfnpQAo
- CyNknqc0Nu+ hDTtqA2vIxHy9+e9SM7PGGQY9vaoWc7vkA2njHpT2J4weh4IpN6mjSVhCwNtzk
- c8imhwW5yaeXV0 wfvHgimAgwbOMnnPpSdiXqtRjfMSNw44ppXBIII4AP1p3zFCSm0n1poAdwV
- JIHv1pxTfUUpRURjf MgGCec5HpTOOOScHpUw+U9Rnp9aiI+bB45qvIhNNJiEjzCOCc81XIJGS
- cKDxVlgmCCGwTkmmEHYT jgnkGpSHpazI2PAZjls5pnU4XIyM89qlbaoye3603PzBR6Von0BbX
- IzGVjB3fMcd6a6Ig3KCSelS KzZG5lJ9AKYzPvIUA49qmfNcabvZkRVmPrxVQgnGTsA447fWrW
- 87sdW9B6VC5UA9s9c1pG6bVimn flRXO7zc8kAdKjdQ/BwQRwakKbpAobnvzUZyCNvPH5Vb1ZF
- 7PzKpxxuBHWqMwMaHaR16H0q6yZbh sZHWqskZLbWHzMRg1DXW44pLR9ShKm0jZjOOAPTvUGxV
- XGc8cZPNaJUeaQB8o79aqsuCx2nJOR/9 al7Rtb6jnJJ2RU2ZTpxVTb5bHGSxIAJq6cknquByP
- eoTBlGG4jnijp7zMltuVCoZs45BxkHFMZSk m3ORjmrAjw4Qk8rwfemiP5sSAkY4x60pWS3NFN
- dWRk7uCNvYkDpQFMbcuNwbn3pxXClgSQH55qRg JFXbg85xRaL3FvsiJRuGQcDviggAHoV60rB
- vNwcbc9vamlDjCnIIyfarml1JnFt3RWLF5gwUehpr Iw+7gZ7kVO+PLHzDcBjAqEqWAUsQuKUo
- p67B73QpNgxnI46E1TYARu4+bB6E81fZFIAzuJ549qqS EFCF7cHjpVJLboS+YzphnG4jBPQCq
- sh4I5znn/CtB9rjGMcdfWqZCGFjyR156k0TjJx2L9pFnGxq fMyh3DoWzmr6HcQQQFAwR3Bquh
- CgkJtbPSrCcHOzAPIFOfvPbQlwdlbQWPltpzkDPPerQ2uQO55x moFyowV+dhj6CrMZBTGVz6i
- m77obTTv1HoFHyqMnv3qUR9DuBA6rimffjXt1JPSpVQqc57c81DXN G97DSVrtEyMQqEkKu3nj
- vUyguQeD6moE6fPkZXFW40AT7/A7DvUyShHmsTzpeoLndllZgep9Kuxf fycBTxj1qJcCIY6Ed
- KtoFKHbhmBHAosmjWUls0WQAYuAQc4q8v8Aqxz04J/CqcI2yNuGVbnirMfz L8wwrGny+RC3Vn
- oWoyAkec/1FWdzcEDI6Zx0quNm8YVsgY61ZjdmBjYqFz6cmla3vA9XsWBtaPqd 3bmgEp83U1G
- pAZeQewFKchGJKjFXzW6jhF7E4O0DcAR6YpxfdI+evrUIztyD9OetPjIYnOBjrxWc 5ajhuy4j
- qVPTcKC/Izg8c4qsjEYHr1yKnj3CbgruPXNTFxTuE421uSGTbGxOcZwR0pQ7bgzfIoGe e9MRC
- 7fMAFJ5Bp7YRxhhwMYNXzK7FzReg7ftckc5OPrTxnhR/EC2KRM8Zwdwz0xUyBAoAB3j37VL 5V
- uipTfzHxgkK2GxjHWrKLulYMCcjiohyAEBwD1qZQwAOwnI5GeRSbbegJ29SzHtwOhHtU652lhk
- A8CoUVi+1Qx4GKsxAlCjL3wKm7QNO5KNyE4yccE1MBhF4OGODxzSLtzhfQk89TU6nCrlcccE0+
- Zl uOl7CkDYAuOOB60zHzgsCMDGalx3GOCOaNoJ6d+lUpaWIWzGlQ2WRgcDr6VAIv3zn7pz909
- asNhV AAw3tSEjILAsfSqvJBzS2uQgLyGbBznrSgENkfTmnhwWOFGW9qMZbcSOOwqeWW7NNle4
- nzHqvfqK lBwSCc5NICVQkkFhxSkAkEDnpUu7Rj1JFEag4ySQD1oJONvyqTzigY243DGfTrSg5
- 5OA3ao6XLjo rpEgwgCPtIJzkdqewPOc7ycDPtURIJyMA+lAJ2A9fTmjVoTpteQ9tyq2Dg/zpU
- IOM8nv6Co2bCqN wIHWmFsSkLycVbV1YqK5r3JgMHJ4BPp1pWkUbcMMDrxUCsW65GKU/KNhIPv
- ik4rqDi21Zkm8M5bg c0ib/M3ZGMYxTVwEw2cdqeWOMEZHqO1K6UtBuTfxEZb5iSdvfB70hJ3c
- MOOOO9KQpTK85PNLwW2j gHkE0732I0TuxXOFUNgjPaowuG9QelBY7SFBLHoKZLuZRs3Z6U1Cx
- UJ3W+4b2LfMDjPHsKc2S3yq eOtKuSfmUjaMGmM4XLhjz0FK6bIabdyPq3HU8nNBC4BwST6dqX
- n5W9RSbcgNjJIwMHGKvbUf2tQY plQDxVc56qeTxxU7ECPbtwRxn1qAqWIw23mnF9w212GFSkY
- cDJ796iZg7AYI7kVLkrGCxJOcYNVn G2HGCSD97NKMXJtsnrfqRsjLKCpGMdahYAZ67/6VZcFz
- klRxjmomAyOo9CD1q4yktFuaOStoVZFz GTg9MjHpVcjMmCSvy4BNWWO5sjdjoQDVeQMMEqxI4
- Jq0/PYmWjViswKqRkMw64qiwyuOcZ4z3q8c FsqGI/izUEqKHXnk9ABWMnrsTblVmUjtV+Qy54
- PNV2V/KbBHtVtwTKOhUHHAqux2uu4jGeKuyd+4 4SkVfmyoztz1zUhB8wc4UDNPZVKlgcYH5VA
- MqMhs/XuKHZ2Ha6GFS0yqxAXGSR3NIwJlBQYOMmpZ FJxnpmkUjadoOQehqua2qKsnoiMKqRtu
- 3EgcEGmFxJz82QMHFTgATkHJ7gE9qCdqA4C4PTFDehjz cujK2Ar9Dnjj1qOUhRgnqRyKsSAZI
- zyevtUDpiQK/OSM1EprqDty6FcqfNyD7/SqEjYkPOT0IFXZ AVZtp5J69uKrv5bFixPJ5x61p0
- 7jgktbGcyAuD0x0xVZowzlOq+1XziOTP3htwRjoaqSkHkcYGM1 HLK4ubojkPvFSdoBOSas7AE
- G0hhnINVQyliuGzjqDxU68lAXA44AFaJPe47qysTocN84wvYmpgBt +XrVc5zgncegx61OmTwS
- OmD9aSuVyJa3LKKWVWI9gRUirhMkr1PB54qJcLgAE+ozTxh2ySAc0byu LVq0noSInO7D8HFXc
- qrKuDuyfxqvgg4JHJ4xUi/NIm48jk570NXd29hP3tyyhOzcAF6EAjpV2I5l VhwCpJB61VjIyM
- 9CfzFXEJyWGBk4A9KzS966QnGz9S4p3Jwd3Ye1PQlY2DfezkVXjOG5DAjjd2q2 CDyPmboT2NU
- 48m5TdtCZJMKWB3ZGPpUgXMI2MRjjr1NV4353KoAqRcsSobGTnPvUweumg7u9yzuV IVJznNPV
- wR94bD0JFVSdu0EhlBwaeJPkZgv0HYVnpzXKUmuhYXJlYKSBjIzUqs3mKmVAI5qgJDuL Drjmn
- 7jhXJwCPTkVcXd2Bxv1NNJCi4Cgj37UAsxP971FUxk42tz2FWQcsDnaaizT0Iah0ZaB3oWd WJ
- xnIPFSId4XYhZsc+xzVRSuOjYX361ZR/3gI4B9KqT8tRtcsdUWN37wg/fLZAP8qnwS2SSv4VWU
- rHPl+hGAT0qwHYkuMYGOKrXroVzNapElvn5vfnPrVwN0KnnuPSq6BQ4Byccn2q0CPlLcA5/Cp5
- rv QiW5OobOQTz29KtRYHBP3ujZqCLDtlQ2QOM1MgOz5x0PGKe+5SldpFsBTtwCOevapgmcr19
- ahjIZ eFbC9h3qbpJ0OSPzqW+g1fqxBncV56/ypwIJUknOaQGTaQVBJ6cUuMR5wOOM4qXbqH2h
- 8h/djODg 4x3pmSX6hR6kcUobcSCp68e9DcnJ4wO9XDXQE9NXqMCFSem007arjKsowOlCnEXBy
- e3eg7SQccjj IqZKT6h7smOHC4PBPOaRi20fIVGOtPx8qgkHFBVi2T09qmLSdw5etyJC4TcQcZ
- qYAFjgZPUUDqVw fpSjIO3+8OvrRJ3Fa6GthGVQNxx1FNb50IB255Apejlt2frScoMDBOemKd0
- hpW1AYCjcDzySaXHz LkjIprPztJwAOKRQTk8nPr3pNuw3orpbgN3mfMOvHBoBfcegUHAzRk7t
- vUA8HFRKWyQScn1quYnV u5M0jgHBBPXpTEOF+Y9cjFMwBwTlh70pY5zlcAdMUou1mSmuxIvTq
- SM9u1OWQZIYYOOCahVwEzkf 7VIXDBmAyQcGm3HZlpq6uWc5XaSBgdqrlmLDaQe5UCm4IkHzA8
- c0xxhmBb5s8e1NJb7mkZa+RYZm DYztOO9J95sFlA+lRo+Y8MAcjJ9qRc7gwzjHc9KOVcphysn
- C4TBIJ6cVH0bdznpjNGcNjB/wppKs oKsAwoW4SWgz5wzEnoKbzuwSAByD9aduycAn+lNwfIO5
- TnGAattpotJ7MhcsPQ9jx0ph3NCM7Tg4 HvTmHOPmI9B2NM+QykANjHek3Ym12J8ipuY5YdR61
- XOCQcgDv7VY6jrtB65qu+ORxzyKT9NQd9iB yGZhk5A4A9ar7vMxnltozU7DJ3Y4Ze3aoSAIsj
- vxmqjazuVFt6FR2YSbVBIIzxUBVjggexBFWzgI eRnHBqrsJhUcjPOfWiTe6DmuyqyEKxbJOeB
- 3qFsHB2MMHknuasOSM5Ukg4TmoWGGIySOhyetOXMt yNempWPO5cYBppXC7QOVP5VI3DAAdev4
- U1wGHOVLfeOajV69DRu6sR43HGfb6UwoVcZBwB1FThcM No3etRF3MhJVuTz9K0jonZkp6asGJ
- 6gD61Ccs5DKT6e1WAGw2VyQeKaWGSp6gVN0tLApLsVnBZkI IL4IpoX5NxYFyp21MQS3GMdc1A
- WLnHAYfnSvpfYznC+xA5xGobBJGRxVR24G5So7/Wr5Q7SrMAcd COao/el28HgHPrTg9BK1io4
- VmByCx5J7VUkVckHOPbvV9ioUjgLyBntVNvv4PHcVd3fmKsrHCs4A JzgZ/WrMeUAYENniqabV
- y+dw6gVZQltwyFOKpT1aREoO1uxZGS2cFsnp6VaHXd2FV0w0KnOCDzip kxkgnPpUxipFddSwv
- LbtrKNufxqwpRh3BPP1NV494Qcg/hUi4Dlic44wDTabauO/ctIMoRjkdR70 DktuBJ2kU2Mkjd
- jaAck+9ShxtJ2nGaiKd9iabd9CaD5XxlSFOBmrkZUzAnIxyeaqJgK2whcc5Izi rS+WyI/8Q4I
- FTK6Zc1zKz3L4KKp3MuC1OQpuPPHb1NVlEXKHnHfPalVUMm4E+mKrlb6kq0VroW1K hyACMjgV
- JvZSMkZA9Kq4c5JcZB5x3qfefLBGCp7YqOZCV97DkcEAPkjrkU7IIJw3QnGagLjACgk9 8elER
- z8vTHB5qancqK5ndE8TAncrY453U9STMy9eeKqqymTAbkA4GamRmEgUZ6c+1S79zVxu9S+j EA
- MNp4yTUyliqsDwRVRGJj3YwmfwqxG6s+3BCf1pRm9yHpsifB3evParablTIxgjBz2qoGw+CDx3
- NTxk4OMkHsa0c+rFzXLgctDtYAkdxU6nJJ45HPtVRFQg5OCe2atg7WPBA71CcFsOL5XoWULMdp
- Gc 9cHrVtSTyV9s+lUo5GVCzHHt6VajYSBSSVHrn3rSLb2QSve7LaEqCBkHpmrqBnChuD1qivM
- pI5bv zVlM7QxbDZwTUyfUvR77l0ElxnoBjjvSsZQpyQAvbFQopwDv5I4HpVgIOm73+tVFRQnK
- 2lwDOQu4 HI6gdxT+DyuWA5poOMNuDMBgAVIMNICflUjpWc22tQe1xSR8u0Y96XaQVx827nFRq
- ylM5LY6c9KA SGJXp/DSSsKy3Q8d/lxzxgUhQEv/AHjjimCbkECpl4O853N29KPeiXyu2ogwSM
- UpyJNrEEegphwT nawPQDNPUuGDHAOeuOtCVtSLu+40BSm3BGCSDSZO7g5OOfanNnftA3c9R0o
- OMt8vQYzVPcL3XcEC 9W4PrjioyzHKtnAPBFPCj7rHr71ExypAPvRC8gTtLVXHMGCbmXcN/FRj
- 7/3sAngUMSyKCTtHPB61 GrfvcnIB9fWlumXyt6Cl8ZCgkg4HvQCGIDDoeTmmM2COmF60ze2d4
- 6UPVE8rS1JSwK8Yz3JoJ2jI PamEfPnIwByaVFCpkHdgnrTjZLUNtRCyliBkg9hT1wApXHI5qM
- AGZmORTcAcjI9OetVdNBKVlZD2 bDdB1496UMG6kEDvQHyvPODzx2pnmDAIHOKSetlqTZtbD94
- 4BXHbPrTgxZsggKBgjFV925+AQONw PWpMp3DdOvrVyaTVg9nJLUkZcg5fIJ4+lIGb5sLxnGaY
- 7sQMD5T3xSKxZckMOefShXaHeyuLnEpU jk8k0g3k/eGMU0yKxwMjPIJqHe3lY/DP8qOVvoFlL
- R6ExJD5TaOMHioWIDLgjOMU4EBipbf6ketR yKu4NtIbriohDUGmpWSGEkqVZgSOtQORuAU4PY
- VK23ZuIYHBzUJx5anBGRyaq73CU2RFzu29s8kd 6rktjpgZqd3BO0YA3Yz61DKWCAYAI5px7Eq
- pJ+RHKoVMnG4DkVUkOIQR261MzfvWOC2Tk88VH+7U 53cEcjvTk+hUYpLUiYq0obepxwBVZ2Bc
- hlzj0qaTBwwBHQAmoHUMc9+MAGs7q7uJWIcrszg49M00 kH72eRz7VMyAOFHT161C3DgkgjIAq
- opJ3KcvdtEQBOmcHd601lRZGwe2BmnkHeTzuBz9aQkSOR1G 0+1PmSepOuxGM5x3HvUL5DtgDI
- +YHHWrGQkJbGD71Fu3tvAbrg47U4xdr20HKKWr0IZMiMltxBPH PIpgUKpZc8HP1FPk3FVCklS
- agAMjKORj360ON1uRb3dxj/MCWIAJ4XviqzIFlPUMOOv6VYZguAyM AD96q5ZndiQpHUcc+9Uk
- 76E3sUpN2/bwRnioZVxl3bPGAF71ZfiWQH5cEYPYmqsvzYK9O496HC5b fVs84R2AYbcYOFwOt
- XI2yyMM4Ixu9aqLtL71O4dMmrkbjcowvfpTSM38VlqWo/uBSWUE8GrUYVYz 13AcelVY8BWx85
- x8oqePJwSu0AHH0qlrpcat1LYHyqADgHjmphjaQeR3xwahRlEOcHOelPicknO3 g4pxTWpa12R
- cGQcDC80qsFnJwORxTIySoB6E8kdqmVdvyfKMEjLd6ykv5jO6UtSRPM2nbgKfUVMv yocHAzjJ
- FQJ8seSx9zUyNubJ5XPSqjF9Byk2ix0ZmXG3PrVpWROSBheCfU1XQuWZSVUDse9SDIJY gHIII
- Hb0rKXM3YrSyvuKwypKZVsjK+tSOSFyGyR1wOlKq/KRnnHBoEacqWJPpnpU86e41JOOpHuV UG
- 0Zz0yakRgRjODnJHemupOdowMcH3pEG6QgZB6YNNxTSuyNL7kvfK4BHUkVYiO5m3lcD8KhWPLg
- kZ9hVlQHKlcHPUd6hJPRG0dOpImAdgJXsQasxjawAGMnkGq6lVnJxnnH0q3zkjPfGarpsRK6aT
- ZN jhufoT0p6SNtwe3GcVCqknaCCAM4qYcKwzuz0AoXK99y5rSxbUq8jZbGDkVYT98nU7emapR
- D5iww uF6N3q3CwEG0NnDU5abGMrpouZIk6ggdqtRsocgkbQetU+d+TgjHFW1I2ZI4OM02rW7F
- u76FtHUZ AxgHHH86tR4xk8gjNUozk/KeM81YXk/Id3baPSlyrqLmaL4KhAnDDqMVIpz8wByD6
- 1TUsspzjHGB irHKwBjkY5PpSa6I1cLJXe5L0bOQAO38qcSdyjrtPJxUR8tlC5OCMHB704AZXL
- EgHnnvT6ahovUc NoZiCMN93NOVy0Q2j5sc0HhBgZOO3egHK4zjNRZ9NQUtbsXy1IOVx2z2Bpx
- YiM45IOKZvJbB4GOD mlj5AJbJx0Bq9be8TNO97kjcw5CkECpVKeWrMGJpMAZIIwRSbcBiTkbe
- 3SpWqNJO6sKANrMemaib 3O0HvmnAtnnOBxStngEgVPK76C5bakIyvzDP3T15pBhgMEBunNLkg
- YKlU7saGIVc8Z6gVdnuydLX IwSHbcQRTWcA4bkAce9O6zcDJpSMYbIwexHUVOtx8sXKxGCDtI
- AzjPPvTGxtH97HSrG1WTnqTyem KgUDcSwJYHr2qU3dg1Faobt/dkK2455poyJFye3SnbSJSRy
- SM8dKMDhwwIwQRWkX3JlorDRsaQBm IGOcGmlSAQvOemKGCbG2LknjOaQBgpGeBjH+FCknqNKy
- 0Yi/LhS3Q8GpG7kYAHQ0w5CnC5oPmEjI CjuSOKaSbuym2noxQABnrntUaSFUwcjnBzSkgnCk4
- xnPpSAkldwHQ9qclbSxMpRa11JztI+U9+/r TWwS3BHPTPeoySytIBkdsUwlSmWbIx196m7urF
- RejfQA2JFO0/WnbhkgLgYz7ZpOkQbjpjJ70xiG CjPA5JrR8vVC01Y8bcBdrHb6Hk1GTtbC88c
- k9qeCQTg9fX1qNmEfmF1znHA71mnYzbTZE+OVyTnt moWYlAQOg4FSN9/c+cnOMdqhyuOWyQO1
- XZNjjdvQY3IC7MN14qFiRIBgZxzUhwW3Bvm9PWoiSykA c5wKlvUbkra7EGQf4TkDFQE4dhlcH
- n8KmLDldrKCeCTVZnLYLHg8VTTew4yitRsg+UbTu9PYmo3A DKBj1OKexAcEHPXP0pGZnjGAAo
- GCSOtDctmRyIiJXlSDuzzjtSPFkDq2R1HSl5ySwJHTI/lQ27dw VAz0PahSl00HGTjezIG/1QQ
- NkmmbFVjyW7A1Lg7MdRnP0qPYCHCkqpbrnpTcrIUX2InG6Tg4cdc9 KEyqjcCpPB96kbAcgr0H
- B9aYR+7BwxGM4pSldFtXepAWKv1xlc4qAtnayDDemanckS/dzjp9KiZQ qgMTg+ntVJkVHFWTd
- iuQxQn5iOpx2qB8/MAwBY9D1/CpnPyl92Ez09aquTztb/cNJe90DlS1uR5U qM5J7+1Z5YbiGB
- 9jVuRm5jUZ+YZPr61TZAFDbScngD0pyavYSTtc87ixsQKeec1bGFkAABHXFVVE eGVD1PAHarE
- IXzVdcjj5iTVuV7a3Q4zsrXL0ZGwFR0PHtVhRluCc5yQD2qqpyfkOBnknvVuJwpAB DNmjmdrk
- pxT0JwvzblBOexqwuSuCAORg1GxYxZACkHrUiBccg8UuZ213CKb1J4wc7WPy+pHWpesi gnAx3
- 7mqyHMyhmyAMfWrAwYgwHI9KLNOzHLe5MpZiFA2/L0I6VMpfy92Ax74qFEK3GSTyKl2soAG QW
- Hc0khtxvYsfwDOemCfepg42luhJxk1AjbSdxUknke9TKA2A3A5ODWeq0Y07bCgsfunJ9RUqFto
- wQSSckimxhAxxuAx1oVgZAT8v8PNOydgt0SFCfMRlyoGSe1SAbWUg7gOtOU5+RjjJ/SpG3kHao
- Ax 6VEm7ibb0sEe5pl9WP6VPG6rMAOQM9KbhgDgcjgZFS42MQMZzSdt0OmlLRk6KNnbGD1qfLG
- PCkHA HBqBAzIvb1qwF4AXk45xTstFcTjrrqSKuCOeCcilCsJFJI46cdRSD7ucccZqQuCOCMk9
- xT1T0Qcz Ww/7pJDAsTVoEBME8DgEVAGAO9zyFySasRhfL6gA9aTk1rYuMu5YTaxXBx3ye9Wt2
- EUDOD1NVkOY 0Cge9WAm9VVux4NTz2SbBSu0pFuNj5pXv296mVtrj5s8cYqpGnIOcsTwAasoGO
- 1gvfJ47VpHlLdk X42/ixn6mp9w5YHPYiqiq52v0XH51KuTwpyAMfWhW2M5RVxYx84fIwRwDVl
- eFAAzk5qNUBReMEDF OG7ewOQc5z2oqT1Vyp3k7IcMSDAzx+pp5RgrOoBbHI9cVEnXjIXNWTnp
- 0PbNEk4kt9GRkBsDjjqO 9TZ4Hy9+/eoVADswyTmrC/NtJBA7kdqzfmXJJWHgfJ0Aye/anAMmV
- fBBHQd6a2FOSS3vS9cAKWAH WoSuHKnsxrcIOABngnvTAGLDPzZPb0qXAaPAwwz1xSqsm8sF2g
- DtWnoKM+UY4zDuOcg4AqJgzDJA Crz05qwpyQGIHHpUZ3BieAOmCKIprcIbtkeDuBxgdeRTScg
- 579KmGSF/rTHUjAGCR6UpJXHFvqQy sVxgY4wajA3YBOPpUx5cAjnsKT58j5Rj6U0tBOViA/cA
- 2sB2+lLtJ3K65QnjHepmXbGV5bPSohu2 YY5XOFIq17y2Ji7bETDkAYVccEimlWEYCjJBqT5WB
- G7GTwaQ/e8vuO9Zv3XsU5PsNziIE5zmo9+S 3JxnkUuG3AE5UnimswHyMMA9CKtJkpxTIyq/L8
- 56dzTkILfN8x6AU18bhyO/5Uwbig5ByOaJ2a3N LpbEzPhBheD61EwJwSvy5xigMxIUMoXqCaY
- pVW+ZiRjjB604NWtYjl0JQ5CFcj5egIpuAYznAx2P Wm5J3E8YOM0EER5B+b0qWS7XGFxtLc4J
- xj0puQckndmlCtyTnHVhTAp2sBjHqa0TQ1roGVMuVJK+ lQyJG+4rzxzg0YAPI4J4NNYDeckjn
- J96lysEbt77ERDBVJ2kN7VAygMfmIbPNSSBmjyT3zxUZXA4 OeRyfemtdWJu6K8pG3uSeeDUWc
- lQpUkL1x0qRvlySRuP3jUBKHAGAGPrUt23GrdCJmcPuK9BgDFO WQNER8qjOeaR/vZznj0/Wq+
- D5oKk4xkVbUXZFOKepIW+YgnHtTCW3HjoKUDdzjnuKQLiQYbBXjBo uuUTmrWGniMdeTk+9MJ4
- OMnNSjHkr8pHXgnkVEWO7GKaV9h0+wmNqKW6kdTTONjYJAx1PQU4McKv 3hg5FI20dTgHsam7u
- ZXalZlUsSm3BbnOc9qZIpB2McjBqVyQ3TAH3ahYhw5LY+tDjLmukO22hA4A jC5DA8darZzCVU
- g4qZ/kUMMEHggVBjO48DPFVLSxMdXcpyttQ8nPt+tVXBK5ByoH61ckC7diqSPX PaqTDGRk4z0
- 9qLrSxS+E873DDLjkHr61ZBAZTn0yBVNPMSLLkMw68VbQLnewbLeh9KttWeu5MlFL YtxZMf8A
- cye9XE27lLEDA7VWAX95huBg/jVhcmXPDLu6+lZX1Ha6si+CGfIzsPr61KhYgchWxjFV k++Cc
- jP6VYUETbu2Oc09X1M9k0TbNhwPvZ/Onpy5Cgpz0NMT95yxAAGOalj25JXcwzyaqTdrGu6e pY
- yWPXHTBqQAsm7dkZOKQGP+LIAqQhfLYLu3LgVClZp2JtrqIiuYyM5I6mpSVZGL5OBxikBwxUL1
- pGJUggZBGD35pNu+ptT31LDBBIGbcvy9/wCVOxmJeCfm5FNAKSGRnVs44IqcDLqVOBu5FHM1pc
- mT UXYWJW8ggkZB+WpxH8oIz7/X0pvUlcEYPapVA3kFuOg96h3uKKbTuKu8ybenPBqyPmZecHG
- fwFRK p8sgDBLcH1qwEIwTwONpppsG0x6lQOCORkj0qVeUBGTnsOtIEyoUDgdDUu0cc7ferc4/
- ChSn1uOU MMhuB6HqKcqgHBJPNCr8ud2cjJPvUrR4iXJ5HXFTKdnbuNSkxVCiFSTu9vSpxgfNn
- BBweKhjDNE4 4GSCPcVJgBSCQ+DzSTsF3HZlhJdhK5VgQSMCpkyQobrgc+9VyoI3cg8AAVYi++
- RgMq9xSvFJ2RSU Wty4vEn38EDr71bUs6hQSQOpqmoBw4PzA4YZq0GGFVSPT/8AXTitBa6WLPI
- UnPBqwDsKFSpK9cVX TOFB5wKlVl8s4GT0U+9PltoNp2SLOTuPoTjpRywPPfrUKu2BgEjOalU/
- Jg/LzjFTblKcZfMmxggE fMOTin7v3hAy2T+VIuGVW3bh6ipo1IBLAZxwacprRii0tGhgzgjHO
- 4U8u+WG3cM/dWl24+Y04ZKc MBkVDs9UFpIcqqFBLKWPYnpinDIbbuyG9KaFyvBUHGfpT0IT5g
- u4nsaNWOT6CEqVA6Y7CmsHLDaa njjDOcgHnkCmvHj5lO0981UJpj92/L37jX5iwAcetMVJATu
- +nSnlmA5KrkdMdabglQRyP4setONl qOV4dhpyOMd/yphA5K8MDx78VO5UJnPzMckZqA8rnBAP
- SktRNcqIlU4GcjnrT2+UsecZ4NLtHmIr HaOetOCsVyQdopylsTe0tSPBJAOGGDUDcr2AHarB7
- YOc9hUfJcgEKO5I71PNbcSTTuQNjaNqkf7W KjOBOSzAqR+dSSKVLbRkgjimsOWPdTjGKZUOVO
- 7GMdx+XqOeahLDOD97vmpNoL8sOnzVD97cMDOe TVWSiHKlsN35KnaRtGMnpTc8sScZ6cVL90f
- ezngccVFsHkkNnJPrUc0dWyelkIfkUMRz0znpUZ3Z P8QJ7dqe2ApVidh6+1R4USPubGcc1am7
- rqSko6XBV+TLHgHtQeMlckgfMc0mVDE8njof0pSNifK2 T/EAaTlc0vZaiFiQo6Meuab8vz8+m
- OafuUkZ4brVZztAJHQYFCvs0TFJ6j5JFEeGXnOD61DLJhQM Yz09hTmysW7GTjnio2BJBOMnnG
- OlOLsEbbkLMyhj2A/WomJIU4PQcVLksXyNvOCCKrmTpubsCcUW bb6icbIhD/vgS2QTnHrUTcS
- c4Ck8EipJGDSbeAP61AVyqqwPc7aqST0M5JrXYhYsVJYgYPXFKxOz jGGFMJCyDP3M8j09ajZg
- pDLz754FG0i0uVXHKF3jduyDliDTiwMvUDPTNR5bJLDHYUAZTLYyD0FW pNi57eg4l/MHQL61E
- 5bIAHyjnipC22M54FBcCQggEgUlUaaVgTTVkMVnk5VMHPTHNMfJX7w3HrxT s/6QCFJ9SDS9mA
- AIzxmolP3hSdtCsC7Lt28k5we1QSMPL5UlurAVJJncSAxHIAHeoDgDoTxVx01K dmiLg78gjJy
- BVNiRJlgd3UDPSrbLsUEA8jpmqLuASwDHI70Xv0IptJtdCCRz5uw4UpyWHQ1TkGSC HBLA5C+l
- XsgIzMADjvWax3YUIw56miM0nsXDV3jsefgsdyqA2CMmrYIJBQ9OoPrVE8bemDx8tWou GYEHn
- APtVSXKm0ZO+7Lkb87cjsT71fUdSCAx5wRVJFXaB/FjrVuHzDArEZOMKfWlOF2pJmr6P8y/ CM
- k5G4ntUw7gg5OKrRsGbK/fU5zVsllJOMZpRSvqZuVxdoB2qee2R1q1Fnapwc96gQ9CQeCfxqVS
- ehO3PSrUtGW4uSLIO1/mUN6CpF3NMyg7ecmokGIyZDnI5PvU+R5KgZzt59aye+w3Lp3H4AgHyN
- vI 496OhRcEHoc+tIUyuDkDqM9qfGr/ADcBiT37U6clcUb8trljaXkYsjEHgVJGCMEABc8imAk
- 8dV9q mCfLwGU+9EovUHK2iJV2gSEg5zzzUyqiDPUkcVCnO5SQRjJqUZDoPLPy8DIrNRaZpfrc
- kxu43fLn irURJTDjPcCmooDcYbdzUyFVHGS2cE0m76WJVncevy52thsd6kT5W5xkjqaAuHI6g
- 9falLELgYLd qqEuZWQpaj8YQ/KRxgUuFCjG4kfeye9NXeXxnJ6gEdalDqwwQWOMDFDlYfNdWF
- QcKSpBPGc0+NC6 ttZeeo9KaC2wK3GetKMK+A4I9Kh33Qoxdrkq7gy57DOfWrKoFAIJBPqaqDI
- GMHaauhFIyT8w6iql JrdFyjH0J02qSeemTzVuL7o+6Sc4/wAKpKMsArAnuvtVwMoIO3kdqfK7
- DUL6J2LYKiNQCUYdQak4 bIVl3dsVCq7hllKn1PvU8S+UVbaWAGMUaJeZLvF6EyAiLYzADvx3q
- RQ5Vdy7sDkAVBltxYLu5B9a urnduHLEdBScrPUavFkijg49cY9KeciPackn3pi7mXGw89frUy
- 5ITIPHU0l7u4pNSWgg2hQvJ5pP 48EEc5Ap5X5/lIOeRUjACIc8HuaTbsKLaV7jVXqcc5wBU2H
- BALIGbtiq4DBCSAQORzT18wgMWXP0 5FO19bml2KU2S7MnB64NOHQgAbS2eaaH/eZyC+O/ak3M
- 2FwPU4FJvUqbskrgAPNy2ScVLujSMdTn oO31qJQyynOCMc5605l6HJOeAPaok13M5STdhPmbI
- KgDPWmncI8feHsKk3nagTHTkmo3yF3D8QKa auJybeoE7iWyDTSV8gZbOcZA9akDAgnYQ2PmNR
- Md6gEr1p7ly9NBhUFyfmGcVWJAdN3G5Sx9varT kAnad1QO6nICgYwCTVQeu1xOV9LXRA25lDt
- yQe1RcKflbLHn1qct+76dT2qry78HC4xnFO7e+xSl aOpICokwcEc9BUPl7W4PGM5zS5AJCtjt
- zSFl3lmPTr2FKO9kzNq5Ex2oFTO71J4NRlgX5baQ2SKl b7vp+FRyDcAcgYHJIpy8h82uhG+Sv
- Xv+dIMrIQenFOJPlhQM49RmmseORgk5B9KTTaDV3EO7Chuu MZHalAKklRnHQ0jOhHIYkYyaaA
- AM5LZGeDxT3uNu0dR2CS2c5x26VAyjJGCRwMelTDduy+BG1RBR v45GeTnpSunsRGdncQ4VQoB
- LBehNQMWdQVG3Pb0qRwDyxII6EVC5XYBuG4ngU0+ZBZXuyJziY9el VpCpi3A855qeRT5hySST
- 1qCQgDYwwM9cVUddAT8yFyoj3kc54yeoqq0iAfezj3qZshcHHI4qux3D sW6dKpIm1yLPRTkA5
- xmlfDRDhT601gUTaTvXoAOpNRD/AFIYqwGfm56VN0nqU+e+iHdQGyMU9Xyh ABHPU1CcrMQeRn
- IPajcxK7gADyMd6q/NuOUko7DiwAAI5I70mSI0zxnjB60qYxjGD15pA+ZMMVPH 5UOavawlG+o
- 8sBCdqncO/vTH3gglgvHUU1nGNwJb6Um/12kHqTWV12FJtLRaCZOAVB5PXqKrswGA e/A4qRid
- pAHocCq8m4nkAHvmtORNh53IflDFdxyF71Ucose7nkcVO/31BBBJqCVVI2Kc+nvRdpak ucb7l
- Vy5baBkd8VRlYHhvlIORirjMSwAypA6mqLsvmY43E8H0pWQk9Tz8KpjJDFgO3pVqHAtx8pL E5
- BzUChIpujONw4FW12F8E4I4rZWWjRKqPlS3LaffXH3ic5q1HzGrfN16dhVKHkxkZy3I5rSjc4J
- I4J5A7VHLJbMq6jqSqQANox6+1XomDIN3JAPPUVUTmRQoAJHQirCLhlGCAOtFr+opJPUlBUqrD
- Ab r7VYXPDsASTnj2qCM/OwwAM8ZGashQ5BcgqAfu+tN8yfkaaLUdGHkUZxnGasqo2jLbiQcKO
- tRKmR hRgYJ68inqSsiY+7twKE+bWxklfW5YjJ+VcEt/dNTKMIG3Luzk8dPaoVBV+Dn196lUDa
- 3t/D9KV0 nojWy2JMDbyQzHONtTooG4knHHNQJ83JUDPIOKtpvEI3Lyeg/pR0stxxajoJySxwT
- zjIq1FxKuQe mMGoUwQepOMfSnqCZPXuSKmWq1Y3YljO6VVIKnGB9KtooE45xkY57VXUkyLkj7
- tWgcle+BzUXtsT JvZDt+JTnJGe1TrzGVAJXvnqKrZ6dAOpzT45SBuyDn06VVna6C2mhOA20FT
- vA9OopSxVidnBHGO9 RlnKrg4PAp+T0JyAODjrSk4p6iV7oAflUkdD3NTHbkA9AB0FRBXJUYYn
- HFSP9xF3Ddijmje472ej JshpAoI2jgipRu3KVyQDzVSMb1O1stnJJqzGxKk5xxwaUk7jcGti0
- rLn7uCOM1ZyXG48jPFUlYlF Lde4Aq2kgEcfqRyKIxafcIt3ukWs7Y8iTI3DrVtQfmzzz1qih4
- AwMdR71dDkNwcelKo5dR2luiwM LwBxjgk1OnyKQoJPc5qIEFACBjP409VyMgj3oik9bDb93sX
- EwARzxwOaUfK5xkIetRxklxjr+hqT d8w6An1ovyuxLbt5D9+F6cdD9KVgNw3nGRkc05mwoPyk
- 49Ka+PL+YZ44OKEwjN20QDiU+XkjPc9a d8y5bB3E0ijcvAKnGeOhoEjPnkL+FQm3ctvqG1DI2
- 44yOMUb1BAGSSOQO1DGMg8nIOevWmgHYflx 82QfarUrIIO6Ji29BjAwcc+lDEeQCOe4qBR+/w
- C4X1p+cuMlR7VD3FyNvUXdhSPvetB3CPkYY8ke lIWZlzlR+HWmM2VGQSc4x7U1eW5Suh7PkAA
- 84wfeo88KBgtio2YuxXBPTBpWxwOVApyVlqLlVtRj MzEHG1c1G2OR8pB/SnM/zYbkED8KjYnb
- kcjuaq7W+hU1MhLgkgnA7ZFRMyD937Zp8o7gYOM4JppC qSxxnGckcVNnYhxttqRA5k6gEdc0O
- +EVtpIPX2prHKliQD14FIZCUyFHsDVKydxxetxh6sXIOeQB 2pn8POQemeooOGl3g4bFMBONzZ
- HfHpmktdWKT5XYT5WRc5B9QakPEHPXvTHIkIKnnoP60gLNuwVJ zTTuKb2YnO3HGO9KcKByOBk
- H1qPc3CjB5546UbS7ZzznI+lVJdxTaaJWc/LyvTn2qsTjncCM9hUj KB909ByM1GwG5iAC23JW
- iDgtCIciegh5UAj+HOf8arSKoZT0zz9KezNtzjaCBkU1pQFJzyOoHaqb lH4TTkb0WxGzNhQAC
- MZ5qkfkQ5+YZ4PrU5kZyw+ZT6mqrnaFUnJ5zU6MXLb1GM5zlVLehqEswTgd uuOKeS6ttGCeB0
- qs7Yf7rZ579az0voC0VmhpYEEpyAc47ioeWYgA46nmnOxDKyjjZ27mmBSGyCCc c4q5WWw7voR
- lHKgHAyN2acmQoO7JOce1NO5txPUnJA9qe7jyVClSx7DtTdSTtZD57jix8tS5AJOS MUpaPYxC
- nkY96rjO5QQWJ/KkYsEyeB3I70pQ6tEQsP6SgjB5GRS5bzmGR6jiogf3bHIB/XFMZ2ST KsOec
- EVad2kiUkk2JJuaYOHGc9qa7Ycg84OCKc+DEoCkAdvX3qq0owxwPcGh2Za1WiB2ONxB3AcE 9q
- ol9wH8QBzuFTOTI20KykcnP8qqMCAQflwOhppxvZkxtfXcYSqtnJYEdfT0qhNuMrbQCoHX1qy7
- rsIRh1/Sq0j7V27i2eeB0NCdpbWEna7OBjLrcgyqT34q1GvKNtYk8cGq0QO/5zyw5J9Par8ZKQ
- 4O OeFNaKKixSUlui1Ep+UYPX8h6VdiQjoG5PQ1SjOUwuR6k96vh2CrgjJ65FJyadkEd9CdOHH
- zZPoO 1WFKYBBye9V1XEoKsGJ9quRgA7SoBx19aXNd6blOcbk65CBSQT16danwR8qg5POagVm2
- qrLjaMA1 bQqqhiRzwKcrp6oIpLzQ7bv27Th8/NS/MzgNkbR9MilC7SCp5H86k2tkM4JyP1oTX
- cjn7f8ABJUK mJwGABHyk09B8rBcnA596ZCRtA6sOOKl5Ac7eSeah26LUpSbRYiZSNjIV2jjNT
- DhVPU44HpVYOS3 KHgdh14qbBaMGs0rajknFpskTJkPG0A9+9T4POM7SaroMRbG5HHNT7hyw3d
- fWolZsaqST10JsgKN 3UcVKpCSYBPoc1WDYOeWBxUpbc20AMc9/Wrd4vTQcnZ2ROhyWx90HmhW
- B42j1BHpTchF54Oe5pqn bHzwTVpXQ7LcslsJk5G09RSo75IBAye4qBWIX5sjihWxkEgqTkGs7
- LZ7iastNS3ucyZJyMYGKAxW VlCsx3dahYgIdvXsKlUkrgHBbrSdkitbX2LEfJ+VgpI6Y61MgH
- mEBlyBwPWqu4hl4yvX3qWJjjad pbGOByKLahHa5cQncp64HfvVxVBTIK4Pas9XPC5B9asoWUA
- 44oS0ugvZdjQUgyEggLwB7VaQlgwO Ce+KqIyBgRk9yD0NWFbdHlQQSMginH3mRz3diynKEnJJ
- 6j0qwvRMZyOoHeqiHG7nBzzmp0fadwPT pnvUSUraGlu7Lqv84HAOQcCpFTPLNli35VSUBRuOd
- zcA5zVkNgYzk5GTWlraoU79CT+IqzAk+lOG DjPJHUCoULeYN3DE9COtSBwDsIAbqTRPunqTKo
- 3uPBZQT37Ugww+Vc8cgUxm+fgHavSkLYOTgYqI pmlpJXH7gCFw24npSbgVIJOOn1phb5g4PX7
- xNIpBlZ/4c4ApuKWqTJfw3bJozxtY5WkYgHK5Oema YDjIU8Z545oYFkUYJwucipi7SuxqUn6D
- 95Em08EHnjv6Um853bScnr6U0H58nrjvSFvlBwVB7Vcm mKKtqOBbcCMNgc0yXnaPmzu9aQnK7
- kYEHoaYzhXGWww6A0t3oHQa7EggjAA4Pf6UxORxuzt6mnkk RAZU46Co2ZSSQ4HajWzByfLYgl
- B6McA9aYPvYGcEkYNSO4XGQSe1QNJlkHXvn1oTk42Y76CEcjJI I9aYx+ckEnjIx6U44bkHnJx
- moJTtAABxkHNS7kryEcsWLAjoBjvSlMKScn3qMsN5zxk9D2pu5t5x 86YzxTk5dNAVO7JFPy9V
- zt49qUYCYHPTkVAhJhztOfbtS5AMgXOd2atxSbQSbegrsAMLxzxmgFsj CnkZyD0PpTfTuBjJp
- ruTFlfuk4oa6WFKAfMcsx9M8UwkdCc9uO9Mzt+UHqcZ9qiB2uTtNEvImEls h7FtoA5UdCB1qA
- suWGC307VI0hOcjAYZzmq2TlyF6EfjSd+U1jGUU7sVmLP2UDnJFVnALYIJ460r 5KjqSfQVCzA
- uCxGSOMdqmDa0Iv1GEnc2OhqBlHzbvl9z2pzOvGC27GDVc5GSx3Y4xVxWlx2k1dDM qBvHHPy8
- 00u2SMKG3dMdKRSGB9M5xUJIOCTknk+xqFHmTZTk0tR2SsuQMAr1NIULOjA/MwzxUTZV 9ucZ7
- ntR5gV9xJwea0SaWhE6bvpuSMWWTHLEelM8xS6gZzjBz3o85TICSMAHFRNIPLD8ccAYqEm9 ZI
- E7rRakm4lmXCnnrURxj5/vZqMFcsTkEnkZxikkC79yt823nPNN6MnlvuSNJk8YHHB9aqM20tuO
- 4Me3YCpN/wA7AgAYyc1XLsycYwy9SKuF9hW5XqhrFN+QTtHU5qvM3ICjJbgnOcUjEH5FPINRMQ
- xA OVLcYB5qrK97jvHQgZPk7YI7dAaqsCsjgHB75q1ICq8ZfmqTjc53EqCKOfm0voEXdO7OOUj
- zF+cE dh9KvoWZo2ypXb0xWdGSQPfjNXot4QKE4J3DNae0bCSdrMuwhS6hsEYyAK0Yvmj2kEkd
- CKpxkM46 YI5AFXEUmIEMN4GM1EbvYybu9CULmJeAFBwR0NXN5BH8LDgk1XjVSSVyST8wzU0RG
- 4koTim02+Yt O/QskZB2LtVj1z1qyhOeQox61XRsDplQcfn3qyAQxkwS3AI/rSumrMWpOQxLbg
- Cc4BoX7+WcD19q i39CPqQaeOHPQkgcVlGLvfsEJ6PsTL94lTjI6irEbkPh0bb3NU1CrErZIP8
- AFz3q0h3x8sc7uRVS lfRAn1LC8KDnHf6UsbjyyRwM/eqKM8HJOOQOakT5QCOVzihrqUlH5lkN
- lV6EYORilUkgbSD9KgDB 3AY4bGB71LnfGAg24GBUxTiU5cr0Jy6CNskAjoM0oBOSGHTg1CcEB
- iMPtAyacSQgZSG5xTdltcLt 6LQduywyS+DnOaejfvgx5zke1V1ZuQBk+1OVgHJUcAHNNyfUuN
- +m5cJJZRjnufpSYJCsB823oKgD 5YkkA4596AxKHJwAO1TTk73YlK3Uubl8vkbGyAMmpI+ZgDk
- ADHJqmADLw+H9D3qRXy4HIyOppyin 1J3TLSkh/vcY71KrfviRnnrz1qupDfNnvU6fMVGPmB7C
- lJdy+ZblxXQsDjCscD2q4gG1MsDxjGf0 rOU4IBx1/KrcQIj5OBUPR2XUlzvHVl5MmTDcY6GrU
- U3zFTngZGKoxv8Au++OhA61ZR8OuVx7049g 1toaMUihQzKST61JGCy7eAR0zVIM2Bt+bnGM1Y
- DAxnIII75oQ4rtoXg2FI9B19KkjI8xx3H61VQq eTnb9alXaXwThQM5HehWdkNJJWJvNcxgsrA
- 5wD6CnqGWQh+gXvUcZDKFB/E05WzknIIPQ1T0dloR a6FBUgDPvTyxzkABQOtQh1VjgcHOBSKw
- 8woT9PpQrvVjlAnRlCHHzHdTSPkLcjB4/CmsF8vjOAOv rQFQIM7sEHPNZ8yvuCjyq7GrIS2Oz
- HJp5kCH5mHFRLsU4zyBwfamsCAG3AnFaJxbsXzJsl+8MFgx BHSnbsSAEEjHGahRv3OSQWzk4p
- u4MSec9AM1m1d3uNkiyElRwo7DFMO4jAIZsZBo6qG4zSAZdWJH XnFXzaaEuKirsRuyngk5zUT
- OhdCFIwMZ9akdzvPTj261AwO08HcOnHFKyW+j9SkrLcj5YAqRjFMw pXcGAYHAqQ8n5QNpHFQk
- FkBQAE8nNCm0gspDWUFm5xnmomBO7d0B59qlwA5Qg9Ofr7VCTtRiSfoe 9SpXVkyXroMccEHHT
- J4pmQY1XcFwMk0u0MysTheuKiPRiULHGMA1XLrqxqSTSaAyeXFxjrT92Y2X bgdSR2qDBO4MNo
- B4pwZmwQQvrx1FVaSsyZWtckLZTtx6VE4PlkZwueBipN4VT0AHTPpVZiruMucd cE4pRm7jWm4
- 0nbCQxIPIFNZgB8xzwKQMecjg9CaiLDCknGOufSm5N9NAlKLsx+4spG4Edj6VE7bl U5LAc5X6
- 0jN8+ARsx1A601G29BgHORUpNq4p8rSbImIR2AYgHqPSoWJ3YAPrT3YEMRjrmoCQS+GI JpXuh
- +VyNiS/ODt4+tRZwgxktnk1IwYsF3Dbjiq0nB2c4BBq4S3QOXQa/CnjC54yOajI/d5AP5VI ZC
- FcfeUtkH6VBJJIXDBTt9BUpyetiOW+vRDB8wCk5HXIppAMTBSODhaQvmQj7vFQOcgqeBjPHHPp
- VR10RXs5Rt9485wSBgg85/nR5gORj5VqKNS0LKTgHpzSNjIXJVfX6UNO+opqO7HeYfMACk57+t
- IW QuScHA7U3eABxyehqsTlAAGxuzk+lOe3mRTS5idpBtDYZc9agbhz1Cg8Uudm4daiZgAoYdO
- BikuZ 7lp+RDkl8bSoHrUL7V2u2ckZ4PepGV+X5yemKruchQykHr7YrX7VhcuuhHIwzlTgexql
- Ix2Nhhz0 Bq2xBBGRuPI4qhKxDk4yB2p00kyVHXU5WIPgA/Kd3TFacbHYM9CeMCscvm4AVmZFP
- PbNaceMgsTg t096cIppSe4oQdveZfhU+YeeM8CrUf3mYfdPvVeIncdw4Pp2qyqho9xOPl49qa
- 31B1d7F2LiIgkZ xjHerYOVAY4XntyKpL9xVzweCcd6tDccLwAG5JqJKzUuwKVtbEqn5gUxgcc
- +lWI/u5LZ5yBnrVcB t+Bjk849KsIACUK4zycGkpNsi2t7DwckqMktzmnD5hnkc4564qLJLjgh
- exFPBbaMnopyfeqcXpct Su/dHsSzsNwx64qxESyBcgKO4qqB8vUcDp3qdXAI4KkfeqbJ+Zb8y
- yrNsYgfP7VOrYUBmJY/rVJW RQ5JIG4Y96lWQPGNnDdBnvUTve2we0didWywC7SM5b2NWFbMi/
- 3vSqQLo42kBe4I5NO4JZt25uow ad76kc3NKzLgIIO7JK/ewabuwMgg5GcHtVbAC5IIGRg5608
- Hk4H+9mk7l1Hd9hxyuMnGTz9alU4P tnJNV0O1/mDHJyfapI2UrlclT696mV2hy12J94BYYLEn
- I9qkJTyzhtxJ6f1qtyZSHODnqKdgAk5z 7UJx0voJSTdkWS6hyQcfhT4h+9y2eDwKpqGExYsCM
- 8CrCtgEEng9fU1F7PRluLtYthgpcjIOe9Tx bsh0OCfWqZYspPXBH4VYVzs+8oDD8jWqatohpO
- yRfUlQTkAk4GatRnIznORgVnIExgszccc1Yhbk FidvQKDWTjzEWT2NKNmO4NxgdRVgHLrwTn0
- qipWRv3bDGOc1aTy9u1m556U3ZO1hve5cUtuKqOCc VbVSAehPQ1TiK+UD2xVuMk9hgqKcm3r0
- QStLoTJklSwzxjC1Lg7lAIVQOM1ArFMsMdOB7VIrgxqS ec8j0qk29hWcWTpkKMMB6e9SEneDk
- HnBqEFdoK9hkCnZLLlVbJGcmm2+oJXWmhPG+ZmIC8c8iohg yjcBk9KcG2Rg8ZxyPSkDqcDgMe
- pFTFNXaCKiODYO1jgLwR60KN8h+b5cZ+tMYswXaBz0NMLFc5Bc AYODilyaaGyjpZMmZflxwoH
- eoxjaf4mB4pT+8AIOBtzg1AASCoO0g9SaairE+9Lcn2gcgH5l7dBT VZApbhiDjg0zls5bjouO
- KamDn5WH9fepkronybJfMAk5+Yd81FuYEkHPPbpQW/esAMHbQMFkG5Sc Z4pOKaG1FbiYfYGYk
- U08qPmKgDn2pWJVflHbpmoHLAAjPv71XvSe4K+iYhOGyOw6CkyMhiMEjqPW lXOct1DcZpjDL9
- T1zUpp+odbXGM2wl1OXPvUBILIT09aectk4/D0prAbQOrY6elKLj1FfR3ISylT jJCjFRMGVXJ
- OB1B9amcEJtUbsjJA9fSmHLR/Mp+lOUtdBR03GphHLHJZhjFNdnyNoAH0pGKonck/ 3jSO7Lyg
- GAvI9qbb0E0ubUiZ8SHcSQTgYHFNJBBUsu7AIGKc3zbWUnGM/SqxY88jIPHHJzTSG7dx zNghi
- eAOMDrUTsrSA/eHTA707dmM7uD0AqEKxXIXAB60PVXvYmLsrLccVUIpOc9jnioyYtuBk8fN z3
- pxb5twO4jjHamEfeIGB/OhOyTuDhciZsqQoBNMJznO0euRTyPnycAZ60xuWAJGAfSnv0KurWKs
- jFcYPA4zTGwTubn/ABqdii43Ju5/CoGBEXHXGcelSlfQr2t0QSMVGMAsB0qEO5UnIz6YqTlosD
- rx zUMshUkDH9aqTSurERtuiOTb5YycMDjNV9xwTkccHFSuysfm7rkc4qH5TDwCCPyoVt2Cm3o
- MGQD1 BPIJody7M2cLn86YzKYs8l88AGneYwjw6YHvROVlexXNqMZkYKqdcVG+WTb368GgqPu5
- wB3qMD94 R3xkGtoqDs0wtZWuPATGMtk9CTTC43MAR+NGAB7fyqJt2PlGQeR71D1l7pDV72Gqj
- LIdxyR0qJ2O MLgt/EfSnklfvZHrmoyDypXhl7VMoO+pm5JNMrsMxkrgtjvWZNkQspPzAf5NXi
- eOAwYDnFUJs5Zc 5OevrW9JvpsXFOMrtHNwKAFBYMScjHetCF87Wbam0YOfSstFO7cSPp6VfVs
- NgDdu5OPSs4qLd0Ja q5qxvldxwUzzVhY2Mqsp3DHOKrQqRlScjsPWrilhGQN3J/Orje/MiG0r
- 8rLqgbQeQccg9jUiFjuG CTuAxUMZwuHwSefepkJDll4c0ScloKCstSVMA8ghgeecUoblMZwR6
- 9aiAORgFjnluxp5faQGG054 PrSUXzaDdtEiZT8uPm5PQ9qkz8uN25h1+tRIMyckZzjJp7IpkL
- Ek5ORipvFsttbk5kAx90NSugd9 24rzwPUU0FQG3dB1NM3dGB+mO9KnJ3skCT3uTO2U2EfKp6m
- pgVUgJ1B4J71W3YOMFjn9KlKn76hv qe1Nyv5BBPUsB3Eg+RmKipN4+V8HHt2qDLBSQQSOpp4I
- P8SjPNRFue5UoqLLDOcKu0AkdaaNzcEg rjFMUEDcpL8/L3NOyrEhmAA6g0pNJ3QLmXmKWY5+Y
- eoxTg4C9Qfp61CVBG9enYCpDgKTtx9fypXY udOKDzB5I5781OHEjEoM88VWiKbyB+JNTK+0n5
- MDoDVW622CKS0sS+ZsfdkYPQd6nV2JGcdPTqaq 7htUnle/t7ipVQkKwJ2rkY96W+5bk3HYvq3
- yqdw3Z5WrMcnzdm4x0qjGNoBYjGeo71bDDIUfpURm m7WEloWlbDYxwO9SIwyrAH3qBSNp3ckH
- oPWrMZ3rgYJBzVRnZbBbsi3HgkscYPAAq2iN8wAHHVjV FFZl5YDvirIBwu3K/Ng5NEn5iXxWu
- aSsoVFBwvcCrKEAtnP0qggJjKnbkDOcdasoDsUg9PWnN3Wj HZbEjt/CDwecjtVlSBncPmA5NQ
- BcFQVOegp4Y43ngHr7Uoy7FxkraEylcHHBzyDVne7TFVIA61VB woB4PBI9alz1ZelEo66slyb
- RNht+5iMY4wKe+1o9xPscDqarA7WQYzxzz3pzEhwGUnjnFFpJlXF3 7DtByMjHrim7ShJHIJzz
- 3oY7WG7oB6UhfAIyNmeDTadrIcJauw7cDxtJUDk/0qPy24xnceoPamBg rHJy3UUquTIxJx7+t
- JOUdkEnLuOG3zfmDeoxSfLuY/NjPODTfMCyHnIIwaY2GYkZLEYIBoctrkPV 6inAYsMkAcjPNI
- rKr+uO9NHQEEbv4hmgFsDcoGGyaHJNWKkiRmQrnkccGmE/KAOVApCQCoYgZ7Zq PI3YXcDnj2p
- uy3Jd2tGDZGC2cHrn1pgYgZPrn1pc8sXwwzjHpUXykswOc9qS5Wxy0Qrk7QFYMSOv qaiJUD+8
- 3c04hD1JJzxg9KRVz0IwTyPSiVSCTfYhy2ZDIu3ocE8ioy2EAY4APX3qRvl2AfMp61Vc HcB94
- fw+1SrSdmXGLkxjjdIkhODjilIOCScHOPanHBAOctTWweeDx+taXb2JmrvcR8KjBcg9OarF WB
- 5wf73tUzMSncc9fWoGJJIUEnHBPem/dVib2dxkpbegUAn1HekYZbkbexx0p45kUlh6sMVHklWO
- Cc96zUrrQtSXYa0ZTJUg4681EWcOA3TsfWnO+IgT1xyKaSdv3SCeMU+l5BJNIRmAYgjd6H1phK
- rG ASMAdCacQPN5YEY4xUUgHBIJA6e9OEbbmfuN2sV3zv4OVJPJ7Uxcqwyy7duQcdanLNvZSpw
- AOKrn DBsrsGOAaOddTXV+hXZgcYYFgckAdagchs7Rk9Qo61OxCyA8MT1qttOeCO+D60JrbuDV
- newwg7SQ Ac8cjpUBkJypAU55OKldiqbXHU5qqQGlY5znGMd6laXRiovqIwKuwIwAfvAVCPMAI
- fLMTkGldynG QADgmohvZevGOfrWkNVdm0bN3X/DjxzGcMAT3Peo844OSxHPtS7ioIABY8gYpA
- Bt3Akdcj3ppva+ gOpaLQ0BgAOdw6/jSO+F2hTkcDmlLsY1XK/722ojlZgQMjj5vU1XNd3fQzu
- t2DszSnI2gDkY61Dy Mj+NutWS+ZCRjOOpqgWBdwFJOeTQouXQatezRC5LHpgDg8VSc7Fz6nqa
- tyA/Pk5x29aoykkHd8oL fLQuW2oSbscvEXB3N0HUHvWhEAATjHGV5rNgceUzltwPQetXomAUM
- Qxz94U3F3aasT8PwmxGSU3b vmOMe9XEfaoHKjOBjnis+3G4ggjYq45/lV5CCwyvy579qFCysh
- Xu7dS6Dld2R14NS5CEHIJJ4qoA d3UYHBFWVIbjac4x9fehwT3ZK0JFdgeOFzyalCKTnDH0qDG
- Fznv0qSMFgRg8HjNVL+6Va71Jd7KR 8wJPQVKWAYAdCcE+hqujqJRwNygjPapAGweMemai10KS
- 8rEgQK2c8Hkg+tKPuEkY5qNVwCS3U5NS bCBgklQeRUu17FRaHn5lxzn+I+tSoSFIBwB6+tVVf
- ORjBzhfcd6njUM5xuwf1pTWlnsOcblnduxu UghsfWkOBM67h97oBSKTGCf4u+ecUiBmf5uAe9
- EYJq/RCT0vHYsLuIBJKrgjPYGmtIwTG0DI5OKX DDojAY6noTTWfcF44OaElo0aQemmogB24BO
- Ac/SpWHDBWzznB71Dvw5wCASMj3p7EOcjIYcnHpV2 bJcFLqSEKZAo4HU49KcBtAcE4xzk0wMh
- JxnJ64peUCYO5AORWEZu9gndLyLOAxwMFs8VLGWwvXB6 1SXesi844I5qaORiegwBjNU4y5bvY
- bskncuq245XO0Hmrsajdy3zHkc9KooPlJ6BQB9TU8LZ5IJI bn3FTvpHYpu0dy6rkoqjBz6CrM
- YbaT3PcVVTZ1Ugk9u9TI58xQcL/e9qpXtYWltGaEO5lJIG7PQd atKAY1bsRwKqL9xQA/Xr61Z
- ickfwlQeKXJLdMSRbT5ARnag6GrAJLp/e9+9VUfeCcHnpxU5diBu5 7Ad6qFuZ33HFSUtiz5jb
- 23HhjwKeoVQwLfMBjGelVjzjJwOwqUMdw45NJJLbQNVt1LQZWIODtI5N P3oMYI56k9BVNZOgJ
- z3x61MAShYY+hokrEyunqywHUD5myCeo705STjHQAgfWq2/quDjscUuS020 EjA5Oe9EdtirW2
- Jzgq28YYjjPpTcbcDdxjkVG0mTg8sMA+1EcqLMSQxAPr1os+XQlybdkOZwJPmX K5GPpTGYLJx
- 1I9aN5KjcFVevNMbeu45UjORx+lCkirJhkiHaMZ9x1pCMMmWGTyCO1RF22MxTPWgP iE8AEf5x
- R1KvfdKw7euCcMMDFMySmwsGJ6GoSSAOhXv7GgsxYlfl9eKTTuHLtaxMSWy5IJHA9qYC dzHID
- DpxUO7Oed3c4oDgxkkgMBjFVJMlwWo9m3AABgAevqaZtYMTn5c8UEqDhSffmoncbM5AGMcV F7
- 6WFboPXBlHOcGkLnO3BxUGSGU5Bx0AphdnYnjj0qlRT2Y4au9ywx+U8/w9PxqNiDEW6cimLKS4
- LbdvUkd6Vj5iuSAVJ7UJNOwnF3SYwcg/UEVGzH94QdpzgDFPHTYDt96TehfO5cdTQ3dkR0ukVi
- Qs Pynvj3pN/wC5YL1zg56ipyEaL1btioXVgzbSvXoKlyj3EktmyAqythTzjmkYsiK2Rk4PSpM
- KEOSd x/Wq5yHAAyB0qU5PU2u7DmZc5wCfWo+THubnBwMUH+7kDI4yKR2xGAOfWr5Laoydo2Eb
- arBiSCeM VCzFmQHsMfWpc8c9e2KjfcTkOARV3TKtdDcgjdzkngVUk28lsgnoKmztnbecY+6Kj
- b5n3EcEcmpb irNAlZ67FVmUM7gDBHeqzE+XnrjJwPrVqZx8uQSuB+VU2fdJgKetJy5TS7b22K
- 7kc43FTzzVd5dq Y4Ge+OtTSBQ4CnHt61C5VtpAwAeAeorRxW7J5o3vYPl3hB+vvULZIC7Tx1N
- K2dmWyQDmkVkZujZ6 4rNcqRN1zXEEny8DvlcjtURkG4b/AJVPb0pzkYK5yWGQF7VCS0jbSR26
- itb2d7D5Fa6HEkMSD26e ntTS+dwB2kcn2PpTG2iUjOPm7/zpp/iLE57+5qtRJXsmSFsBGP3jz
- VRzk+hxyfWpBvaLocVE5UPG OhBxTp3TEmkRuwE2Wxg+lULggN1wMA81ckZQxU4K5+UiqExOeR
- uQjrjrQtdNrgpdzlLcLt2nue/r V5CDcKT17g1TVmEYBKqSCcEdKnjyp6goRxn1pxmpRV9xcrR
- sqSIwFwcHnFWY2YcBwQKzF/eR9Tw2 G9quptC5Vvm4xnuB1qFbW7ErPQ1g4OC3c4qZJcSbkPAH
- f+VUIwSmVPfuasglUzgjjOe1WlGWjMuV bssmYmFRgcYzxzUivuIXlXHBqHLBjnAA6kipQqgoc
- 8k4J9aJuNti+aNuxOFUHO3cR90jvUse0yY2 sGAHWq4DKxDAlQ3B9KkQcr13Dk1k0mm2UpO2g5
- QDPySd2fyoO7eu4nBFI3EhOcL0zT2zxyDjnFCY 7dmPGDjAO5epqQf3s8YxioEJ8tTzyOR3zUg
- OUwzYHXP0qbu9rijpLQnX7xLHginKwMeXGCTUKA7Q Mg7jU552pj5geBim7LSxd3fcUM28bySN
- pwBSfOCAD25GKjLOW4+THHI6ikjU/OfmLA8VUYpapgpv XUnBAj+YZYnj2pW2FxtIA24z60Fw2
- Mqxz/FTCVYZA2gHPX9KNXrYyTe9iRGRdxYEgDjFODqUyAWV uuOxqH5ZGbAIBHTPenKxCJzg9R
- 7Vnyxi79S0kt1qyePGdrA5xjJqWNV285GTjmojk5crtxUqkG2Y k7z14FP2lxxeti4BmE4557V
- ZR8EYZeOnpioIyFOBgAjBz3qUDg8AjHbtWSnG9jaLi4l8EBV6MT/d 71YABbI45qjCFb7pIGOR
- 6VawAB97d654q+ZNkpR2vY0oxmHk8dRnvU0WDJllwM96pRD5gCe3HPtV 5WXCBlOSOue1Jc1tD
- Nx6LUn8shixb5cZODUytmIdMDoTVZDuBwdwJzipWKiLGCCCCaHZ9Q12LHmf N8oUEAcmpFLlmx
- jAPp0quDuUkDL5FTLkPgZG73pylG4XdrJC5AHzcMD1/pUqOcDLDkflUDZMzIeA uDk96RDgtxw
- OgzV8nPqXJpKzJiWLAnB9cVIZG3HLA9s1AN23Bz/hSY2p3Yk5Yil7t0Q5XsiZD++D MwPHPvSg
- nBGB1yCahyHwRkDHQinA/MduSDyD6CqcmU273uSbsyKcbgTSOxy2eFx61Hu3IMHJA59q iJBY5
- BIPIJrJJp3C33Em4mDryp549aZuA/2Wx0Paos4yM9emKXdn+JQQOBjvTkne4pX7A0uZXB5R ue
- D0pA2EBDHA61Az5IIQ5Jwab0Hy/Pxzmh9UEGm9SYuBk5xjj61BG4JI7Z/E0zcMckEn17VCXKue
- wzwR0rS9o2YOyLJkG4rjn69KYGCtlhnnj0quXbeSB8vGfembgWVhnJPc1HvdGOfTm1sWgyqAc5
- 5z SkqVDD5SeeKr72YLhcADoKRSzsRnr93j86Ol7hJR6FjLAAgDIyMAU3ewjQFSF7ketRlmwOm
- WPIpy ksm0YUA5we9Ta24003ddBWcA5/p0NIrAKC4UkjsKQkLxtLHdwR6U6QgQBgCAOORVwu1a
- xKvIYGHz Y5HsaiJcyZCkc80rn5xkgITkgdqjLeYCoODn8aU421Y5JSkrobI53YZCee1QgrwcN
- 05Ge9PYqsIU /M/cUw5ZAQRVr3V6kPYYxAm7n1wcUEgRMU4U+v8AOgsMKWIzjkVE6kng4Hfnis
- 5PYbswZioBOHx3 FR7gz4Gdx7D2okKiJuTnoTUAbEhAPAAJwOa2i9DSUdE0xzk7y2OAeMjp9aq
- vuZAM9QMAVYEm5Sg6 EZqAgeVgj5T3704zsmmjL3272sRHgnLDaowc1VdiR8hHP3j3xUrqC2GY
- sSc9euO1VmYZZh8ox0NZ JXC+mpAzjaFK98Bqru2A5XGCcZxxUhy0o+7jGM9qiYbot3Bwegq52
- SL1lZCMvzjJ+UDmoCxzkrgA 8mlYtglgctyfbFJIVXf8w3HHFZp9EJaaDfNydwPU56U1RmZstt
- PPXvTQSFHzDI6D1p24hgGGTtNW 9CXGw1SAdg5XPJpPlDFEJPuaa20LlGPH60Fj8zgj3x2oe+r
- G72uMbgkk5QnjFR5DjAPQ9+tBYHcx Ofl4AqM+oPPc+tW5XVzJS11K8u3e/GFU9aoyktje6g5/
- MetWJGYI2RuGc/Sq0pVTlFJ479jVOTlG yZautf8AI5SMtKRwR6A96uIMRkvztzgVQi3ZJzkMc
- DHarS+aCfYDGRSg76tk2cn2NGDLg46Edv4q tIWBIUFRjlj/ACrOjkOGz94A4CjGKtJlgNxIXP
- AzTs7astrlgaSEh1O4qDyCatpuEqh2Jfb09az0 KeV3yDgjPSraksDI5Jx6U4prXoRFrYuId0o
- BJDHOM96srgJt5z3HrVIK7Mr5AYcirEeTnuTgDAqb LdBbXcsB2aZfmO2p1yIjtOXzwB1NVeQw
- QnA5O4jgVMpwwYn5QOtHu9LBp3JlbABCtjHPFNLfJtJy yjkAc89KGZxvAPyjHNIoOZGLBvXFV
- G6V7jSVtCdSuxcNtcgcmn/dIUAMevFVQ5WLOFHapFLBtzDg 8E9vrWbpu92aOF47FknCphfwBp
- 4bMYf5s98/yqJPvbicANjrSNIFyAOjc0lByZMVJqyROwVlO0nr gUAKrhQ2GJyMmo8hWYtnO7j
- 3pw+djgEknjFTy9wimtCQFi/JGAccjrSgkvkfdB6VF1YAnjvz0pFP +kDdwo6Z7Vak0i5OKehI
- h/dEcbi3GO9Skny2+XjcMYqsXCttB6HginjcioSCV7+9Z1IpPzGnqtCy WDMx3deasBsY2jO0Y
- PvVPzE3jIyuKmhYs+Typ9qhpJik4vVFoHJ4ywPbNXkPyYbIU/mazyMNuALD 2PWrYKkDkqfQ+l
- Obi2mNSfQvRHPygMBmrCu+/DLuAbg/1qou/goRluoNWkIWXG8EDgiiKT95jcW0 y7ESQSPvdDV
- xQUVQysR0+maoQPjb1GRmrMbnbnJGPXmh/F5Ecr6F4Hy4Tzk9x3pUJfGSMDqMVEGx uZj6dafv
- XaQrAgcHHWk0gu7WJsjzRtJI9qN37veu4E9aqpIN2Oq59amG5WY7sqT0rT4dRqNupYST OflJx
- xUhZVjU4JzwKqxHI3MduTkink/u2wQcdDU/MTUWyRmZioDdjg0FyBjBJI7VArDcCM7s4p4J bB
- bIbg1EnYNVp0LGW2bgdxxzx3pAxL9QGPXiq4O0sOeT60iFlLbs4/h9abcXqg0ZMZSIsdTjBIHF
- DkeXGD90Dj3quzjeQVYgjpS5XYDg5B6Z6Umr6hyajzIWITjIYYNM5MjAnoRzTC5eY7SMk/Lx+V
- Re Zg8tznJq3toilFW0JJCGIAb5hy3tVdnKMGwfmPTPanMAZHYA5LDAFQOT5uCc4PGaFq7MqPY
- eSDnI ZQOBk1A7nkxlQMdSP1p5ZjGSRk96hZ9q/Njbjn1xTjduxKuhWcsgBK9QRjvSbgH6Ajoc
- DvUGcqCO gPSlyhcBj3zwalvoyZJvRonZgAAThgeSPWlVgp6E88c1XEuG+ccnue9CnccqRgdhT
- vrqwTsixE37 3kEHOeTU/mZyEA39c1TEkaz7gWz2p6kFgUYBei+tDjZ3a0CUepbA+QkMCB3xUR
- J3krx7Z4qNiyFl PcY49aUMhUZ4yPWnDmtcLisQVLEdeMmmvnawXHHem4DHhjjPAph8wgYyfXF
- S9VqNOMhJCSEPAzxU OSHb+HnPPSpSNpyPTjPrUEj/ACsAOeBTbVhJdENdhtTZtb+8RUO4AlSe
- e1K4ZITk44qDqSw+4OAe tVpYdl0ZKzA5yTioy2NwyMjofam9X5yQR096iZmUsTjavAoTjazHK
- 0WrClhuwOucZHpTHD7Gw459 ulIz/ug2BuIOAOtQblRck/Nt6ZqeRPYUY3I2l+ZdpUgDP4VXMg
- MZAUnBJBpAzCViFyOhpu/94eR0 xjFXyRWlhcq16jH24Vty4B6Adc1UkkI3JhQAe4709w/PKbB
- 3xVZ93m/Ng88E0RjHVFQ31FILxgZA 44GKrF1D/Nz1GfU1K+4jeO3FRsBtAce+aatorkyu3zDQ
- Dx37kU0LtZiSSccc08ZUqOQDn/8AVTXO 11BGPX/ClOWtgc3JaC7gykg59qiB4yWUKRnn0pec4
- 4A6moZWG0EAYB5I6CiGl0jPmauhQY1QYPP6 VGZG344A9xTSf3TBCAcdD2pj/KQcj0OPWnNt6W
- uNxtqROWxhsbjxnHBqnL5iySE8j+H2qVjtUt82 BkcnpVZyzEnO0gZwTUrnirrcqk3FnMoFZSM
- nIPPsam3MqZyc5G6qyFDtByB61KSwK7fulT1Nb02p PkbFCfOXBOD1YZ+nXFXbeUmND/F/Ks1N
- vm4PzY4AFWomKjA4IGeaThFaW1JaSdjTjZSCFHG45zzj 61MjgySDJOOlZsMrgeUFwauI24ggB
- eOR60rSStYIxcXroaCy5QY5bp+FWklGAOQV61nAbnyCQe5q VQiyHcwI4ORRypPbUtST3NMN8h
- JU5wM5qRJS3yMPlAwDjrVNGBJ35H49qkRwQevqvtVciT0Kv5bF osQNucZ7U9WY4I645I7mq6k
- EAntyc08PlvMAxyciqTSexE2nsSxmQjgKTnk4qT5sNuPfp61WL/IM kkjsKkTax3fMWHbNZOLS
- ci2tidB90BiQRkg1YVhjoMZ9KqAgAqM5wATUwRRCGLcdeKzlNaLqTKLW o/zMR53oVye3NPDNs
- 4YAbutV2UeWQAVwRtzTztb5NxY5HT+VaqXLG44vlSaHnaYiQQW3cnPFCnbJ nIJ9xTPlRDkHIy
- eaQujfN944zgVF27dhqzW1yQyq2xM5OetOUzSOSBhe31qDzNpBMZweoxzVjcfL BQrzyBUybto
- S5vsSA4ZiSd2fwqyPmiYAhXBqiCS+Xz6nHaplY4CgEg1EU76Cv1ZfRxsIZieRg1bj dTM2RgDu
- azI5CJAxPy9GJHQ1c3Hy02+mC2M5ob63K5tDQibDgEH5hn6VZjI6Lwcck1RR+FUN8xGc elWVY
- 7h3GODRF2K23Reib94MOMhcgYq6shBIY8HI+tZiY3jHLY5x2qwrbiRnqauotnuKW2rLgkYf KC
- ckdDVhjj5Mc7dwqisgG2Rh16D+dTGTc6nhvQ09biU1cs+YDtBHG3kjvThKMb9rECqxZhHy6g/T
- ofSnBm8plbkg8epNFkDkm7ItiRWdgSMgYwO9GflxkYYVAM7gDwSckgU7cAcFx0NZKLvfcXXQlO
- 0K Q3zD245pDNu2kcHOPoKrmRQwRiXGM8HrTy2JMAqAOuacoq1maKyWo9WZXBYgnHSpeDzkjnH
- Xiq6u zA4BJHT3pWkURnggZ5FKdrabmUtWmTl03D+Jj+mKiZsPkKykng9qg3MjoY1Zwe4/lQST
- FvOd3YZp xSTsy2rInQZJzjI6c1FvVAey+4/Wog5zvORk460oCnJORgc+/NUopJpkKbUrsVpAE
- 65weTUb8kk4 B64PWlDYBPBXd0qu7MA2GXI6Aj9KlRUmXFtrQC7MeCMY4pCcsSQ3A7UbiAASOn
- YUxuFBJ4H605WS CU76XGNjgYO3tUeflyGUYPzE9qCQyEbu361BuInA5IAzz3qqabjdFNN2J8/
- KOdx7VIoB+YEe1VlI LKd2MHjPepuTu4ycZPNQp3eo5/DclHI5OR1z/SnoQuGGNpORVeMHAJwC
- vT2pzMwj3ZJx6Cq5r3I2 67lnfucjIzjOD3pPMbLJ8u/OR/hUO5XjQ888/WlDK7Es2DnPPYYo0
- S1JSTeqJTy69jjioSSzHk4z g4pxfMTHjp8opm44APB6kd6Oa6uO3RBIQGGM4Ax1qBypJy4Az0
- 70rP8AJ1yQ3PtUZfJwAMepHein a9hra41i+0Asufeo9xMJXIUDBzimtJkAnp1J9aY7Mq4JB7c
- d6bTfQGlLqIznLZ7ntwTUDOC3cknI 9qldm8oAKC3fFVVYb2JIyetSo21auIWORTKQWzkHmoWw
- ASGzx1NTOyiM44B6MB0qoVAlJzv+Xrmq VrdhOXYQsHQ4wr+lQH5lY5A5HSib5ckZBGDTVfgsS
- o6np1qpNdCYySdrETAFT83IPrVZidzNkHtj FTtIARhSQc81A58uMOgJJ4ORRdtFvfyItwYspB
- OB2prL/o4LHljnGeKU4YnjBx9KgYqI+pLA560J pNJEtdEDsfl5AGaRtvmMfmPI6GmuS6rgZwc
- emTSrw4O5cE/NmrltZEpaNtjQ5MZPJB7Y5qFlxGzI cHuD3pWYtKcDJPSq7FgwQsGIB6Ci2uhX
- PZWHO+bfI4yO45FRO2QQx4OMmnsy8YOFPXNRMwwU424y OKm7a02Jb6WK7kqzBsHBwM96rvgxb
- jnrxUhO6Rh94YBzmqUhbywp+bk4Naq0ZJX1Hfn0OaVmBw2O OelSCUNtKHcvIqrHiQ78nKnnnr
- VxDhfkwPTIpKKb5uoc0bNrqTq+VYDO4kfjVrBJzzjg7qrKSrkM ue4x2qwjbshc4PJJ9aG1cSl
- ZbFtWCyEg5weoqyGBjHXLHNUlZFBAYccEVZRiHxjgD07VD91Dbk1c vrgOBndyCMVZYgMB1GDt
- OKpoTgBCCMfL/jVgblxlgSDyxHFCfVCk3uWkYr8p5JGcjoanJwPTjnmq qEgKQRycCnqxySxUj
- 2FUtWJa6l778RAYcenehThixIK5waqKQV3Ipz9aCxymDjHUUkpWZcWnp0L2 7AZGxjtx1oBHlk
- LgHpzVXeTtZ+BgYAp4kB+bgc85ocJLYOVK6SLqkbwrNyVxxSkjYOcDIyKqiQ7g WxwMcDvS7+j
- jle4qfZtvUtKz3LIYgkHLAHpUuWRC4AA3DHvVQElcHC89TR5hGd2WA5FW7WaQra/o WncswLDA
- waiB+fPPHpQZAzEHAGR/+qotxwAMH09qhb6ExbtZqxM8hwzbScfrSJJ1G5Rxx7VXL/Ju H5e1K
- N4fGzI7nFTy2+I2SsXQxfcSdw3DO2pEdhls/TFU1fYjKDwx9KsIxchPup9OtTNSvqZKbitC 5F
- Io2jgluevFXRKTIEJwO2Ky4yAcKQVBJJx0qzC28lyDg5/Cl7jfMLmcmaQbCb8jhueeatLIzHrw
- elZqksSCpVeM81ZSQKMD5gXz9KSSvoyoq2xphwp8wn5c4wOoqZWbPDDJ5OOlZccn77BG5Tzx7V
- dV wMEcZGTU8/Ky57abl9GUjG7d7elPWTqvJJ6Gs9HbqBxg4zU0LuwUEEgLkt6GtL2V2yaitZp
- F4N9z JJyccVMCobnkY79RVIvhVPTj73anKxKgnJ5GeetO8rXuKMVqjQV/lPzDJqMygwg7SWHB
- qMSKof5c 5OMelJhd+7kDvRdaihZbkhZty52gZ9Ke0gyeRjFQAOJSeo4xn0ppYYAP0pWTVjRS1
- 0LBmIfCHHAJ JoeXEhLdevHvVbdjnHQflSblKN1YD35NDjGJPLbct7yqqPfJpTKM4PUHt3quGV
- iOchvSmJgH7rHa MHmp5Y31E9XqtiXeS5JXqeMUrKzSAbdox92oXZnU+x4pu5mdxnGDketXEpw
- 2bH7nLjO3AFNdSpPB L9aYWGGJPWo5pG2BOcj86Uua+hV76iuRvBz0HamBizNwRt9+tBx8rbXA
- 2VGwKsPmOxuQauLurDbW 0RJCBIpz82OcUAEHhgVNMBO4MT14z6VG2clFDDkEZqeS79A5+ZrXY
- mfgLjv+lSEEREr8pz371CCu SGBwB0qMj90x5GW4PahxStYlSu73LasScMw47Cl8zJTuuccGqw
- bDh8hmyQRinq6iNhkKO+exqGnb QN3ZFkkLEMck9famkgEvkF81XD8rn1/KpGkVkJKkDuaTi1o
- Db2JCB5YGTu6ioncHaeQT1/wpuT5K tzwcnJqMOzBtykKTnPvVJcruhRl+A5mC8DJz3pm9lG11
- IZumaU4AJYhR1x3qJ2BbeMgD5sH071Sd 9LCnNRFYKVCg9Bgiq7MFlIc4HT6Uu9ch+vHTvUBYt
- IzldyZArWF0rkpNK4M5KEg9DnH0qENmDpli Tx60skp35UYUcEe1RO4weOMYX1qJPsK1la2orE
- BNmd2R3/hqvwAVB4xkihpBkDn0YUw4ZGyMccYq oRvuNq71GHexJHzY9aiJf5m4PTIpzMyOCAW
- Ug/yqPK+WE2NnBwc0QulsU7Kz6EW0+SwLZ9R+NRs3 y4ZsnPBFOH3/AJQRxyDURwM78Ar2pv3m
- J2bI3I3MAQA3vyKiLHJC8sODTz87EEHnmo1IBw3IzQ2k tR8sVqwHACBucc+1Rs23O0n3zTcru
- Y5O4DkZpnmbc56+9K2l0Q5/IduZW+XHGSeKruQWDZySOR6V MMKBkj3NVzlpCSyZI+UAdKFbuV
- HRa7jG27cLxnrk1CThWHJwOQKkcbtuMFfb3qu0qguFU5HynFXL
- l5dTOd1qxgYfaS4U4KY4NUJA23byzdMDrVl2D2+5c9gT71Rk++cB85yMmiEXF2Jb5ndH/9k=
-UID:934731C6-1C95-4C40-BE1F-FA4215B2307B
-END:VCARD
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/934731C6-1C95-4C40-BE1F-FA4215B2307B.vcf (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/934731C6-1C95-4C40-BE1F-FA4215B2307B.vcf)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/934731C6-1C95-4C40-BE1F-FA4215B2307B.vcf (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/934731C6-1C95-4C40-BE1F-FA4215B2307B.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,993 @@
+BEGIN:VCARD
+VERSION:3.0
+N:Picture;With;;;
+FN:With Picture
+EMAIL;type=INTERNET;type=WORK;type=pref:withpicture at example.com
+TEL;type=WORK;type=pref:777-777-7777
+TEL;type=CELL:8888888888
+item1.ADR;type=WORK;type=pref:;;1234 Golly Street;Sunnyside;CA;99999;USA
+item1.X-ABADR:us
+PHOTO;BASE64:
+ /9j/4AAQSkZJRgABAQAAAQABAAD/7QA8UGhvdG9zaG9wIDMuMAA4QklNBAQAAAAAAB8cAVoAAx
+ sl RxwCAAACAAIcAhkAC1Bob3RvIEJvb3RoAP/iG6hJQ0NfUFJPRklMRQABAQAAG5hhcHBsAgA
+ AAG1u dHJSR0IgWFlaIAfaAAEAEwAJADEABGFjc3BBUFBMAAAAAAAAAAAAAAAAAAAAAAAAAAAA
+ AAAAAAD2 1gABAAAAANMtYXBwbFYcEOZVYuhIRg5LwLIi62wAAAAAAAAAAAAAAAAAAAAAAAAAA
+ AAAAAAAAAAA AAAAEXJYWVoAAAFQAAAAFGdYWVoAAAFkAAAAFGJYWVoAAAF4AAAAFHd0cHQAAA
+ GMAAAAFGNoYWQA AAGgAAAALHJUUkMAAAHMAAAIDGdUUkMAAAnYAAAIDGJUUkMAABHkAAAIDGF
+ hcmcAABnwAAAAIGFh Z2cAABoQAAAAIGFhYmcAABowAAAAIHZjZ3QAABpQAAAAMG5kaW4AABqA
+ AAAAOGRlc2MAABq4AAAA ZGRzY20AABscAAAALm1tb2QAABtMAAAAKGNwcnQAABt0AAAAJFhZW
+ iAAAAAAAAB7vQAAQXsAAAJL WFlaIAAAAAAAAFYqAACp0AAAFF9YWVogAAAAAAAAJO8AABS1AA
+ C8glhZWiAAAAAAAADz2AABAAAA ARYIc2YzMgAAAAAAAQu3AAAFlv//81cAAAcpAAD91///+7f
+ ///2mAAAD2gAAwPZjdXJ2AAAAAAAA BAAAAAAFAAoADwAUABkAHgAjACgALQAyADcAOwBAAEUA
+ SgBPAFQAWQBeAGMAaABtAHIAdwB8AIEA hgCLAJAAlQCaAJ8ApACpAK4AsgC3ALwAwQDGAMsA0
+ ADVANoA4ADlAOoA8AD1APsBAQEHAQwBEgEY AR4BJQErATEBOAE+AUUBSwFSAVkBYAFmAW0BdQ
+ F8AYMBigGSAZkBoQGoAbABuAHAAcgB0AHYAeAB 6QHxAfoCAgILAhQCHAIlAi4CNwJAAkoCUwJ
+ cAmYCcAJ5AoMCjQKXAqECqwK1Ar8CygLUAt8C6gL0 Av8DCgMVAyADKwM3A0IDTQNZA2UDcAN8
+ A4gDlAOgA6wDuQPFA9ID3gPrA/gEBAQRBB4ELAQ5BEYE VARhBG8EfASKBJgEpgS0BMIE0QTfB
+ O4E/AULBRoFKAU3BUcFVgVlBXQFhAWTBaMFswXDBdMF4wXz BgMGFAYkBjUGRQZWBmcGeAaJBp
+ oGqwa9Bs4G4AbyBwMHFQcnBzkHTAdeB3AHgweWB6gHuwfOB+EH 9AgICBsILwhCCFYIagh+CJI
+ Ipgi6CM4I4wj3CQwJIQk2CUsJYAl1CYoJoAm1CcsJ4An2CgwKIgo5 Ck8KZQp8CpIKqQrACtcK
+ 7gsFCx0LNAtLC2MLewuTC6sLwwvbC/MMDAwkDD0MVgxuDIcMoQy6DNMM 7Q0GDSANOg1UDW4Ni
+ A2iDbwN1w3xDgwOJw5CDl0OeA6TDq8Oyg7mDwIPHg86D1YPcg+OD6sPyA/k EAEQHhA7EFgQdh
+ CTELEQzhDsEQoRKBFGEWQRgxGhEcAR3xH+Eh0SPBJbEnoSmhK5EtkS+RMZEzkT WRN6E5oTuxP
+ bE/wUHRQ+FF8UgRSiFMQU5RUHFSkVSxVtFZAVshXVFfcWGhY9FmAWgxanFsoW7hcS FzUXWRd9
+ F6IXxhfqGA8YNBhZGH0YoxjIGO0ZExk4GV4ZhBmqGdAZ9hodGkMaahqQGrca3hsGGy0b VBt8G
+ 6MbyxvzHBscQxxsHJQcvRzmHQ4dNx1gHYodsx3dHgYeMB5aHoQerh7YHwMfLR9YH4Mfrh/Z IA
+ QgMCBbIIcgsyDeIQohNyFjIY8hvCHpIhUiQiJwIp0iyiL4IyUjUyOBI68j3SQMJDokaSSXJMYk
+ 9SUkJVQlgyWzJeImEiZCJnImoybTJwMnNCdlJ5Ynxyf4KCooWyiNKL4o8CkiKVUphym5KewqHy
+ pS KoUquCrrKx4rUiuGK7or7iwiLFYsiiy/LPQtKS1eLZMtyC39LjMuaS6eLtQvCy9BL3cvri/
+ kMBsw UjCJMMEw+DEwMWcxnzHXMg8ySDKAMrgy8TMqM2MznDPVNA80SDSCNLw09jUwNWo1pTXf
+ Nho2VTaQ Nss3BjdCN343uTf1ODE4bTiqOOY5IzlgOZ052joXOlQ6kjrPOw07SzuJO8c8BjxEP
+ IM8wj0BPUA9 fz2/Pf4+Pj5+Pr4+/j8/P38/wEAAQEFAgkDEQQVBR0GIQcpCDEJOQpFC00MWQ1
+ hDm0PeRCFEZUSo ROxFMEV0RbhF/EZARoVGykcOR1NHmUfeSCNIaUivSPVJO0mBScdKDkpVSpt
+ K4ksqS3FLuEwATEhM kEzYTSBNaE2xTfpOQk6MTtVPHk9nT7FP+1BFUI9Q2VEkUW5RuVIEUk9S
+ mlLlUzFTfFPIVBRUYFSt VPlVRlWSVd9WLFZ6VsdXFFdiV7BX/lhMWJpY6Vk4WYZZ1VokWnRaw
+ 1sTW2NbslwDXFNco1z0XURd lV3mXjdeiV7aXyxffl/QYCJgdGDHYRlhbGG/YhJiZWK5YwxjYG
+ O0ZAhkXGSxZQVlWmWvZgRmWWav ZwRnWmewaAZoXGiyaQlpX2m2ag1qZGq8axNra2vDbBtsc2z
+ LbSNtfG3Vbi5uh27gbzpvk2/tcEdw oXD7cVZxsHILcmZywXMcc3hz03QvdIt053VDdaB1/HZZ
+ drZ3E3dwd854K3iJeOd5RXmjegJ6YHq/ ex57fXvcfDx8m3z7fVt9u34bfnx+3H89f55//4Bgg
+ MKBI4GFgeeCSYKrgw6DcIPThDaEmYT8hWCF w4YnhouG74dUh7iIHYiBiOaJTImxihaKfIrii0
+ iLrowUjHuM4o1Ija+OF45+juWPTY+1kB2QhZDu kVaRv5IokpGS+pNkk82UN5ShlQuVdZXglkq
+ WtZcgl4uX95himM6ZOpmmmhKafprrm1ebxJwxnJ+d DJ15neeeVZ7DnzGfoKAPoH2g7KFbocui
+ OqKqoxqjiqP6pGqk26VMpbymLqafpxCngqf0qGWo2KlK qbyqL6qiqxWriKv7rG+s461WrcuuP
+ 66zryivnbARsIew/LFxseeyXbLTs0mzv7Q2tK21JLWbthK2 ibcBt3m38bhpuOG5WrnSuku6xL
+ s+u7e8MLyqvSS9nr4ZvpO/Dr+JwATAf8D6wXbB8cJtwunDZsPi xF/E3MVZxdbGU8bRx07HzMh
+ KyMnJR8nGykXKxMtDy8LMQszBzUHNwc5CzsLPQ8/D0ETQxtFH0cjS StLM007T0NRT1NbVWNXb
+ 1l7W4tdl1+nYbdjx2XXZ+tp/2wPbiNwO3JPdGd2e3iTeqt8x37fgPuDF 4Uzh0+Ja4uLjauPy5
+ HrlAuWL5hPmnOcl56/oOOjC6Uzp1upg6urrdev/7IrtFu2h7izuuO9E79Dw XPDp8XXyAvKP8x
+ zzqvQ39MX1U/Xh9m/2/veM+Bv4qvk5+cn6Wfro+3j8CPyZ/Sn9uv5L/tz/bmN1 cnYAAAAAAAA
+ EAAAAAAUACgAPABQAGQAeACMAKAAtADIANwA7AEAARQBKAE8AVABZAF4AYwBoAG0A cgB3AHwA
+ gQCGAIsAkACVAJoAnwCkAKkArgCyALcAvADBAMYAywDQANUA2gDgAOUA6gDwAPUA+wEB AQcBD
+ AESARgBHgElASsBMQE4AT4BRQFLAVIBWQFgAWYBbQF1AXwBgwGKAZIBmQGhAagBsAG4AcAB yA
+ HQAdgB4AHpAfEB+gICAgsCFAIcAiUCLgI3AkACSgJTAlwCZgJwAnkCgwKNApcCoQKrArUCvwLK
+ AtQC3wLqAvQC/wMKAxUDIAMrAzcDQgNNA1kDZQNwA3wDiAOUA6ADrAO5A8UD0gPeA+sD+AQEBB
+ EE HgQsBDkERgRUBGEEbwR8BIoEmASmBLQEwgTRBN8E7gT8BQsFGgUoBTcFRwVWBWUFdAWEBZM
+ FowWz BcMF0wXjBfMGAwYUBiQGNQZFBlYGZwZ4BokGmgarBr0GzgbgBvIHAwcVBycHOQdMB14H
+ cAeDB5YH qAe7B84H4Qf0CAgIGwgvCEIIVghqCH4IkgimCLoIzgjjCPcJDAkhCTYJSwlgCXUJi
+ gmgCbUJywng CfYKDAoiCjkKTwplCnwKkgqpCsAK1wruCwULHQs0C0sLYwt7C5MLqwvDC9sL8w
+ wMDCQMPQxWDG4M hwyhDLoM0wztDQYNIA06DVQNbg2IDaINvA3XDfEODA4nDkIOXQ54DpMOrw7
+ KDuYPAg8eDzoPVg9y D44Pqw/ID+QQARAeEDsQWBB2EJMQsRDOEOwRChEoEUYRZBGDEaERwBHf
+ Ef4SHRI8ElsSehKaErkS 2RL5ExkTORNZE3oTmhO7E9sT/BQdFD4UXxSBFKIUxBTlFQcVKRVLF
+ W0VkBWyFdUV9xYaFj0WYBaD FqcWyhbuFxIXNRdZF30XohfGF+oYDxg0GFkYfRijGMgY7RkTGT
+ gZXhmEGaoZ0Bn2Gh0aQxpqGpAa txreGwYbLRtUG3wboxvLG/McGxxDHGwclBy9HOYdDh03HWA
+ dih2zHd0eBh4wHloehB6uHtgfAx8t H1gfgx+uH9kgBCAwIFsghyCzIN4hCiE3IWMhjyG8Ieki
+ FSJCInAinSLKIvgjJSNTI4EjryPdJAwk OiRpJJckxiT1JSQlVCWDJbMl4iYSJkImciajJtMnA
+ yc0J2UnlifHJ/goKihbKI0ovijwKSIpVSmH Kbkp7CofKlIqhSq4KusrHitSK4YruivuLCIsVi
+ yKLL8s9C0pLV4tky3ILf0uMy5pLp4u1C8LL0Ev dy+uL+QwGzBSMIkwwTD4MTAxZzGfMdcyDzJ
+ IMoAyuDLxMyozYzOcM9U0DzRINII0vDT2NTA1ajWl Nd82GjZVNpA2yzcGN0I3fje5N/U4MTht
+ OKo45jkjOWA5nTnaOhc6VDqSOs87DTtLO4k7xzwGPEQ8 gzzCPQE9QD1/Pb89/j4+Pn4+vj7+P
+ z8/fz/AQABAQUCCQMRBBUFHQYhBykIMQk5CkULTQxZDWEOb Q95EIURlRKhE7EUwRXRFuEX8Rk
+ BGhUbKRw5HU0eZR95II0hpSK9I9Uk7SYFJx0oOSlVKm0riSypL cUu4TABMSEyQTNhNIE1oTbF
+ N+k5CToxO1U8eT2dPsU/7UEVQj1DZUSRRblG5UgRST1KaUuVTMVN8 U8hUFFRgVK1U+VVGVZJV
+ 31YsVnpWx1cUV2JXsFf+WExYmljpWThZhlnVWiRadFrDWxNbY1uyXANc U1yjXPRdRF2VXeZeN
+ 16JXtpfLF9+X9BgImB0YMdhGWFsYb9iEmJlYrljDGNgY7RkCGRcZLFlBWVa Za9mBGZZZq9nBG
+ daZ7BoBmhcaLJpCWlfabZqDWpkarxrE2tra8NsG2xzbMttI218bdVuLm6HbuBv Om+Tb+1wR3C
+ hcPtxVnGwcgtyZnLBcxxzeHPTdC90i3TndUN1oHX8dll2tncTd3B3zngreIl453lF eaN6Anpg
+ er97Hnt9e9x8PHybfPt9W327fht+fH7cfz1/nn//gGCAwoEjgYWB54JJgquDDoNwg9OE NoSZh
+ PyFYIXDhieGi4bvh1SHuIgdiIGI5olMibGKFop8iuKLSIuujBSMe4zijUiNr44Xjn6O5Y9N j7
+ WQHZCFkO6RVpG/kiiSkZL6k2STzZQ3lKGVC5V1leCWSpa1lyCXi5f3mGKYzpk6maaaEpp+muub
+ V5vEnDGcn50MnXmd555VnsOfMZ+goA+gfaDsoVuhy6I6oqqjGqOKo/qkaqTbpUylvKYupp+nEK
+ eC p/SoZajYqUqpvKovqqKrFauIq/usb6zjrVaty64/rrOvKK+dsBGwh7D8sXGx57JdstOzSbO
+ /tDa0 rbUktZu2EraJtwG3ebfxuGm44blaudK6S7rEuz67t7wwvKq9JL2evhm+k78Ov4nABMB/
+ wPrBdsHx wm3C6cNmw+LEX8TcxVnF1sZTxtHHTsfMyErIyclHycbKRcrEy0PLwsxCzMHNQc3Bz
+ kLOws9Dz8PQ RNDG0UfRyNJK0szTTtPQ1FPU1tVY1dvWXtbi12XX6dht2PHZddn62n/bA9uI3A
+ 7ck90Z3Z7eJN6q 3zHft+A+4MXhTOHT4lri4uNq4/LkeuUC5YvmE+ac5yXnr+g46MLpTOnW6mD
+ q6ut16//siu0W7aHu LO6470Tv0PBc8OnxdfIC8o/zHPOq9Df0xfVT9eH2b/b+94z4G/iq+Tn5
+ yfpZ+uj7ePwI/Jn9Kf26 /kv+3P9uY3VydgAAAAAAAAQAAAAABQAKAA8AFAAZAB4AIwAoAC0AM
+ gA3ADsAQABFAEoATwBUAFkA XgBjAGgAbQByAHcAfACBAIYAiwCQAJUAmgCfAKQAqQCuALIAtw
+ C8AMEAxgDLANAA1QDaAOAA5QDq APAA9QD7AQEBBwEMARIBGAEeASUBKwExATgBPgFFAUsBUgF
+ ZAWABZgFtAXUBfAGDAYoBkgGZAaEB qAGwAbgBwAHIAdAB2AHgAekB8QH6AgICCwIUAhwCJQIu
+ AjcCQAJKAlMCXAJmAnACeQKDAo0ClwKh AqsCtQK/AsoC1ALfAuoC9AL/AwoDFQMgAysDNwNCA
+ 00DWQNlA3ADfAOIA5QDoAOsA7kDxQPSA94D 6wP4BAQEEQQeBCwEOQRGBFQEYQRvBHwEigSYBK
+ YEtATCBNEE3wTuBPwFCwUaBSgFNwVHBVYFZQV0 BYQFkwWjBbMFwwXTBeMF8wYDBhQGJAY1BkU
+ GVgZnBngGiQaaBqsGvQbOBuAG8gcDBxUHJwc5B0wH XgdwB4MHlgeoB7sHzgfhB/QICAgbCC8I
+ QghWCGoIfgiSCKYIugjOCOMI9wkMCSEJNglLCWAJdQmK CaAJtQnLCeAJ9goMCiIKOQpPCmUKf
+ AqSCqkKwArXCu4LBQsdCzQLSwtjC3sLkwurC8ML2wvzDAwM JAw9DFYMbgyHDKEMugzTDO0NBg
+ 0gDToNVA1uDYgNog28DdcN8Q4MDicOQg5dDngOkw6vDsoO5g8C Dx4POg9WD3IPjg+rD8gP5BA
+ BEB4QOxBYEHYQkxCxEM4Q7BEKESgRRhFkEYMRoRHAEd8R/hIdEjwS WxJ6EpoSuRLZEvkTGRM5
+ E1kTehOaE7sT2xP8FB0UPhRfFIEUohTEFOUVBxUpFUsVbRWQFbIV1RX3 FhoWPRZgFoMWpxbKF
+ u4XEhc1F1kXfReiF8YX6hgPGDQYWRh9GKMYyBjtGRMZOBleGYQZqhnQGfYa HRpDGmoakBq3Gt
+ 4bBhstG1QbfBujG8sb8xwbHEMcbByUHL0c5h0OHTcdYB2KHbMd3R4GHjAeWh6E Hq4e2B8DHy0
+ fWB+DH64f2SAEIDAgWyCHILMg3iEKITchYyGPIbwh6SIVIkIicCKdIsoi+CMlI1Mj gSOvI90k
+ DCQ6JGkklyTGJPUlJCVUJYMlsyXiJhImQiZyJqMm0ycDJzQnZSeWJ8cn+CgqKFsojSi+ KPApI
+ ilVKYcpuSnsKh8qUiqFKrgq6yseK1Irhiu6K+4sIixWLIosvyz0LSktXi2TLcgt/S4zLmku ni
+ 7ULwsvQS93L64v5DAbMFIwiTDBMPgxMDFnMZ8x1zIPMkgygDK4MvEzKjNjM5wz1TQPNEg0gjS8
+ NPY1MDVqNaU13zYaNlU2kDbLNwY3Qjd+N7k39TgxOG04qjjmOSM5YDmdOdo6FzpUOpI6zzsNO0
+ s7 iTvHPAY8RDyDPMI9AT1APX89vz3+Pj4+fj6+Pv4/Pz9/P8BAAEBBQIJAxEEFQUdBiEHKQgx
+ CTkKR QtNDFkNYQ5tD3kQhRGVEqETsRTBFdEW4RfxGQEaFRspHDkdTR5lH3kgjSGlIr0j1STtJ
+ gUnHSg5K VUqbSuJLKktxS7hMAExITJBM2E0gTWhNsU36TkJOjE7VTx5PZ0+xT/tQRVCPUNlRJ
+ FFuUblSBFJP UppS5VMxU3xTyFQUVGBUrVT5VUZVklXfVixWelbHVxRXYlewV/5YTFiaWOlZOF
+ mGWdVaJFp0WsNb E1tjW7JcA1xTXKNc9F1EXZVd5l43Xole2l8sX35f0GAiYHRgx2EZYWxhv2I
+ SYmViuWMMY2BjtGQI ZFxksWUFZVplr2YEZllmr2cEZ1pnsGgGaFxosmkJaV9ptmoNamRqvGsT
+ a2trw2wbbHNsy20jbXxt 1W4ubodu4G86b5Nv7XBHcKFw+3FWcbByC3JmcsFzHHN4c9N0L3SLd
+ Od1Q3Wgdfx2WXa2dxN3cHfO eCt4iXjneUV5o3oCemB6v3see3173Hw8fJt8+31bfbt+G358ft
+ x/PX+ef/+AYIDCgSOBhYHngkmC q4MOg3CD04Q2hJmE/IVghcOGJ4aLhu+HVIe4iB2IgYjmiUy
+ JsYoWinyK4otIi66MFIx7jOKNSI2v jheOfo7lj02PtZAdkIWQ7pFWkb+SKJKRkvqTZJPNlDeU
+ oZULlXWV4JZKlrWXIJeLl/eYYpjOmTqZ ppoSmn6a65tXm8ScMZyfnQydeZ3nnlWew58xn6CgD
+ 6B9oOyhW6HLojqiqqMao4qj+qRqpNulTKW8 pi6mn6cQp4Kn9KhlqNipSqm8qi+qoqsVq4ir+6
+ xvrOOtVq3Lrj+us68or52wEbCHsPyxcbHnsl2y 07NJs7+0NrSttSS1m7YStom3Abd5t/G4abj
+ huVq50rpLusS7Pru3vDC8qr0kvZ6+Gb6Tvw6/icAE wH/A+sF2wfHCbcLpw2bD4sRfxNzFWcXW
+ xlPG0cdOx8zISsjJyUfJxspFysTLQ8vCzELMwc1BzcHO Qs7Cz0PPw9BE0MbRR9HI0krSzNNO0
+ 9DUU9TW1VjV29Ze1uLXZdfp2G3Y8dl12fraf9sD24jcDtyT 3Rndnt4k3qrfMd+34D7gxeFM4d
+ PiWuLi42rj8uR65QLli+YT5pznJeev6DjowulM6dbqYOrq63Xr /+yK7Rbtoe4s7rjvRO/Q8Fz
+ w6fF18gLyj/Mc86r0N/TF9VP14fZv9v73jPgb+Kr5OfnJ+ln66Pt4 /Aj8mf0p/br+S/7c/25w
+ YXJhAAAAAAADAAAAAmZmAADypwAADVkAABPQAAALA3BhcmEAAAAAAAMA AAACZmYAAPKnAAANW
+ QAAE9AAAAsDcGFyYQAAAAAAAwAAAAJmZgAA8qcAAA1ZAAAT0AAACwN2Y2d0 AAAAAAAAAAEAAQ
+ AAAAAAAAABAAAAAQAAAAAAAAABAAAAAQAAAAAAAAABAABuZGluAAAAAAAAADAA AKPAAABXwAA
+ ASsAAAJ5AAAAlQAAAEwAAAFBAAABUQAACMzMAAjMzAAIzM2Rlc2MAAAAAAAAACkNp bmVtYSBI
+ RAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA AAAAA
+ AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABtbHVjAAAAAAAAAAEAAAAMZW5VUwAAABIAAAAc AE
+ MAaQBuAGUAbQBhACAASABEAABtbW9kAAAAAAAABhAAAJIjAgAqqcBCT4AAAAAAAAAAAAAAAAAA
+ AAAAdGV4dAAAAABDb3B5cmlnaHQgQXBwbGUsIEluYy4sIDIwMTAA/+EAQEV4aWYAAE1NACoAAA
+ AI AAGHaQAEAAAAAQAAABoAAAAAAAKgAgAEAAAAAQAAAoCgAwAEAAAAAQAAAeAAAAAA/9sAQwA
+ CAgIC AgECAgICAgICAwMGBAMDAwMHBQUEBggHCAgIBwgICQoNCwkJDAoICAsPCwwNDg4ODgkL
+ EBEPDhEN Dg4O/9sAQwECAgIDAwMGBAQGDgkICQ4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4OD
+ g4ODg4ODg4O Dg4ODg4ODg4ODg4ODg4O/8AAEQgB4AKAAwEiAAIRAQMRAf/EAB8AAAEFAQEBAQ
+ EBAAAAAAAAAAAB AgMEBQYHCAkKC//EALUQAAIBAwMCBAMFBQQEAAABfQECAwAEEQUSITFBBhN
+ RYQcicRQygZGhCCNC scEVUtHwJDNicoIJChYXGBkaJSYnKCkqNDU2Nzg5OkNERUZHSElKU1RV
+ VldYWVpjZGVmZ2hpanN0 dXZ3eHl6g4SFhoeIiYqSk5SVlpeYmZqio6Slpqeoqaqys7S1tre4u
+ brCw8TFxsfIycrS09TV1tfY 2drh4uPk5ebn6Onq8fLz9PX29/j5+v/EAB8BAAMBAQEBAQEBAQ
+ EAAAAAAAABAgMEBQYHCAkKC//E ALURAAIBAgQEAwQHBQQEAAECdwABAgMRBAUhMQYSQVEHYXE
+ TIjKBCBRCkaGxwQkjM1LwFWJy0QoW JDThJfEXGBkaJicoKSo1Njc4OTpDREVGR0hJSlNUVVZX
+ WFlaY2RlZmdoaWpzdHV2d3h5eoKDhIWG h4iJipKTlJWWl5iZmqKjpKWmp6ipqrKztLW2t7i5u
+ sLDxMXGx8jJytLT1NXW19jZ2uLj5OXm5+jp 6vLz9PX29/j5+v/aAAwDAQACEQMRAD8A+To438
+ n5HVipGCR1q/Em6JiwOc9PrUEKbY8KCd3SrsUZ UfNk7sZA9ayje2jCbjHboWoh5gIJCsp71PH
+ ErQltrDcfXrTQvz4I4zwAeasLE8bcA4HTNNU07u+o oKLd0yaJFEWQCGwCD6CtJBFsOCSx6nNV
+ wuxE+Q5HC59KsxxuACNufUjpWjp33ZLm0TwpsdgnO3HG fWrSjEnG7nr+VQRIWYqQ3J4q7GMNn
+ OCTilaNxzaT1Y+NWSJQw6Lipo13RM3pgfjUa5RzuO5asxKi kEkYA6VqtI+pLaWhZZRJFGDlyM
+ kY71MgUIecgjn61HbguCjoyg9+lSxoVbDHbnkkjpWaj0fQldri g52+4yaFj2y8AqvbmlCls4O
+ DnAJ9Ksqylxxz0Fa81mrIt3jHTqKkW/DZ5PJxUgQhmByXBz+lAAUA k9D2qcbAFZTuPcntUOUm
+ 7Iq17SYigNGoYbiF7UhGJfQAZz2zUiliduRwdtTkYQhSFXd1NDk927lR Scit85kG3gnHFLhwA
+ FHOc/Wp4wNuVHHc1IACxB7D16VE2+a6QX3uiEJlD8pyTkj0p2394QQSQMYF SnOSMhe3Sm7Cjq
+ mecck1fNeOpMoprRgFOCOetOcMSrHg9elPAxIN2MHkcdqcRltuDg85rG0m9EFu zI/LRWwAScZ
+ Jz1qdYQ6EkYZe1KiFnG0qPXNPwSQpJI7Y705yfRltSe5XdHLbu2enpT1HzorHOBjH tUmxhkkP
+ j+VTJGwYFgNp6k9qXncIzTViAxgAjsPU0Kn4euas+WcOx4GaBHk7sL1GM96q65Xd3KjF W0ZFj
+ aSQQ425NSrGSARwTkA9s0qRqCSenQVZCrsO0MOelRKSTSSM3q9Csy5Us3LnH51IYyYs5zg8 U4
+ JyTtJwanCsEwMHDZxjkD3q4JNle8rscgySOM/xe9O2sMdTjJx9KmSJt7FgASevtT1j3MCucg9/
+ SpcnfVGcVGz1K5A3K5545+tChshh1PUmrYjUxlQCy57elBSQPwowDjmm9VuNP3WV+EYEhsZwD7
+ 1J 5J8hPmBPU+9WFBEhLAcgHkd6MFQvTI45PQVV9dUOzbuiqy7QY14wc59qV41LYHJz61Oqqck
+ 5Ck55 70BAH3N+B9aXMloWm4+RD5fBU8Z70/CGIIRkgCpH+UZYZUd6REClRz64PUVUWum5Ls3e
+ 40KOWPJJ ySO1O2ARk7SecjHrUx4G7aCuc1IIt5G3PqwpXfVA1FSRXXPkruO3PApArlxhgFAw1
+ WOA4UgLzyT/ ACppUuu3B2selWvd3RDTTd1Yr/OZM54PU+1QqDu5OwDocdauldockfTNQAttBl
+ AKdiB1pdW2i3Hm V0tCA9HwDgNjFVHAM24Da2CRn0q80eWPGFIz9DVdo+u75h0yKiN4PRgrRV7
+ 3KgDbl24B5xnvUbB9 x2jAH86uBBgsM5XvnpUBDEE7lJ7itLoEkiIKwO5Dgn1HWlbywxDk5HQ+
+ 9TE7odqDaucjPUVGw+Ut wxPYCsVBbBy8xG65jcHJycDHeq4j3LuHAA+6fX1q9jeMgDJ6HFQbS
+ YsZ6frXRBNJ3JbtuisBiTK8 8jj3pXZyjsOCCOoqfaDKCo+975pxVhHtwAMYJNZ631FzK+xS2h
+ gxYcqvr1qIYDbiCMj86uGPDBWH TuO9Q+UCgC/fz1NaXhs2a+0SVkQBR5me5/Wo2CkMGyvHU/W
+ rRTM23DZzyewprKwKbgSMdMVlzLWx GiaZWZjgncMdAKh2MZPu4wME5qw6cL8v3RSNteQqnUD5
+ qSduhUndaFJ0zDg9R1xUBjDJtwdpXIIq 4yb3L7H9wTTHXbBtTHzdCaq/K77icVoomeYiybHO0
+ d6rSJt6EEdiexrTKtxsXLY5Paq7RMJgSAV9 AO9aRloxwhzbsoup2LzkHjFV5EJCg4Cgg5J71o
+ vHvZQT8oyenJqJok3spy2RkUmna19yNFKyZmkF Qp77TkfWmYAUKwK854rQ8tGiBLLnuPU01kX
+ G/cCc8nHFRey1KbVrWM4oPvAhWB4J7etV2RfkUckN lauunm4JAHJOMVXlj2gEHoM5qKcYq/mV
+ Dkdmys20naecZGRVcqQAduATyPStJAB8q7Xz6dxULsm0 gRk49T1pTcl0M5rXRGaYwqAschhgj
+ pz2pohVSH5DAYPPerk4Aj2k8KPkJqMqhVsjA+8aykoyV1qy KrbdmcPbIxtyuVDk8fnU6rIGwd
+ ofkdO3rTIInVztB4HrV6AMW3Pznnnr9K0Ti/e6lxS2JI0Cpuxu I6Cre1WXBIyM8jtUaBWcFM4
+ PcnirO35gvOSMgY61cZrZjcHGWwi7iuCcj+Ee1X9hXaW4GOcnvUSq DtAIYDgEVbRCBubJwcVS
+ kk77EX5unyFQnapVjwfWrigjkHDcHJ5qFVIjD4G0jgkd6srGGm5IU4ye a05+idg93axKCN6kr
+ vBPIFWU2iR8grn7opIIwTtDAgc4qYRESj5DhelJVOgRkh6K5X5uFHT3NWM5 D56ds0kSkCMEdu
+ hqVEEm4ZAb+lNpLV7DhK716CAgAZIXb6ipCSUYfx54I9KcEGdzfMO3vUwUsQdp 9zUOSvciDs7
+ iRhfJAI3AEDOeasEqVZQhB3YzTFXksFOMAjPQ1KuWl5GOetOUrlNSXoBQHaOjcfnU ka5U8YHb
+ PenlOeDnjipowcruCliMAAVkqivoTKWtyNFwrZHy+3rTRlnYpww9RVlk25yehAxTdv8A dGQD0
+ qlO+xoprls9xrMCB8wyf0pCuWHfHenCPMuzGOeuKmSM7kBIZTyfX6VF9GPVOxCvLZPOD27V KE
+ bIJypB4PtTyquWCgkcdOtSohG35WbB4FKLk7hLyIwrGU7umck461OqnIBABxxgUoV15QBhyRxU
+ 6eYeoyMfLxRe1rpBza3bsQZQjcfu9TUygsrkKewB7U7CBcbcZOOaeyB9wG4ZYdDilLlRLV43aI
+ TG uPmJ6cD0p21sBODjgcdqnxmPIU8cD6UuTtw2D6cYzQ46FySk1cjCfu8MMgHjFAQlNoGVz1q
+ 4qghS VyT2oC4QYVjk5JHalC9rijOMVpoVkUhyQC4x0FWQB5bYG3d1Y/yp+z+JAct/KpnX5Azj
+ bz0NNTaY k4vcr7ZNoyQwHT61NtO44OM/p7U5doVdx5IPSpgCEAGCTxWmluxUdFqyMAhcHO3GK
+ cSfJORkAjp6 U9cFgMH/AGs96fszGV2n1xUzXSwKdyIE7GUjjqPWmMMjoTk8+9TjG7ZyzY5FLt
+ 3KNwwR29KXw9Ac bNELITCVOGbsQKaAxkI6AdzVkgliFU7R3pdqm4YlSoOMZp82gcyk7EGNyjc
+ QwxggU0A+aXCYY+tW lTO3b3PXFKEKJwCcVSfvWsVztIgbe23cPy70/Y2CBnPXr0qVfmBVl4Jy
+ MU9IwAR0HbJ6Ucz6IzTa 1kyLYMDncWHTHU0NkAKFxjjOOlP2Ethn4HpSAbnz144BPNNy1CMdd
+ dSsWy43jdj26UhH948jsBUp QtIwxtIPT0pWX94GUHO386huI3Ho0UXjV3ZS5Uk5AzTRGFkJOc
+ dvarGBvxgk7eD6VHs3RKTuDZwc 96J3ve5bTatcpMSrPgD73PFJJHnHG47cjjvVh4w82GO3NN2
+ lmbkbRwvrRGpZak8qbK4RdoPc56U1 Yssc8Lj8an2FWA2kDrmjaSJF7Acgd6qE2nZAoroVzkRh
+ MHryR601lLK/RecdKsbVQhs5BOMZppUD LDoWyM/zpuUn5lUuVN6FFDhsgcKecUOQVODhifu1b
+ KjcSBt9TUJQqRlcE8Z60m1fUUpq2xAQxG4Y yM5yKiY8jcV6YOBirexgvIyvUkVXKhotuNp7E+
+ 9J27ExlddyJkzGqA8g4+opgQ+YTuyoOfc1MQm/ axIIXtSIqHDchgp4zS5ujJcXazKnl43NnIL
+ fjTCWBJ+UZzxire0OMgYXnPNRMpZQNpx9KtSuh3Kp QH0BA4Gar/NuJVlK5wAfSr5HAJwQOw6i
+ oiBsJI2nODx1pqL6FOS27lRlj3ZUnntmogqtuyCBwTV4 qBDwPusB7moggxgMApOSKz5o3sDir
+ XKJiGAD/A3B9cVG4zCCAADkZx61oFi2F2NtyeSKg6wfKowD 0I6+9a03fWRn71ym0YEW3AyOp9
+ aq7Au4ckdh6f8A160Xj/dDdnPbNQeSCDuJIz69TVWuro0VmUHC 7AuGG3pVfyQ1xt3YBB4NaDQ
+ 7Mt1LHHNRybWK7mULjhhwBUPmbuiZtRdkzLwF2quA5GR9KrvG29lY ckdfStJkJj3BVYLxkVCV
+ dUBxuycnik5a3BN82hQ2c/N8317VAY2JZuMY45q+6b0Jyc+npVRl2Abu F6hu2ayjPmlpv6BNy
+ scWkR3jy2LAnB9quLjesZZX+nGDTY1jLkYZT6E9asDHnMccqeKelryZTjL1 FjYKFB7cYq2u1S
+ u70xn29KYqLlTtPIz06VONznDHCYHUd61cby00RKcXezJgQASANuMgDrU6FimE yVwBmmIMbVy
+ B2JI4qwmVOM454GOoqXZIlOUVoTJhjtY8A4/GrCqC+FPze/NV0CEMNxGfSrKAL8oy W65PaiKT
+ WzE27WluWgxVtnG4kc46+1Wgrn7o4zwT2qqh+QO3UNjOPWrhP7kBWUELzSu001oRy20S J0VVG
+ 7JYHoKkjAAdz8pNNjKsFVVZiF69qVmDBAM5x2rSMeZe87lXT0JFXmMs4GB/KnLkRhgxyR2/ Wl
+ 2guoxyeeakj27UGQp64PaovJO6Qk3HbUWP7/Xv8ue9TkOzPtA4PpQEBnDg8Dr6GpSPkLA8Yxij
+ nTZq7qwJnYA+SSOwxzUwVlkG44OM/SowMptTjB6mpwoZSTnI7Z7VTb6iWj2H5wDlctjIPpTRyw
+ zn 3xT1RSRklcdc1IFCsGzjI9O9ZxUNbAnbWxXCKThSR3GT0zU5XaVXOWPp3pwX93uCk/NyalR
+ RyWyp BOM9jROS66E8zvoNU4I+Vl55JqULnc/IYniiMeZHuHQHk9jT1x5mARzzjHIp8umxq3ro
+ PXBVQMgj v2qQLu3jBUfWpPLOVw67SvI9akXaRxlm74PWnLTUjpdkIiHnsDkgHIqdFYxFmVcjv
+ ilCsMSLxz1N SspDc4xjJFQotyuLV69CGMkPyOnftineWwkIC5FPKDY69SDSKjFzjKgf3jRbRt
+ sbgl5Cbdy4YEYA BqTaCjIOdvB9alVG8xsLlcYA9akCYIAOecDNPmu97FJx7kYwD042kcdqVc7
+ djfMAOB3p3zbgOPfj rUobLDK9tpx2pRi7aohNX1GgKAoIB9Tj9Kbty6gZHGf/AK1WNoQhscA0
+ 9VHILA5746VVkXGViEr8 4VO1WAFADt0xg+1TBQCCRnjOaDGrx/KOSOM9KhN3DmVyodyvlRkg5
+ 6cmnsN8gYqykjBx61N5W5lH LEccdTStGQQAhD9GJqrpyDroQfO0ez15/AUpOTt2HINSiLPAzu
+ we9PIIBADZI4/xofI1sNMhKYkB /g6/So1VPMU/MCw/vVcyWhIBDY4IqIgAqMHjP+RUwV2xvfc
+ RP9Y3QkYwaXG+ZW3Acf8A6qGUlAFw CD940Z287d+ey0cvYjldrpjH+/k/dJzSIcHd/CR8pqZQ
+ Cxdl3Hdzjpmoir8jrk4A9PWtFcE00MIA BLN8xP6VEdwZW4wPxqc4Uj5fl7Z7VEyjcSu7BPU9A
+ aUtgcrtXehCQxcE460wuRKqEdRgVOR84Bx0 /P3prYAUYDY6kVCk7lOGlyFolCbm5btUBBVs42
+ qvT6mpMNwwbryBUwAywyCSfyrR3jZkxauVAoK4 bOM8HNC4yybfmPUipNuUAYZOfzoCsZAOAev
+ /ANamo33L+FXW5WZFEhHX0BpxXzHcgDdx19BUzEbi WXBJPHeo0UtJvCsgHBU+lTL4dTOfe5Xd
+ B5pZuMHIUd6iYZbvgkVbkG1TznHf0qFkzI46D+92qeZ7 iSRWcZn2nIbODzTCpGSMHB61YZf3b
+ ZGAO+eTVd4yA2MnPvQk927DVlKxG6jBJwcnk+lQmMFTjJA7 9KlkBWLnO3pinEfIQpHT8qpNtI
+ NU79Cru2odvA6NSMjcBTkY61JtJhDNjlhkAdKeU+UgOFbPJP8A KrejuOL10RVK7IyGI64B9Kj
+ 2Ah9wPTKk1O0bFgjnbu6d6btAdhtLc4P+NLRa9yb62bKm3EQDA4xk ketMaM43Y/i6Y5FW2UHG
+ Bmo34PAy9KU7i1asirIGKY7kjgVEQSp3HbjjPqKsSKQTj0BNMPzbsYLd 89Kq+qsVfuQk5YFun
+ QAjioyp2KQDyc1aZThkG3JHSo/4FTBOD8xH6UlG6egnLQolN2dvQDnNVGBC 4AUg9citd1YLjK
+ nJxgcVUkiUQMG4YccU4ya06Cg0tWZ3lLu+X7uM4BqrInzttcH6VpHcG/2T271W aMoF55Kk+9S
+ k3rsaK6bszPkiITczAFuQPQVUMSlPmOQvH1rSePMALZLH37VWkiXJ2HaOvNVzWRPN b1OQ4YLt
+ VTjrxU8cQMiylcDrnHenIvzljjcTgAd6nXIjYN0B5XuKxTd9URHR3S3ARE5wAARmpo1Q hiQ2R
+ j5T1BpQF25DfLwQfapFXDuFBwTz64rW11ZsfM7bCoMHOCTjn61OpUv8yjJ6EdqbyoI5JPPW pl
+ GG46EdamMW90Wo6PUkCAN/CBnJOKsKp2fKC7dGqNYgmCT1NXIwPIC9VPP4+lVTcWmupM5K2jJI
+ 1O1QBwTnj6VZjj2jcWyccj3qJUwqYJXnvVlVUBSCBk9D2qV7quZubY5RkEDIAHGKlVeMbSpzzU
+ aq AQc4GOBVhAH4JwelP3+UpxaW+g+NldSDw+MEU5QWlBAx2xjrTEVftDY5I61Nyr5IJPt0pcj
+ TLj7z 0JkY45IBI6YqQr8gHUDpxUaIABtB5PP1q7tGPccYNS0oscFyvQYnzH5cZPTFTKORwRxw
+ PWkC5BOM g8DFSrncFPUcim1bUUnfW4g+VMhSc9c1IY8y7ucA9DQY9/GTz3HSplAEOQDknvT3W
+ gruyY1FyhLn AHIx61LtBjDf7XNNCtySMZ6gU5cnae/92okubqNJLYkCbVIGMY605OeSAPl60r
+ DYM9MYxzTwMvgH APX0ojZ7scZN6MlVCyKXGAv4VKFUSsSQBnK+4qJA7Bg2Qo6L6fWrqrGdxUb
+ l449KblyqzYWdvIjV RvwPnDcn8KmweODgccinEYlAwMdqUoTIoLdOabWt2Jt3syHbmTcOucED
+ uaVtoVQ3BPGTUxjIz6hs 5qTblG+Xjd+lU5K1xrTV7FYJljlzgHOe1TbeSVVs49e9O4JcsMKT0
+ FSKQrjB4789awblbRAoys2Q KSrD5Tk8gnpVhUAKkLuGCTShN69yw6Uqn5ioBU9+atJuzsN2+H
+ uC5VyzggehqVETaBtzz69RTkiM krPj5SvQ84qRlOQxI2jsKeknbqNWTsmO2AH3HamfMpXcBnG
+ Tx0qTkwED0zn0pu3OCRnjinFa6kNN qzDsc8DI5pwB35Jy3b3p3BTIBA6nNAQl+CM96hrW76By
+ WtYjA+YKpw3XB600LIsvIOO4P6VM0Yxu AO49TT8HA7GqUU3ce22hCMkDPDdzSYUjef8AgIqXy
+ y2Rnae9DR7EHoOopJR72BRs1Z7kecJxt56i onGEOB8zDtUyoAvBOKQ7BKvXJAzzTh1sCa2K4f
+ BVf4R0A/nTF5OC+M9qtMIyWO3B6A1CF3bcA8Lx xThNrQJLaxExHmE9RxgetM25cAErg8ZqfGE
+ IYYbFRH5SgK5yOPXNTzeYN3diEjAOBnLcYprIRIVY YGOtT7S67sYYjcaZ5TYJIOc4BpJWepLT
+ tbYYRmIKFI3dKhIw6jnuM9quKH/dgjgc5xUG3Mm/rk/d x0q0+bQtVJR0RFyGVSQcDgUMAXkJU
+ gk5OP5VPtDOcckHtUcyEoSAVJPA9Kq8X6kc3NIr7cyAMDgj jHtRkrICSAT7VMuPkJyG757Gkb
+ PJK5J6YpO70YSd20VHUvHu3AnPboajbCOFOArHBq2w27VAHTp6 UEApjAPf6VU2r2HotEUnU7m
+ Kjfnt6VGwcnbt2kDnjp71aIDMVOdxOQe2Ka6kZ+916VLp3KXLzLuZ +Nr84Y+9H7sx4XhzyRVn
+ y9z/ACglsY3f1qLYBDjGCT1o91oLK+4xwTGGjAbPUCoMZi+YgOOntUxy nVie4x2pzKu7jJDHG
+ aHZLREu97FSSNmdVLDdnAIFMbchCEAsBzVqRS0eSOfT+tRTZMTv6kA4pqVk KMddSmR8wDZwR8
+ xqPgP0zxx71ZCKpyQcBe571AB1yRuHFVKSauXBLUjI5BK4Lc4PVaj2neVcbV6H AqVVLMcfNxy
+ TTSDuBJ68GpgrPUma7kZCMDjIG7hqgWT5dp5OCOP51aZVdODx6VF5ahSxI3H5sY7V cWr+YrNL
+ QiK8DeOT0AqHY2/Zt4znJqwSx9MY4JFRYYHrhRyfrR7NpPQUm7FKdRvHXKjk+tVH+b7o IbsfQ
+ VpMv7sBgOhzkVAy/IW4AwO1EZpoqLcUrGbJBjb1YdiOlVJUGG2Ho2AOta7BQW6nB5qjKWZA Rh
+ cHJOOvvSjNteRK35n+Jx2zLkFgB1x3qwvlvCSDknuPT3pkKctv+ZmH5VbWFfKP8K+1S3Hlu0VT
+ UktBiKcxqMHaMfWpxuMhJGMcYNBVdyYyScZ9qlCjJOfl3Zya0m9UiHT1fMJ/GyhSnOSakRQck5
+ JH 8IPSnCIhiSmSGxmpYgysWABOemKlRvKyKUktiVUVlIySOuM81NHGpmj+96gg96EXeylDjHH
+ Tp9au Ip3cYP0p8zQRu9ESAHY2SBhh19KeFbaVHzKCOlNwjxkMD8w5we9Tqu3Z8hPfg011E2kt
+ h3lg4JyA Dz7VKuQcfLkHPPYUKMyZJ2jqM96kCq4BAZiTyajTVMvVR1YKpbBHIz27VaWMKvJye
+ 3vUIiKybRke v+NWgCFBPJxzTlJP4SVGSSdxIwjOWyRnselWE+Y8fNkcUxAxVTtB+g6Va8sYIT
+ txnNRdKSTKbYCJ ieMrgjgVKsZYNjIG7IzzRGDu4BG08GpVXMiu+evABpzk2LXdkUauHOCMHqK
+ mUbZguCR0JPalYEKw PUdcCnxgNEzEYI9TWU00hWd7oQJ8mCc4HBqQIqKpzu29/Q0oBOFwATzU
+ xBCMduOeRUtONrlNydk7 EcQDIykEtnj29qmaPJXAK4680KOFbBJI7elPCbo1wduFJwfSq91dC
+ rPo7Eq/eUBflxxUwABOcgf1 pqACFN5UsecipkAMg3Hg88UoyVr2JVm7Dgm9t3UEZFP2hFBJxu
+ 55pFB3gKrFgOlPKMV3N3PB7VTv a7GoJvYRow+5lYDP5mlXIZOp4pdjBFUtlRnGKkERBBIbnt3
+ FO91ua83LFJ7BtT5jwBnFO2jPsB19 KkCkgLtO3GfrSgEvgkAEdMcmkrcy0FGzW+gxQCoKgtzj
+ ilwROARySefapgm04UfNnnFKfvqSvXpU 3voTe70/EWNZd3P44qRwfOAAHPUYpEVmkYDpnjH6i
+ pwrB9nU4x9KmWjJu1Ig8vLAFTg9DninKCV2 gd+CalUk8A/MTmpFIbKt1rbm01RTjK9yAAyOfu
+ qRwB604ghAuMMcGpU6kDGD3I5FIqfMWJwVOMno aj3t0hTjeRCsZXJbGPrSYPmYP3SMZqcbiow
+ uQvf8aUhCVycEH7o61oovqGlyAwhFzye3JpPvNjnb jg1Ic/eLA59+lIMeaG2nG39ahxa3KbaV
+ 2RqpPzbhjBycVEUxGoI5PB561YOzgdCOvvTijMSy8qDk AGs3U5CFzXuykw7DnHQe1NHCHjBzz
+ U+MlQwwCOPemMm98DGM9a1TuhtQZAdzEuudpOee1K8ZIjA4 GcZqVdzPtBG0j06CkMTAbVy2Dk
+ k0ot37F1IrS7IGDrnHQjFOKtt4BJH8qs+WPLKnOexzTFVlQKCc k9x0FJz0uZOloVlGZc5xx0P
+ WmlNrLkH1B9qteWeD0JGaULx1DN2HpVRWliZR5VdbFXauQAOvJwaJ NocgDdnpk/pVp4gTkgjn
+ gd6Ro2YLhQB6nnmnyor3U1YoFPn5ZeuAaYYzgAjLDg81om3KgMSOpxgV EFBfKc84PPSn7ttAU
+ le5RcEsdq/IMc1AEZpGypxnj3960XSQEkqcZwABTGDIuSuG7E96b006im2m upRMYDE49vrTcE
+ 5bBzjvVzbwCwJpCpCBQMjHQ1jJNNXHvdGeRj5lznOPlqNVUMhUqX27SDVpk53Y YAVDJEzSDkK
+ DVJSW5KtaxTkwBgEE9+KjI4AwTkVZljjyfm+YDGPemmPKKB/31VruXzqBUGCp2ggA bagkTajD
+ kAt69/SrTnAcFW4P51WkHAbady9vWkpMKl3a5G2DHs7tzioZFbzNoAHPYfrUrhjhlO1Q e9MZR
+ 5rLuJY9MdqqTWmolFpX6EGMZAYDuD60MCVBf0z9aTB3KpHzECjDAEMQCtJQ1LuyMhsZUcdO el
+ JsUplWAYDGPUVKcHpjgcj1NLjMK7gFbkkdOKpS+8ht21Kzrh/7xxx6YpOAuOhz0xUuR5YGRkcH
+ im7kGSfWhQbQlq7MpFSZNxOBjvUbkBdxGQTzV0puk44P9KqyREbSMYBPXvWnkJK7V9ikwwrDac
+ no KoyRjA7FupNarRsAxIJdT+BqnIoHJwxNYqd37o2r2OSVAH4GTjC+9SiIlP4hgYPPWiNd77O
+ hJzU6 AeYcnkeveiEWndijKXzGoCHGep5H0qdAAEfHBHBqRFH3mU4fJ4pUA2hdp5Hr0q/djqhR
+ ldXGkjyj jOSct7VYgwfmZl4OD6Gh1Uop2MrVaSNBGcDHPU9DRNqMdQ91LUBh5o3UjgnIq0oKH
+ LL8nG0+ntUK Bvk3LtYrk8dCKuqSQC/3OpBpJ6aIbTi7t7iCE+XkcEHOB6VNGgx0ZVbqSc0Ko8
+ xtpLcYODVlVxDw 2OOQR0qm77j1T1ZGisFYZQgcDAqzHt2lMYx/OnIMrx2qxGAJMbcEc896xem
+ w48sVdoYqc8DIIx15 pQgXDEcHg896kMbs8gzhQePpUmzKkcsMnJPrVQlpZDjo9BYlcxsqgZHT
+ PcetWUXy4+oJzSxx4hQn IYggCp0T5cBSXHOaiSbIkuXYYiK7cH5icnBqQja4KA4VtuMVKNq4b
+ GWyMYqRkPmAg9fmxVySSByf qVwuGJbI78+lPXaTtwMZ/KnbSzgYYnOfrTRw6sw+XoR3xWerRV
+ lF3e5MFKjOw4HJz1p+A8oySccq PX60IPlDBWwTzk9KaquFJXnBwPU1Lbej0Jv5jWB8racx5xg
+ +gqeMkZx82FI3evtTQvmJgttbpzTx Gylg2CMYAFOyvqbuCfUkiX92rZGO6+lW1TIwQduetQxD
+ 5WVvbOfSreAm0cgYzyaOfo2TNxi9NRMv t447fWpduIvusefu55FQ5YMDj3z1Bq0qsTubPsBVq
+ KWz0Ks762EjjBwwPc5zUvJcAggAZ3HtQPvk 8BfSpQrllHXNS7bszlve43nb6+9TKp2EgBsDOA
+ OaRYz9OM4zV+9vZb65SaZY1YRqg8pNowPp396i bk9LaepajzPRmfg5BKnlue1TMmZcAHJ5APp
+ T+NoCdz370pG1x/Hx2pqLbuK1t9xgRkXeCCCecVN5 b+Q/UH1oQMEVWUk54o2s0yqrMuBg5PU1
+ pZ6CteWoKo3biMBTyaVTmMhyFOPSgxs0PzffHWpApxyM Y4FTy8q3KlFNakQiJYdQQeT/ACqRB
+ hQuQMg4BGakAyoyOB3po5OSmMjr7VTTb1Rnz31vcRQNmRyp 600xqJ+clgORnmpAMoB1Q8ijaD
+ 8zHH+16099UNR1uV2QbuhHp/jQFXaC/BwenFWFUFJG5PHY1Hsd WBbgEdetZ7p3ZUrIiCEZC9y
+ MsfaoWOZCMhc9fSrbgEsGVjxke9QDbnLAL8vfvRG27VxKz2IlTOD0 471GUUkHIABq0QGTABPP
+ H+FIsW4LkHimhQ5lqyExlV29s5yKcFcENwDg9qlGPPK5G0E4HrQYyeit nPXNXSWqvsZ2VyFjn
+ eo4bIIzT1jTbvY4PYetTGIh8hRtH51IqEE5TaeM0pRXQ1bKeHIPy5xx0p6R F2Bx1HDdqvqik5
+ cbDg/Q0rRhIDg4GOtQ5JWJk/spbmf5Q3Hb3PGaaqDewPYY/Grvl7lDZAyMA4pp UGfnhh7VUUK
+ a7lVe7MM7fWqzrlchcAHJxWg0YGdoJWhYSVK4ySMnPtWqilqNNJ7FBl+fAzzVdoix GQSoHb1r
+ U8k5Axs5PXtUHlPg8fMvUe9C0d0yYyd7GcIct3z6VE8e5dwyW9PSr5+7u2kjdULJzuCk HJyPW
+ p13Zo7bGcVLSsM/MePpUEifeHBx1NX3Tjbzu/vYqCQARAEEsfQ1knF7MUU07FCRG3AhR24x z7
+ 1BJGBvYZX6mtArlivO4HNUpPvsCOR3q1JPboHNLmKkoUx7sbiRyAagkGYgeSw4xVvZhRgZbHT2
+ qsy7mIH8XQ0Jx7g+V9SEgDkc+tRBSCHHHPGe9Tsq5ywO7HrUHP8AF8wWko3ZCiloiJgGdjkfSg
+ nI baARxjingMYCP4euMc0mEIB3gY+8K1lBRHzJ6kIUs2NuHz29aCrdcZI4/OnHGVfJPGBj+dO
+ KEqVA IyecnvTSW9ynOzTSGPGViAwCOnHWoXTbg5Hr9atlGVCHXPriq7cI2MZDDnqKE3bUUOaW
+ xHtJTBHI NV5CMgEgHkirjcy/eBJ64qPYm8qCG5y2am9tQta6KbqcbsZz1xWdKFXLMpJA4wa0Z
+ G/dfM2FJzmq L7TMSGAX0PUirjFv3rCglbU5gI4j4252nGBT448oqgdfzpVJVmABGehNTopfGC
+ MjP41Ki0op7E87 WiSBFILqc8HP41Oqh2OGAOMn61AgKguMlu/vVqMAptcYz07YFU6cbrW4NRX
+ QkWPFxgkMAD+NTeSS BlwOfTpTWXaVyCX3Yx7VaUOzZUcZwKlT00exMVZjfKZ12hup/MVaUMAd
+ zKADjBFLs4I6joAvUmpo 1fG35ST1z0xVJprc0cm9LhkqAFAzj045qZIzu5OSq80KAQC4LKRzg
+ 4xU6D98Aepx1pN8q0H7vKHl k7XBwAatJy24gknoaiWMMw56enSrMarGuQ3bvzUWctLi0+YBVz
+ xn73Q9c1LHDlgTnGfyqeJN5DYA 559qkCncBjAPT3odrWW5MXK46OPERJIJAyfzqXy2EkpJxkZ
+ XHepY9oX58cHp6VJtVwMkjBznPWiD aRUpcysiDYcAYIPoamcr5XzIyYbrUwUljkg5GcntTfvb
+ gAGX3qGtSoK2+xERgccseSSOBTNi7Mry QclR71Y45DEc+1Q+VsACFsk561UU+4rx6gpIifeQA
+ TwKQ7vOKFW2Zxu9Klk+cfLjrxxSkseA3XkV KvLYPdWohUxlVGNucZqXbk5GB3570iYYDJyOuK
+ lUZJwQDnafapd47sFG7HwuGcHIIJqYrtUEjLHr UIj+f3x0FWAm2QJnIA7mhRb1iNQSJljUgFs
+ 57c1Y2ttByB/eqCIh5VHXvVlQFOfU80Wa3KkmtEOw fLCLHjA9OvNSiP7rbjkjkUcHAwSMcVYV
+ Ay7gu0k8Amna7uhX8hnlqNp5zjBz0zT/AC+eF6DvUywn eRn5QevvTzFzuAI7EUdRwik73KqJw
+ OwBxzUm3EXylQcd6mCOFYcD5snPenhVVtxIJJ5HoablO+hT ty2GBGf72MZyGFBDbUyoGfQVaA
+ O3gAY60hVyCSFDdv8AGjfcmMYy1IgvGADnPfvUvlZbHY5PFAUF l3blyM9elLsbJJyOMGs5pqV
+ 0LlTVrkBQNnnDdMZpWGCB9457DrUoQE4OOBTdjbeBuOeNv86pK8bi Sa0uR45XgAE8CkIBf7wH
+ J4x0q0wIAUqOnJxTAozyRhTx70lrrsDepDtG0IWGc9cU0g+bjBbH86nO 3eSRnB4pAp39MN/Oh
+ WauKz2ZAc4XbjJ5INQFQxOEJIHGKuH74yVB9cVGw67APbFXDUFHl1KoG+P7 rEg9qeiHYeMHPe
+ pwhRctyp7U8gKv3TkcDjrmhtDjdvQhaLMZKDcQc5pyIVZOOCMkGpEVs9CFxyB6 1KxJcFQMdia
+ SvYSp3umRffBKr3wMVIsMjElmGSMkY6elShQsauDkk8gdKlCKWGCSV6gd6Iu2liL8 qsVvLDqF
+ 79/rR5DNK2WBAGDnpVsR8YwQcdaapYAZIPvjrT1GpaalIoVwB0757U7ZldxXqcVa2jyv xxz3p
+ AoVsO6sMZzVSiujLeq0KjRfMQAVPoaaAWHVSPbg1awAz91PWo9i7cdT1wKztzego766FYqd 2W
+ bPrxUZRSd24sOTnNWNowzFSOw5poT5cDA7E1qlrvYTSKDoVL7cbcA59feqzLuwWyfpxmr5DZIT
+ BUdPpUUiZB2kYbr7VMnfdjaa3RmvtbHynHtVSRFwDtJwOMelakiquFIKkjqaqSg+Ruxls4z2xU
+ cz a0KppddTKnD4QqOW6VVmQguA3zZwR71pN0wc5zwKqug2sfmOTkVV7eQNqKM6XIjBHBIxVdg
+ Rkg5I 4Aq267gcsCR0A71Ds3SMWXaCRn61T1WoorS7Km1uMjGBTTwhJXBPXjrUroVbuU3du9I+
+ Fl6qeMjN Du3ZA4EDDahYnAI6U0xlWOFGcDPvT8HY2c7u2akWMeXnlznrmqlLRGclGG/UrBAXY
+ N6jgdqD8vQE 89TUmAXYchup96UHjdkYA446U1d9DW15e9sNyfNAxkE/pUPAOBjGcgYqdchieK
+ YynfkjOWolG8bh F2TRXIJyAueM8VAAAzNnDn9assA4OzcSc57Ux1VoY/UDqO1EWuVXFdJGfKo
+ LfN2GMVVdQI/ubsdV xzV+VSCwAPzEduaqyZIZsFfqaqN09xt30RzGAJMLjG3gVLGEEMW/qD1F
+ RjcAHI/H1qYHLsevoPT1 p662MG4t2JT8+CFIUHjijZnag5GOfpUwLBlxjaVwvFCEHIzyp61nG
+ pNPRF35rpIlJbJPbjFWowRG Cx6dPU1WBLI2Msc1OoHkDJI9vSnyXVy46R2LCKwk4X5BjBqym4
+ Lk/Xiod6MwyG4461OAw5ByB2NK K0ZEo/zDtuUKg7geRg1ZjU+WeMgcD1piA+YdwGVHHFTRjbt
+ ZeMjBz70NN2Vxuy2JwihcNkEdcd81 OBGG+ZgMdSKjjyVxtYe5HWrJVMJ0PsOtTfQbtzWHqFWJ
+ eCDmrCK0sakcbT0oVB5TKcbhknNWYubf BOCcEKOopw1iDVtLD0QmIsANucY71Kik4yuec0kSF
+ pHboB2p20E919M1K6hyXYIvzcZYHoe30pyx gBU+73P0qZFORtIyD0ppVmlbAANKTb6g1LoyB+
+ 5Tpu7imMjYJzyeRgdqtDhcFc4HT0ppHHHzZ6Yq nIlS0sQlSqk8YA/Ko/KbKMCPKUZNWCreWpz
+ kEcj3pVLbSqr94c8dKhKUXdMcFf0GgDfgsPy6VIqj YAx9/rS7dibQCQeckU5fvBQBnPU1HNdm
+ junoPCrvycBck4FTKrFchhgnqeabG2VZTjgYB9aspGdg A9OnaiLktBx3HJGFB44z1xViNCbgF
+ VO0DAz2poBOAAcdqtqTvUMM5AB7Uc0ktyLvcUJ83TcAOcVM qAzEnK46Aninhf3hO3p39akZN8
+ vPyj1p01Lc0i9fUaqkZ6lieBUyBgpyj+1SxodqrjBweT3qVIiQ OowepqrX1sZyiQ4byk2jnHc
+ U5YlLtvGT2IqYxsYiFOSDg0oRliA4PPPrTXky3tqRIMlh2xT3Rtqc YGOM09FDOF6Lg80/6AkA
+ 49aOV7ia1uV9u9tv3mI5pAjE/N0I5qwvysTlVOeQe9GwBiOpxnFG2iJc tdNCHywGIJGDxkd6Z
+ tZQWC7WHBH1qyCAjAcjOeabyMbs5I5JrJxaditdSEqdoUDH071GinkjDDIz kdasBVY/KrA4OM
+ 96Ty+VxnC9MGr5rKxMo2SZAIyMgDqeSe1OdSBwuQDwKslQfYAdD1pgjxyTwOAK FZopR01Kyod
+ 58wbe3P8AOoWAVioBGF5Iq+yhYMN14yfemsmC3HOaIvleocz3K+C2FI4zmnYJjAIz /M1KEIVe
+ Rzxn0p6ALJubOBkCqnZvTUGVwu1WDKQM+vNIFO8hmG32FW3BHJwf6UhwULdeelLmsiYu T6EeB
+ nIDA+hFCBstg/NiplT5iMb365BqTO2MkDDg5YU4zsu4KWpCBzywGCM5qUL83IBUDjFOdV3D 5h
+ 83r2qMKu9s8jtg0KSlqx8rlqgKl5QUX5TyM1HtGWPAOec1OMoMZIA6AdqaQoRjz9aLvoK1tyP5
+ JeGwvFQuqqgIP3jx7VPgHAPOB2ppTn5iMKck4qnZE31K7Rgjbn5aidh5RIyGLVaPKjjqQTgVA4
+ GT gYPai19Ane5WkwUO3GSvPvVUjEOHDZP3jVzcykEKHGeRUEjP8oIzlc5os27Ar7dDPlUBDyT
+ gd6qh Q4DE5GOlaLKpBBbnghu2KpOGbG3AwcYx1pabLc1im42RVZQMAfMQccVRdWBPB4/WtHcC
+ zYwMYxkV UcNhgSFPv1NRJyi1clU0tDPkj+QsoxjoRVKUtvywAbPKitBowoIfIBHFQSKu7JUlg
+ vBFVfTVaj52 ml0KZA8rHUkVWdNx4OCSO1XnRwm47QPYVHngYUgjgn6UczW24Nte8iu0P7/ZnO
+ By3bNI5bdtVtuD gD2qVl3KnUk54B5qEI7H5R8o4IPWq1t7zH5takYBUMAQ3PIApuTsG5SOMmp
+ 2iCsSScDrTCcp9OmK 0g7aolct9iONgQoZSM8g4psgCjG8D0OamyA2DgAdD601gwTGwnjk1Upa
+ 3JUnFpWKyjM2Mjj0HFQs NpUNkGrMqb4s459qrEMgALDnnkVKSfW7E3e8iGQ5Dbhxk45qkcM+5
+ RkYzzVxtxJBxVSRvLj27eB2 xzTUWjVL3fM5bdu5cnGOmasjBb5Qc/xH0qoqhQyuQ4H3CKsJIq
+ hVJwT61PNFrTc5X8PmTqHDFj8w xxjvUy/eUkblzVbcywkEgNnj3qdJGdQw2/THH1ocnbUtO9r
+ lsMoBKgc9DntU0aYLfOOeTxVdFVnC qdwPX2q2VUZ/hI6AnNSk0xNJK1ydAqbSRxyOKmjznAO5
+ umPSqy8sMg7Satp/ruy55zitG0tLjUep MnDHnBJwc1aViCxwuMDjFVwvPykEnnpVmI/MOMA9c
+ 9jUTit3qhuS5kmizHubO4AAHGMVJ5W6bI4x 1zTVA8zkkDA6dc1PuYBduGDdRilG0dEaJuOqWp
+ biIZUyMnGAR3FWlCrKMc5HNQxJt5bk9QasY3Ho cHr60Kd2Qvi2FRMLnc3XkZqwi55JyAM9OlM
+ wwwCNzZxVpcKi5BII5NJspxaIwGCAhSvzCkO4NIcY GR19anZQ0eAxGeQKiAwzNznPSmttUSo6
+ 7jDhkI54HNR5CDBU4PerI3BMHBweOKMA4LY9MGlJ8ysy tnsVuSS33jjj0pVA8oZBJ6ilB2DC4
+ INOB/eqTzgHJ7VlPRWG4ybsNwwXJztqVI1eQnngc+9NJY8h SwUZwKmjBUHaDu6Gm9tB8slG1h
+ 8cYAVlIPHI9KsqpUsRjnqPSkAywBG0Hmp40Y4YDA60r31Y7skT HIPHNWSmTnBOaIugAUk44Pr
+ VgKCRnuOT2FF1zaISs2EasEAYc/xfWrKAZGcHmnImX5yVx2qcLjrg 8gCtZVL6Gb3uMjAMmATk
+ dasAMFORwMjNKoDBmA4AxkCngcKAwx1JxUX1vsauOl0AjcL1JJ9KFG0Y bIXvkVJltz5I+Xoe2
+ KeD8/zY2gdxQrtXHPoQhQWODz2o34l+bgjqcdak48wgKeRUY5yx6elVGCl8 RUrdxMKzNn74pp
+ B5ZSCdv+RUu1g7BsEY5I44p4iw5wGOzGMd6TcY7shuKGbRtweh6cUzZuBxk461 OVBUNyAOcUg
+ VA+TwGHOegqVKz0E3G2pA0Z8xSMgEYBpmwGIBQwYnAFWvLyowR14B707YWkDKBn17 VV0yeezv
+ cg8vJUhcr60YViQcepx2qTBE2F+YL05qdkUpv4B6ips2y29SiYmfaoOR700pubGCSPvE VcyNm
+ Rgtn8hQyhWyMHjPHen72w9ehRUMp4AGBzTtpJZccfSrZ2luAWyM8DrUQD4JOeveq3VxWSaZ Bz
+ zuXk80pUByGyAeKsnHA3AknnjimOiqAyEhQO/NRe+yFomRB9qrtBye9OC7lc8nJAFLwpC53ev1
+ qVscgpgZ7etVDcmWm5HgkHLKoPHIpiqozk5Iz0p2wZY88nIBpM7fw4OKculgcLIUlQvUMM8Uxs
+ bQ B0J6UhHXdxj7tHCt8x68jntUR0eg9JdAfaCSDjB/OmMOuDkdzUoG5GJI44qDJDY4xyTQr7d
+ Qcbaj PmKEH2PSmO373joD1p5kyDzk9uKiOdhGCxYZAHatNiZJW0KsknynYnUHr2qEhfKcsSTg
+ Cp8DhQck dRUDhxk4BBIzipUo27AovRFaR1wV2NtJ49qruy4wePWrxUkglehPJPBqiVLMpdG+b
+ 7px6U5W0sSr NWuVSueR8ueSTVaVSJDjGR2PXFWn+8wY8fyqq4ypK5OTyc9qvlu9TXZ3KjKTzz
+ z0qu4ZigVSMAZP rVp0G9gAwBxwarMWG0cHA/GsrXbS/EpQbTdyuyHYp2tg9vaoVAOBwT3qSQM
+ yr1X09qjbakxIBAzx n9avl7mKg2vQYygTIRkgdcetN2/OCGUDrjH60/c+8k4BPI96coACse+d
+ wqm31HV5lYjZSWzncoFQ uA+8N8oB6nip+fLZsgI3AJpjnEm3cuTgn2qU+wlK1m27orMCFUBSx
+ x+VNbMkZDvhQw6cVZbAcAq3 3aikAyApHyjn1NUpN6JWDmTXvIrOjAMVO4KePeo5DuAyMjPQda
+ mLAqSoJBODz+tVmz5WBx6moa7h K60KrszHgcjrVd0weeHHUE9amcMMAAnjPFR7gG3OTuz3rdt
+ pJi5mk9NDjY1Ckclhk4qXc2NyqDt9 qrhpRMWG0fUVKoLBN5Kk4wKiCVtWh6W1ZIrSyFQw9uB6
+ 1cVNiYPHTr61WCDBJfkn5cfrVtGVsDna RkUpaPRFqXRLQtRYCAMRz3FTRsr8569/Wq4XbsBwy
+ Afw9asqMOjDHoBRrvuQ5R5diYKckndhTxir kYJfBPbA9qpBf3ipuIx6981eiwG69u9DslqSk7
+ E8ZXcSf4atxIoB5OOvTvVVQQdv97HarUZ+RgCB kd6Tu9i7K9rFkDKqGJyeQfWrUa4b1I6jv9a
+ rK24qehA/KrW3d1ID4AIHem22yox0szSiI81iFBIG M468VYRfnXBHIyQfWqUHAByx46DrV5cG
+ 3yMdce9Q6coshLWxajUAZHJx6Ui8rypIJp4+VVLNjIwf rSlHVSgOQeh9Kuj5lJ20FH3nIxnPH
+ HQU+OMAgbx170mweuOak2bHBYnJPWsp2u9SddiCQAyqOCQM getRyBW4UFcdfxq2RkhsDHqKjc
+ BztIyM8047JNGvw6opjG3aMEr0p467SAwJ6etTCNR04zR067Sf aiatr1BSSV+hCissnB6jlT1
+ xVuLcVCYBJ6eppoXBUgdanjC5zv29xU3VtiZrm1QqowYHGOO4zVmM MiNkYAIH1qOMLvwWOSep
+ q2MK23IDDgZp8y2aKtZEkZBwRkgdhU678BAvPeo1Q7hhGUnnHtWiqsij GDgUKTWwJxWlhY0Uq
+ ufl9asKijhQTzknPFRhTjnk+1Tx8NnaQQO1G+7EpNdAC4fKjqeRTzy7FlAz 0FSMoDhiflJ5FI
+ yoZFzk4HUVSSkwbjfmeg4YECqy+5NOIBU4GfWgKwiOG3YNOLED5lJ4OQDUcqWy CPLuiLYdvA2
+ +uaCoD9Rn0xUm7jHQ459qYUYliGUn2PJptoq4m1TgnJLdvSpNrA5UbgBg4pyYwO2R wTQEGeG3
+ AHHB70tGzPnSViHJdQudqmjywAOSeOD2qUtiYkqRxT02lVyeAMe5qrS3aNZaLyGKpEee G7ChE
+ lckMdo7YFTortIG424x0p4HBA7fr7VHMldEvV6WINiqAARuxz7+9KBxlVJI7VYSIPg9Oxpw T5
+ SF7HiiTRL1ZXEZZCSoUHpTNjM53AD+oq0uPKAz17UeUcdOn6ZpN2dmVJJPVFOOPapY7d2etMZN
+ x6g55NWyCpyB8gqJl3AsMD1HcVUVuwk9SuF8sZUA7h82e1MJAZQG3ADk4q75bmMg7QD047VWZd
+ pb ONhPFQnd2QWTYwEmMFgHOeMDt60jH5G2jvnHvTmXLjnGeDjvRsweOctiqtswirDQDu+cgHr
+ UDKPM IU85qUgrGAw7kdaXaSm4FQT7d6ak4vUhy1uRKocc5ODUe07sEEkH5cmpTGynIdeOuPWm
+ kqOW6ZOM VUXqW72uMDsuflBye3Smg5U9h796kC/IvB55qJicjAx6ZqHHrbUl6sj2YZm3A/3eK
+ h+UPnIyM5NT uGB7HsDjrUL8IitsPckU0iZOxXZ/vBRlgM8dSarNllBww4yee9Wn3NI23A4yOK
+ idiJFBHUdaFo9E XIqtln28mPqaqPvxkk+w7c1eGwyDaedvIzUMoIySc5I6DpTcrbg5JOyM9lG
+ 7aMbcYyahZG3ZwVxV mYMQfmG44NVnL7MZ4GOPem17oc7lqyo4PlhmPJPTvUG0AvjnuAetWJFb
+ eCQWAOKrudr5b5cjj6UO zVjRp21e5AyDb5jbunK1FIC8KnGORnI6VZO10DFiD2FRE7l2nqR0q
+ rS00MG0upVIyhZM5AOc0pGY 9+04IyamG0M3XpyMd6Yy/uxgNucDPoKTTk7GvM7bkW0AxlsgDj
+ BNI4ILYVSfUjpVoRAHghiOlMyN r7sY5FVdOWphKzd2yg24yE5zkYxUbxnJIzx0NWnQhAcgnHG
+ B1FNO4xkk4XGF9aG23p0HzRVuUqMC FIGM9yRxWczvgkD25H51fdWWQ5yQx6elQOVKkAAkrkjH
+ elGV9C9tTPJJU/KzkH17VCwBRhkjPTNX MAnBBGRyAarSZ2hgQMcEYpxbb1By10RxAR2DHaw2n
+ vU4wEDclx0FRZPnbVHB65NSr/q/mwSPSnaa 3WhnGMpOxIkYcqSSM9BVpOAAgyqnGfWokRTJye
+ /SrUaFM9CSc1LlZ2RUn/Kx6D5k3E46H1q2QTMv zDnt6UyPY0jZHfr6U9UbILevH0rSTbI5dUi
+ wIxuDHd+BqyobrjPNQxbVcjdk+/NTICrBdrgE5JJr K8upspa7k6glsAkHHFXUXDIQM4B/Gq0b
+ KNvRmHJx9atBQNrBjgHOParcrWZLm5aNXLShCFYkg55A q6FGdzHn+9VWMANuySAMgetXI+Y8k
+ YO4Yz6U7q17ii2mWkAVVbgn1FX0j4Udx6iqaj7gHJ3VeiBc 5OSOq4qJau6HfXUsIw2gn5yemO
+ lWUzk4Xgd6gj4QYUdenpVgYJYsTszwB1py2JaSe4w/cGfv/Snq uSpY8gZ5HWpAAyBipCA85PW
+ m8Ky7eR79qhtWdhJdLDFUsxIZcdcYoCnBJUkAcU/bmQtwVHXbQwBD Ku7pwM1Tir6s0TuiLAJw
+ x6DnFRNwTwc9qsBMQvkEnIwaTaRJkjGBSja9nqJct9AHysw4J7HHFSIm HHTBHFNXAycZOOpNW
+ YwrsgByMZIrOTsrlNW+YRhUm5G8Hgn3qxg7iDgcYyajYbQchWXqNvepskjb jt8opxSdmPVsmj
+ JyAWB9eOlaKKTlgRgcEVSjGcHaScfSr6qCPQEZNXNx6E3k1ZE4BXBYZHQU7LC4 BYZBPUUiqRG
+ AgOc5INToEIGckg4UZ6VKskNNpa7j2GSBjjGQaYNocEhvb3pyZUYLAH3puO+B6del LZgiTIEn
+ Q7j2NJ87yMT0xmmgsOh4A5yOTTgrErjOP50k3uUoqIMGRiw+6Rkk01sja5xu64p0mTEQ 3AzTC
+ 6tHySCOo9adkyYRfVCnJ6nnPrT14kBBAznioYwc7gc46A1KhJcswBXODijW+5vbSyQ/hijb Xb
+ Pp2qVVTzM+vWmqpUZ3LjPSpgqFxjj+9k03boZbxtIMLtAyQc+tSIAYzn5cdz3pCwLYHJ7Yp4AL
+ bW4XPPNZuPViau0mRgMsxKsGHoKmCEkYU8ZIHpT9pU7sYXPFNAdZyVILMc/hVKTa3LsmtBCpOF
+ AA 9OKWVBv2jPIyTmpI4280biAR2oCKfvE49aUY31bM07S1ICi8ZyQOvNQ7EycffPUE9KsnAfB
+ ySTz7 VGUVXbB3MT070WaepSVr3ZWdWCbGJwOgqJjgJ3HHFWnDOw5xyOMUwqMtjHDdT60NpbkS
+ baRX24wQ jYJORTXIwoIwR0+vensjEAjPShozgFCpwDmnyxHBJdSAoG+YZb0GajcELjcAMdKsY
+ /dsCME4Ix2p N2YBjacdeOarm122G073IQqrGzFWwegJpjBSwYKcE8CrBi/dAksuB0NDLk9cY6
+ ij3G7CUnzXuVDu EgBztHTimEDByxBPSrRGXwrDH8xVXYJGAPU84pytYaW5HJneMsCOPzqFhuY
+ p1OcD6VaaLGOMAdjT HCBgWIBApSfREN32Kflny8HOcYB71DJEzHjqBz9atOPnznGDnk9KgYP5
+ Xy/Mfapemty48zXmUplI A2qzNjGRVd1eNRvUirrABn6g8d6pS7i+SSf5E04663G7oqspI3EHn
+ mqzFVG4DGD3qydwbaPSoJSu SvGT2xVySJnLVXKxJLKvzZ5JNV2A8vjax28DFWJOm3BJPTFV8f
+ vOeBSslcE7FfgpknbjtSAF5c4y CPpxVna2SdoIz3FRHaGyM5zx70k09A50lfYiKBo/pwDTQCG
+ yvOGxx2qQ7NuQrZDcinMP3jHGMnkV TSSVyXKzSRE2M7h1PXHemeWrAF+GI4FSbQhAOPahm4x0
+ JPGe1OVraAo2d7laSNsIy8YHQiq0iHLs WAK4wParxU8gsT61CwBWQMM/Lx70SlqiXOy5Uigy7
+ pGLbj0JPaqjhWkbI2kdavHeocYx04NUyjA8 nODyfU04xS2ZTs7vqUnwp3AgfhVaQFXzjapGST
+ V+URhlGeRnI9OKpH51XJ4C8+1XGTetxKz1OGRH XcDx9asx7gCQBuxk8UkSs7Efe+lWYlZRtI3
+ Ang+tSut3sZ36WEVC0W98KCQRjirKFgQMlhu5zTNp Jz2z8q5596sRxEfJyxJyOcmlLk5WaWtH
+ XYcrMJNvUHjA7VaUoyqSTlSAfc1GigTjIx681ZQASKcZ BGcinFWjYVknZDgpzvZhjdgECrfBA
+ ywztzjHeokUgbsrgcnI71IpJVeV3d+KUWzRt3LUWBGDxuPT /CrKhmlIJGR6DqKgQgkbcZA54q
+ ypDMrEgHGc1lZpkJ66Muop2oCQCO/Y1cQfKpPfpz2qpGcBQCNn bPfNaMIUJn5S49appvcce7L
+ MKhSGHIJAq3GoTGc/n0qpHxKp6gnIFX41LMGOMHnFJvuOTV7lvomF GQCSakKkrljnIzxUXcbT
+ nd2FS4G1T/CB07inTTRCg1G6Q4DCZVlYDOOKcf3sYIHJ75oiXfvDEKM8 U4K2xQqkjtVcqva+o
+ S27jUTG49e4pxUKvuRUyjLnbkAg5yKaDlScZGCOnNZ2uaJ63IGTjklh/CV7 077rDb1I5BpzKT
+ EqnjIwKbtZVGeDuwM85qVDogatuAWNwDnHHSnLgElTtBPODTiCGXJUIO+KcQuM cHb7VUlbR7C
+ +1zE6BSGGak27skZHGefamAEFdoABGcVJEH5LE4PAzUpMaSaumWYlZpc89KuqSpxg A9PpVeKN
+ hGfmJz6VOi5JwNpB6k048vQNd2WflKhs8qe3enqh3lhwe5ojAEahirDHHvQFY4YHK45q k7PQN
+ 3qSMVyCoJbvzQ5yFDEKew70oAIBIPB4pCSc5HIPU0nFNXQknsMD4O1OWxRhiAd4JHtSqB5i j2
+ 60joAcp1+vWqtdGrS5rCgk5JwfrSkArzgE0iqQ5I5AGKcuHwOgrFtqW4SsnoMUjzAVOTnpUyqG
+ JwCPX61EmFbI5yOanVVZCyMc5zyf0qpRd7kys3cVIyRyGGeetTlcgEYz1I9qSPlQ3HTgCpQPl4
+ wW 6fhQpS26GbkubXoL1YN1xx0p2wALngE8jv8AWm/MHx3qwqq0ZKZLdDROTXU21vZAflwFcNg
+ 88dKV hsdWxkbevvSbEaIspJYHOD3FO/e+YMKOnSiLu9EDbuG0GNmw27sD3poG3g5IzkAU8OPL
+ Kty9O2bc EDBxnk1V7XQa21GsmSrE4JPP0qDy/wB4SMjHUntU+zd1O0DjOaTYFLMx9MUrPl3Mp
+ S7MpucDLDq2 aaykJkcgmp9mVbODgcH2qEAhMA9+/an00NY2IAJGyCpOc5x2oZWBOVJ+Xip1yM
+ jIBPrTWLAgsCeM ZqVo7ESqW2RXwfLz91elRONuBjIq0yfNkg8DpTCAxUbWHGDRfUV0lcjHCjq
+ fQGmhOehJY1JJuWba vX0IzTdpI3Ecn+VDWgOOt9rkBB5JGDjBqLBRzgEsB1q2R8nIO4+1NbJb
+ aCuc8gindsV1LS5TzkkS McgVE8fI+Uk9B6GrLIxkZuFxycimFQYyRnccY5q7NLRjjZ7FNlLFt
+ q4GOaruDtUc5A4Aq0V4JAY5 zxUcnyxqXAXAqLaA+XoUXHOTgMB0qjJkSABgcDpjmtGQ5XcF5I
+ wapgNuycZHBJq1dxLc5LqUmH38 kBic1E5LNkgA1cypUsu0MQeTVNwBGScnPIqJxu1czTUpabl
+ BlIkKkcN055qHC72KqST2PNWySGJ6 ZHGRUJQEHBye+PSteW6syp9rkZHyg7wdvHHeowu5VB+X
+ 6npUhIJxyF7+xpm3EuQS6Zx71PK72bJj OVmhNuEO5gR600xksfm3dKlUAsVwCucj3oCBo22sR
+ g0c043JipRd0MCoVBAbKnp61CoBlJ6kt0qx 5alzgkY5+tMkTanyDJ3ZOPXFOD08hyV3vqViWD
+ MMADtUAAMbnDcNzV4gBFU/ewMmq7HAYdM0SfMy HorWKTJukZ+QO/1qlInPJBbqeKuurByM4H8
+ 6glRfMwW249aXO47Mr2fZmcY/nbcpJbuKptxPg4K4 xj3rSk3IWwQwHT2qhIrlCRg554HSrU0x
+ p6WbOTRcKu0jp2p4R3dQWGcZwBjFNRW81t3HHIFW/LO1 WBz+FVCavqzBSEKuBlyBsOOPepVGE
+ B3HjgGlBA2ZXI6kZ61MFOSI+M8gHtUxkn0L1vqCLgjcCy9R irMbbhnAwTUfV1Gff6VMgJjOSM
+ 54IpuF1sXZrccBhsn0xU6hWCtnPHWmqNxIALHpx3qVFAAUgg9R T0tpuVa7J9gRMqu7ng1cjBf
+ kId+MYqBcjb0VcdxVpEODIWAHQYqOawrW1ZcjRvKAPQfdJFXIlKx9 MYPaq0R/dBScqOM/yq6u
+ 4qQDjP3ie1SnKXkTF9WW4wjELnHynbmrUKu6Zx04OPWqgTJ3dyOFFWEY jOcnHUDvRGBV7rQux
+ qI5h1IFSnIQHI46+hqDkhSowD1zUysFbOM9unQ1ST1ZKvfclU7HyTtB7GpT kP8AKcHGc9qiLo
+ 8Y5B569Kk+UF9z5AXFZ6PdGkeZvTQAoZSQ3OOMVKoHA+YZ96jXbs4buB0p+AFT HT1zV811a5k
+ m0mMdVDKc5YcAf1pz7gABjOeR60qjdIWOGbOOBUgHO0927dai7RbnZLuQqp8ojg5P GR0qRVzk
+ cYByPWgrmRuRweacOH9QAMkUpO6Ls2rdRyZaQEHoOKurlmxjKkn8KrxjlvlYgAbWHcVO AQ27J
+ UEZxUyV3oQ4q+hOmAo6lhxyasCNkbkbh0z6VXADMVGMjn61bRSF+6SDxgdacUoaopavUkVf lD
+ EEAHGKtqPkLbhtHU4qFVygKhgw65OalQ7WOcFcYx6Vo5NqyCK0EB+Qb+mc8Gnkcnn60KMLuJVu
+ OlPADkAg5xSaT6FNkZjxFuORnktTGyd+CBg8Va2g5zzjpzUAYBSpPKjAx3rNNvVgm5q1iMbg+5
+ iM GnjY0nJxx60vCg7uv8vakCZBCg9eM9vam2xqKd7jesgUY+tSqm0gZKkN/OmquJV5+cE1MQS
+ MBWIB 5x3o5ul7FbpxFO8QFRjcDgEDFTAd8980i8QscHr370oYttIwfUY6VUddzLWyViQEuwcL
+ x0x71MIg sg3NhvQVAMEDBBqYN+93MpB6VN0ti32RIVO9FXovBNOZk8rvjOajDHL4VmPfnpQAo
+ CyNknqc0Nu+ hDTtqA2vIxHy9+e9SM7PGGQY9vaoWc7vkA2njHpT2J4weh4IpN6mjSVhCwNtzk
+ c8imhwW5yaeXV0 wfvHgimAgwbOMnnPpSdiXqtRjfMSNw44ppXBIII4AP1p3zFCSm0n1poAdwV
+ JIHv1pxTfUUpRURjf MgGCec5HpTOOOScHpUw+U9Rnp9aiI+bB45qvIhNNJiEjzCOCc81XIJGS
+ cKDxVlgmCCGwTkmmEHYT jgnkGpSHpazI2PAZjls5pnU4XIyM89qlbaoye3603PzBR6Von0BbX
+ IzGVjB3fMcd6a6Ig3KCSelS KzZG5lJ9AKYzPvIUA49qmfNcabvZkRVmPrxVQgnGTsA447fWrW
+ 87sdW9B6VC5UA9s9c1pG6bVimn flRXO7zc8kAdKjdQ/BwQRwakKbpAobnvzUZyCNvPH5Vb1ZF
+ 7PzKpxxuBHWqMwMaHaR16H0q6yZbh sZHWqskZLbWHzMRg1DXW44pLR9ShKm0jZjOOAPTvUGxV
+ XGc8cZPNaJUeaQB8o79aqsuCx2nJOR/9 al7Rtb6jnJJ2RU2ZTpxVTb5bHGSxIAJq6cknquByP
+ eoTBlGG4jnijp7zMltuVCoZs45BxkHFMZSk m3ORjmrAjw4Qk8rwfemiP5sSAkY4x60pWS3NFN
+ dWRk7uCNvYkDpQFMbcuNwbn3pxXClgSQH55qRg JFXbg85xRaL3FvsiJRuGQcDviggAHoV60rB
+ vNwcbc9vamlDjCnIIyfarml1JnFt3RWLF5gwUehpr Iw+7gZ7kVO+PLHzDcBjAqEqWAUsQuKUo
+ p67B73QpNgxnI46E1TYARu4+bB6E81fZFIAzuJ549qqS EFCF7cHjpVJLboS+YzphnG4jBPQCq
+ sh4I5znn/CtB9rjGMcdfWqZCGFjyR156k0TjJx2L9pFnGxq fMyh3DoWzmr6HcQQQFAwR3Bquh
+ CgkJtbPSrCcHOzAPIFOfvPbQlwdlbQWPltpzkDPPerQ2uQO55x moFyowV+dhj6CrMZBTGVz6i
+ m77obTTv1HoFHyqMnv3qUR9DuBA6rimffjXt1JPSpVQqc57c81DXN G97DSVrtEyMQqEkKu3nj
+ vUyguQeD6moE6fPkZXFW40AT7/A7DvUyShHmsTzpeoLndllZgep9Kuxf fycBTxj1qJcCIY6Ed
+ KtoFKHbhmBHAosmjWUls0WQAYuAQc4q8v8Aqxz04J/CqcI2yNuGVbnirMfz L8wwrGny+RC3Vn
+ oWoyAkec/1FWdzcEDI6Zx0quNm8YVsgY61ZjdmBjYqFz6cmla3vA9XsWBtaPqd 3bmgEp83U1G
+ pAZeQewFKchGJKjFXzW6jhF7E4O0DcAR6YpxfdI+evrUIztyD9OetPjIYnOBjrxWc 5ajhuy4j
+ qVPTcKC/Izg8c4qsjEYHr1yKnj3CbgruPXNTFxTuE421uSGTbGxOcZwR0pQ7bgzfIoGe e9MRC
+ 7fMAFJ5Bp7YRxhhwMYNXzK7FzReg7ftckc5OPrTxnhR/EC2KRM8Zwdwz0xUyBAoAB3j37VL 5V
+ uipTfzHxgkK2GxjHWrKLulYMCcjiohyAEBwD1qZQwAOwnI5GeRSbbegJ29SzHtwOhHtU652lhk
+ A8CoUVi+1Qx4GKsxAlCjL3wKm7QNO5KNyE4yccE1MBhF4OGODxzSLtzhfQk89TU6nCrlcccE0+
+ Zl uOl7CkDYAuOOB60zHzgsCMDGalx3GOCOaNoJ6d+lUpaWIWzGlQ2WRgcDr6VAIv3zn7pz909
+ asNhV AAw3tSEjILAsfSqvJBzS2uQgLyGbBznrSgENkfTmnhwWOFGW9qMZbcSOOwqeWW7NNle4
+ nzHqvfqK lBwSCc5NICVQkkFhxSkAkEDnpUu7Rj1JFEag4ySQD1oJONvyqTzigY243DGfTrSg5
+ 5OA3ao6XLjo rpEgwgCPtIJzkdqewPOc7ycDPtURIJyMA+lAJ2A9fTmjVoTpteQ9tyq2Dg/zpU
+ IOM8nv6Co2bCqN wIHWmFsSkLycVbV1YqK5r3JgMHJ4BPp1pWkUbcMMDrxUCsW65GKU/KNhIPv
+ ik4rqDi21Zkm8M5bg c0ib/M3ZGMYxTVwEw2cdqeWOMEZHqO1K6UtBuTfxEZb5iSdvfB70hJ3c
+ MOOOO9KQpTK85PNLwW2j gHkE0732I0TuxXOFUNgjPaowuG9QelBY7SFBLHoKZLuZRs3Z6U1Cx
+ UJ3W+4b2LfMDjPHsKc2S3yq eOtKuSfmUjaMGmM4XLhjz0FK6bIabdyPq3HU8nNBC4BwST6dqX
+ n5W9RSbcgNjJIwMHGKvbUf2tQY plQDxVc56qeTxxU7ECPbtwRxn1qAqWIw23mnF9w212GFSkY
+ cDJ796iZg7AYI7kVLkrGCxJOcYNVn G2HGCSD97NKMXJtsnrfqRsjLKCpGMdahYAZ67/6VZcFz
+ klRxjmomAyOo9CD1q4yktFuaOStoVZFz GTg9MjHpVcjMmCSvy4BNWWO5sjdjoQDVeQMMEqxI4
+ Jq0/PYmWjViswKqRkMw64qiwyuOcZ4z3q8c FsqGI/izUEqKHXnk9ABWMnrsTblVmUjtV+Qy54
+ PNV2V/KbBHtVtwTKOhUHHAqux2uu4jGeKuyd+4 4SkVfmyoztz1zUhB8wc4UDNPZVKlgcYH5VA
+ MqMhs/XuKHZ2Ha6GFS0yqxAXGSR3NIwJlBQYOMmpZ FJxnpmkUjadoOQehqua2qKsnoiMKqRtu
+ 3EgcEGmFxJz82QMHFTgATkHJ7gE9qCdqA4C4PTFDehjz cujK2Ar9Dnjj1qOUhRgnqRyKsSAZI
+ zyevtUDpiQK/OSM1EprqDty6FcqfNyD7/SqEjYkPOT0IFXZ AVZtp5J69uKrv5bFixPJ5x61p0
+ 7jgktbGcyAuD0x0xVZowzlOq+1XziOTP3htwRjoaqSkHkcYGM1 HLK4ubojkPvFSdoBOSas7AE
+ G0hhnINVQyliuGzjqDxU68lAXA44AFaJPe47qysTocN84wvYmpgBt +XrVc5zgncegx61OmTwS
+ OmD9aSuVyJa3LKKWVWI9gRUirhMkr1PB54qJcLgAE+ozTxh2ySAc0byu LVq0noSInO7D8HFXc
+ qrKuDuyfxqvgg4JHJ4xUi/NIm48jk570NXd29hP3tyyhOzcAF6EAjpV2I5l VhwCpJB61VjIyM
+ 9CfzFXEJyWGBk4A9KzS966QnGz9S4p3Jwd3Ye1PQlY2DfezkVXjOG5DAjjd2q2 CDyPmboT2NU
+ 48m5TdtCZJMKWB3ZGPpUgXMI2MRjjr1NV4353KoAqRcsSobGTnPvUweumg7u9yzuV IVJznNPV
+ wR94bD0JFVSdu0EhlBwaeJPkZgv0HYVnpzXKUmuhYXJlYKSBjIzUqs3mKmVAI5qgJDuL Drjmn
+ 7jhXJwCPTkVcXd2Bxv1NNJCi4Cgj37UAsxP971FUxk42tz2FWQcsDnaaizT0Iah0ZaB3oWd WJ
+ xnIPFSId4XYhZsc+xzVRSuOjYX361ZR/3gI4B9KqT8tRtcsdUWN37wg/fLZAP8qnwS2SSv4VWU
+ rHPl+hGAT0qwHYkuMYGOKrXroVzNapElvn5vfnPrVwN0KnnuPSq6BQ4Byccn2q0CPlLcA5/Cp5
+ rv QiW5OobOQTz29KtRYHBP3ujZqCLDtlQ2QOM1MgOz5x0PGKe+5SldpFsBTtwCOevapgmcr19
+ ahjIZ eFbC9h3qbpJ0OSPzqW+g1fqxBncV56/ypwIJUknOaQGTaQVBJ6cUuMR5wOOM4qXbqH2h
+ 8h/djODg 4x3pmSX6hR6kcUobcSCp68e9DcnJ4wO9XDXQE9NXqMCFSem007arjKsowOlCnEXBy
+ e3eg7SQccjj IqZKT6h7smOHC4PBPOaRi20fIVGOtPx8qgkHFBVi2T09qmLSdw5etyJC4TcQcZ
+ qYAFjgZPUUDqVw fpSjIO3+8OvrRJ3Fa6GthGVQNxx1FNb50IB255Apejlt2frScoMDBOemKd0
+ hpW1AYCjcDzySaXHz LkjIprPztJwAOKRQTk8nPr3pNuw3orpbgN3mfMOvHBoBfcegUHAzRk7t
+ vUA8HFRKWyQScn1quYnV u5M0jgHBBPXpTEOF+Y9cjFMwBwTlh70pY5zlcAdMUou1mSmuxIvTq
+ SM9u1OWQZIYYOOCahVwEzkf 7VIXDBmAyQcGm3HZlpq6uWc5XaSBgdqrlmLDaQe5UCm4IkHzA8
+ c0xxhmBb5s8e1NJb7mkZa+RYZm DYztOO9J95sFlA+lRo+Y8MAcjJ9qRc7gwzjHc9KOVcphysn
+ C4TBIJ6cVH0bdznpjNGcNjB/wppKs oKsAwoW4SWgz5wzEnoKbzuwSAByD9aduycAn+lNwfIO5
+ TnGAattpotJ7MhcsPQ9jx0ph3NCM7Tg4 HvTmHOPmI9B2NM+QykANjHek3Ym12J8ipuY5YdR61
+ XOCQcgDv7VY6jrtB65qu+ORxzyKT9NQd9iB yGZhk5A4A9ar7vMxnltozU7DJ3Y4Ze3aoSAIsj
+ vxmqjazuVFt6FR2YSbVBIIzxUBVjggexBFWzgI eRnHBqrsJhUcjPOfWiTe6DmuyqyEKxbJOeB
+ 3qFsHB2MMHknuasOSM5Ukg4TmoWGGIySOhyetOXMt yNempWPO5cYBppXC7QOVP5VI3DAAdev4
+ U1wGHOVLfeOajV69DRu6sR43HGfb6UwoVcZBwB1FThcM No3etRF3MhJVuTz9K0jonZkp6asGJ
+ 6gD61Ccs5DKT6e1WAGw2VyQeKaWGSp6gVN0tLApLsVnBZkI IL4IpoX5NxYFyp21MQS3GMdc1A
+ WLnHAYfnSvpfYznC+xA5xGobBJGRxVR24G5So7/Wr5Q7SrMAcd COao/el28HgHPrTg9BK1io4
+ VmByCx5J7VUkVckHOPbvV9ioUjgLyBntVNvv4PHcVd3fmKsrHCs4A JzgZ/WrMeUAYENniqabV
+ y+dw6gVZQltwyFOKpT1aREoO1uxZGS2cFsnp6VaHXd2FV0w0KnOCDzip kxkgnPpUxipFddSwv
+ LbtrKNufxqwpRh3BPP1NV494Qcg/hUi4Dlic44wDTabauO/ctIMoRjkdR70 DktuBJ2kU2Mkjd
+ jaAck+9ShxtJ2nGaiKd9iabd9CaD5XxlSFOBmrkZUzAnIxyeaqJgK2whcc5Izi rS+WyI/8Q4I
+ FTK6Zc1zKz3L4KKp3MuC1OQpuPPHb1NVlEXKHnHfPalVUMm4E+mKrlb6kq0VroW1K hyACMjgV
+ JvZSMkZA9Kq4c5JcZB5x3qfefLBGCp7YqOZCV97DkcEAPkjrkU7IIJw3QnGagLjACgk9 8elER
+ z8vTHB5qancqK5ndE8TAncrY453U9STMy9eeKqqymTAbkA4GamRmEgUZ6c+1S79zVxu9S+j EA
+ MNp4yTUyliqsDwRVRGJj3YwmfwqxG6s+3BCf1pRm9yHpsifB3evParablTIxgjBz2qoGw+CDx3
+ NTxk4OMkHsa0c+rFzXLgctDtYAkdxU6nJJ45HPtVRFQg5OCe2atg7WPBA71CcFsOL5XoWULMdp
+ Gc 9cHrVtSTyV9s+lUo5GVCzHHt6VajYSBSSVHrn3rSLb2QSve7LaEqCBkHpmrqBnChuD1qivM
+ pI5bv zVlM7QxbDZwTUyfUvR77l0ElxnoBjjvSsZQpyQAvbFQopwDv5I4HpVgIOm73+tVFRQnK
+ 2lwDOQu4 HI6gdxT+DyuWA5poOMNuDMBgAVIMNICflUjpWc22tQe1xSR8u0Y96XaQVx827nFRq
+ ylM5LY6c9KA SGJXp/DSSsKy3Q8d/lxzxgUhQEv/AHjjimCbkECpl4O853N29KPeiXyu2ogwSM
+ UpyJNrEEegphwT nawPQDNPUuGDHAOeuOtCVtSLu+40BSm3BGCSDSZO7g5OOfanNnftA3c9R0o
+ OMt8vQYzVPcL3XcEC 9W4PrjioyzHKtnAPBFPCj7rHr71ExypAPvRC8gTtLVXHMGCbmXcN/FRj
+ 7/3sAngUMSyKCTtHPB61 GrfvcnIB9fWlumXyt6Cl8ZCgkg4HvQCGIDDoeTmmM2COmF60ze2d4
+ 6UPVE8rS1JSwK8Yz3JoJ2jI PamEfPnIwByaVFCpkHdgnrTjZLUNtRCyliBkg9hT1wApXHI5qM
+ AGZmORTcAcjI9OetVdNBKVlZD2 bDdB1496UMG6kEDvQHyvPODzx2pnmDAIHOKSetlqTZtbD94
+ 4BXHbPrTgxZsggKBgjFV925+AQONw PWpMp3DdOvrVyaTVg9nJLUkZcg5fIJ4+lIGb5sLxnGaY
+ 7sQMD5T3xSKxZckMOefShXaHeyuLnEpU jk8k0g3k/eGMU0yKxwMjPIJqHe3lY/DP8qOVvoFlL
+ R6ExJD5TaOMHioWIDLgjOMU4EBipbf6ketR yKu4NtIbriohDUGmpWSGEkqVZgSOtQORuAU4PY
+ VK23ZuIYHBzUJx5anBGRyaq73CU2RFzu29s8kd 6rktjpgZqd3BO0YA3Yz61DKWCAYAI5px7Eq
+ pJ+RHKoVMnG4DkVUkOIQR261MzfvWOC2Tk88VH+7U 53cEcjvTk+hUYpLUiYq0obepxwBVZ2Bc
+ hlzj0qaTBwwBHQAmoHUMc9+MAGs7q7uJWIcrszg49M00 kH72eRz7VMyAOFHT161C3DgkgjIAq
+ opJ3KcvdtEQBOmcHd601lRZGwe2BmnkHeTzuBz9aQkSOR1G 0+1PmSepOuxGM5x3HvUL5DtgDI
+ +YHHWrGQkJbGD71Fu3tvAbrg47U4xdr20HKKWr0IZMiMltxBPH PIpgUKpZc8HP1FPk3FVCklS
+ agAMjKORj360ON1uRb3dxj/MCWIAJ4XviqzIFlPUMOOv6VYZguAyM AD96q5ZndiQpHUcc+9Uk
+ 76E3sUpN2/bwRnioZVxl3bPGAF71ZfiWQH5cEYPYmqsvzYK9O496HC5b fVs84R2AYbcYOFwOt
+ XI2yyMM4Ixu9aqLtL71O4dMmrkbjcowvfpTSM38VlqWo/uBSWUE8GrUYVYz 13AcelVY8BWx85
+ x8oqePJwSu0AHH0qlrpcat1LYHyqADgHjmphjaQeR3xwahRlEOcHOelPicknO3 g4pxTWpa12R
+ cGQcDC80qsFnJwORxTIySoB6E8kdqmVdvyfKMEjLd6ykv5jO6UtSRPM2nbgKfUVMv yocHAzjJ
+ FQJ8seSx9zUyNubJ5XPSqjF9Byk2ix0ZmXG3PrVpWROSBheCfU1XQuWZSVUDse9SDIJY gHIII
+ Hb0rKXM3YrSyvuKwypKZVsjK+tSOSFyGyR1wOlKq/KRnnHBoEacqWJPpnpU86e41JOOpHuV UG
+ 0Zz0yakRgRjODnJHemupOdowMcH3pEG6QgZB6YNNxTSuyNL7kvfK4BHUkVYiO5m3lcD8KhWPLg
+ kZ9hVlQHKlcHPUd6hJPRG0dOpImAdgJXsQasxjawAGMnkGq6lVnJxnnH0q3zkjPfGarpsRK6aT
+ ZN jhufoT0p6SNtwe3GcVCqknaCCAM4qYcKwzuz0AoXK99y5rSxbUq8jZbGDkVYT98nU7emapR
+ D5iww uF6N3q3CwEG0NnDU5abGMrpouZIk6ggdqtRsocgkbQetU+d+TgjHFW1I2ZI4OM02rW7F
+ u76FtHUZ AxgHHH86tR4xk8gjNUozk/KeM81YXk/Id3baPSlyrqLmaL4KhAnDDqMVIpz8wByD6
+ 1TUsspzjHGB irHKwBjkY5PpSa6I1cLJXe5L0bOQAO38qcSdyjrtPJxUR8tlC5OCMHB704AZXL
+ EgHnnvT6ahovUc NoZiCMN93NOVy0Q2j5sc0HhBgZOO3egHK4zjNRZ9NQUtbsXy1IOVx2z2Bpx
+ YiM45IOKZvJbB4GOD mlj5AJbJx0Bq9be8TNO97kjcw5CkECpVKeWrMGJpMAZIIwRSbcBiTkbe
+ 3SpWqNJO6sKANrMemaib 3O0HvmnAtnnOBxStngEgVPK76C5bakIyvzDP3T15pBhgMEBunNLkg
+ YKlU7saGIVc8Z6gVdnuydLX IwSHbcQRTWcA4bkAce9O6zcDJpSMYbIwexHUVOtx8sXKxGCDtI
+ AzjPPvTGxtH97HSrG1WTnqTyem KgUDcSwJYHr2qU3dg1Faobt/dkK2455poyJFye3SnbSJSRy
+ SM8dKMDhwwIwQRWkX3JlorDRsaQBm IGOcGmlSAQvOemKGCbG2LknjOaQBgpGeBjH+FCknqNKy
+ 0Yi/LhS3Q8GpG7kYAHQ0w5CnC5oPmEjI CjuSOKaSbuym2noxQABnrntUaSFUwcjnBzSkgnCk4
+ xnPpSAkldwHQ9qclbSxMpRa11JztI+U9+/r TWwS3BHPTPeoySytIBkdsUwlSmWbIx196m7urF
+ RejfQA2JFO0/WnbhkgLgYz7ZpOkQbjpjJ70xiG CjPA5JrR8vVC01Y8bcBdrHb6Hk1GTtbC88c
+ k9qeCQTg9fX1qNmEfmF1znHA71mnYzbTZE+OVyTnt moWYlAQOg4FSN9/c+cnOMdqhyuOWyQO1
+ XZNjjdvQY3IC7MN14qFiRIBgZxzUhwW3Bvm9PWoiSykA c5wKlvUbkra7EGQf4TkDFQE4dhlcH
+ n8KmLDldrKCeCTVZnLYLHg8VTTew4yitRsg+UbTu9PYmo3A DKBj1OKexAcEHPXP0pGZnjGAAo
+ GCSOtDctmRyIiJXlSDuzzjtSPFkDq2R1HSl5ySwJHTI/lQ27dw VAz0PahSl00HGTjezIG/1QQ
+ NkmmbFVjyW7A1Lg7MdRnP0qPYCHCkqpbrnpTcrIUX2InG6Tg4cdc9 KEyqjcCpPB96kbAcgr0H
+ B9aYR+7BwxGM4pSldFtXepAWKv1xlc4qAtnayDDemanckS/dzjp9KiZQ qgMTg+ntVJkVHFWTd
+ iuQxQn5iOpx2qB8/MAwBY9D1/CpnPyl92Ez09aquTztb/cNJe90DlS1uR5U qM5J7+1Z5YbiGB
+ 9jVuRm5jUZ+YZPr61TZAFDbScngD0pyavYSTtc87ixsQKeec1bGFkAABHXFVVE eGVD1PAHarE
+ IXzVdcjj5iTVuV7a3Q4zsrXL0ZGwFR0PHtVhRluCc5yQD2qqpyfkOBnknvVuJwpAB DNmjmdrk
+ pxT0JwvzblBOexqwuSuCAORg1GxYxZACkHrUiBccg8UuZ213CKb1J4wc7WPy+pHWpesi gnAx3
+ 7mqyHMyhmyAMfWrAwYgwHI9KLNOzHLe5MpZiFA2/L0I6VMpfy92Ax74qFEK3GSTyKl2soAG QW
+ Hc0khtxvYsfwDOemCfepg42luhJxk1AjbSdxUknke9TKA2A3A5ODWeq0Y07bCgsfunJ9RUqFto
+ wQSSckimxhAxxuAx1oVgZAT8v8PNOydgt0SFCfMRlyoGSe1SAbWUg7gOtOU5+RjjJ/SpG3kHao
+ Ax 6VEm7ibb0sEe5pl9WP6VPG6rMAOQM9KbhgDgcjgZFS42MQMZzSdt0OmlLRk6KNnbGD1qfLG
+ PCkHA HBqBAzIvb1qwF4AXk45xTstFcTjrrqSKuCOeCcilCsJFJI46cdRSD7ucccZqQuCOCMk9
+ xT1T0Qcz Ww/7pJDAsTVoEBME8DgEVAGAO9zyFySasRhfL6gA9aTk1rYuMu5YTaxXBx3ye9Wt2
+ EUDOD1NVkOY 0Cge9WAm9VVux4NTz2SbBSu0pFuNj5pXv296mVtrj5s8cYqpGnIOcsTwAasoGO
+ 1gvfJ47VpHlLdk X42/ixn6mp9w5YHPYiqiq52v0XH51KuTwpyAMfWhW2M5RVxYx84fIwRwDVl
+ eFAAzk5qNUBReMEDF OG7ewOQc5z2oqT1Vyp3k7IcMSDAzx+pp5RgrOoBbHI9cVEnXjIXNWTnp
+ 0PbNEk4kt9GRkBsDjjqO 9TZ4Hy9+/eoVADswyTmrC/NtJBA7kdqzfmXJJWHgfJ0Aye/anAMmV
+ fBBHQd6a2FOSS3vS9cAKWAH WoSuHKnsxrcIOABngnvTAGLDPzZPb0qXAaPAwwz1xSqsm8sF2g
+ DtWnoKM+UY4zDuOcg4AqJgzDJA Crz05qwpyQGIHHpUZ3BieAOmCKIprcIbtkeDuBxgdeRTScg
+ 579KmGSF/rTHUjAGCR6UpJXHFvqQy sVxgY4wajA3YBOPpUx5cAjnsKT58j5Rj6U0tBOViA/cA
+ 2sB2+lLtJ3K65QnjHepmXbGV5bPSohu2 YY5XOFIq17y2Ji7bETDkAYVccEimlWEYCjJBqT5WB
+ G7GTwaQ/e8vuO9Zv3XsU5PsNziIE5zmo9+S 3JxnkUuG3AE5UnimswHyMMA9CKtJkpxTIyq/L8
+ 56dzTkILfN8x6AU18bhyO/5Uwbig5ByOaJ2a3N LpbEzPhBheD61EwJwSvy5xigMxIUMoXqCaY
+ pVW+ZiRjjB604NWtYjl0JQ5CFcj5egIpuAYznAx2P Wm5J3E8YOM0EER5B+b0qWS7XGFxtLc4J
+ xj0puQckndmlCtyTnHVhTAp2sBjHqa0TQ1roGVMuVJK+ lQyJG+4rzxzg0YAPI4J4NNYDeckjn
+ J96lysEbt77ERDBVJ2kN7VAygMfmIbPNSSBmjyT3zxUZXA4 OeRyfemtdWJu6K8pG3uSeeDUWc
+ lQpUkL1x0qRvlySRuP3jUBKHAGAGPrUt23GrdCJmcPuK9BgDFO WQNER8qjOeaR/vZznj0/Wq+
+ D5oKk4xkVbUXZFOKepIW+YgnHtTCW3HjoKUDdzjnuKQLiQYbBXjBo uuUTmrWGniMdeTk+9MJ4
+ OMnNSjHkr8pHXgnkVEWO7GKaV9h0+wmNqKW6kdTTONjYJAx1PQU4McKv 3hg5FI20dTgHsam7u
+ ZXalZlUsSm3BbnOc9qZIpB2McjBqVyQ3TAH3ahYhw5LY+tDjLmukO22hA4A jC5DA8darZzCVU
+ g4qZ/kUMMEHggVBjO48DPFVLSxMdXcpyttQ8nPt+tVXBK5ByoH61ckC7diqSPX PaqTDGRk4z0
+ 9qLrSxS+E873DDLjkHr61ZBAZTn0yBVNPMSLLkMw68VbQLnewbLeh9KttWeu5MlFL YtxZMf8A
+ cye9XE27lLEDA7VWAX95huBg/jVhcmXPDLu6+lZX1Ha6si+CGfIzsPr61KhYgchWxjFV k++Cc
+ jP6VYUETbu2Oc09X1M9k0TbNhwPvZ/Onpy5Cgpz0NMT95yxAAGOalj25JXcwzyaqTdrGu6e pY
+ yWPXHTBqQAsm7dkZOKQGP+LIAqQhfLYLu3LgVClZp2JtrqIiuYyM5I6mpSVZGL5OBxikBwxUL1
+ pGJUggZBGD35pNu+ptT31LDBBIGbcvy9/wCVOxmJeCfm5FNAKSGRnVs44IqcDLqVOBu5FHM1pc
+ mT UXYWJW8ggkZB+WpxH8oIz7/X0pvUlcEYPapVA3kFuOg96h3uKKbTuKu8ybenPBqyPmZecHG
+ fwFRK p8sgDBLcH1qwEIwTwONpppsG0x6lQOCORkj0qVeUBGTnsOtIEyoUDgdDUu0cc7ferc4/
+ ChSn1uOU MMhuB6HqKcqgHBJPNCr8ud2cjJPvUrR4iXJ5HXFTKdnbuNSkxVCiFSTu9vSpxgfNn
+ BBweKhjDNE4 4GSCPcVJgBSCQ+DzSTsF3HZlhJdhK5VgQSMCpkyQobrgc+9VyoI3cg8AAVYi++
+ RgMq9xSvFJ2RSU Wty4vEn38EDr71bUs6hQSQOpqmoBw4PzA4YZq0GGFVSPT/8AXTitBa6WLPI
+ UnPBqwDsKFSpK9cVX TOFB5wKlVl8s4GT0U+9PltoNp2SLOTuPoTjpRywPPfrUKu2BgEjOalU/
+ Jg/LzjFTblKcZfMmxggE fMOTin7v3hAy2T+VIuGVW3bh6ipo1IBLAZxwacprRii0tGhgzgjHO
+ 4U8u+WG3cM/dWl24+Y04ZKc MBkVDs9UFpIcqqFBLKWPYnpinDIbbuyG9KaFyvBUHGfpT0IT5g
+ u4nsaNWOT6CEqVA6Y7CmsHLDaa njjDOcgHnkCmvHj5lO0981UJpj92/L37jX5iwAcetMVJATu
+ +nSnlmA5KrkdMdabglQRyP4setONl qOV4dhpyOMd/yphA5K8MDx78VO5UJnPzMckZqA8rnBAP
+ SktRNcqIlU4GcjnrT2+UsecZ4NLtHmIr HaOetOCsVyQdopylsTe0tSPBJAOGGDUDcr2AHarB7
+ YOc9hUfJcgEKO5I71PNbcSTTuQNjaNqkf7W KjOBOSzAqR+dSSKVLbRkgjimsOWPdTjGKZUOVO
+ 7GMdx+XqOeahLDOD97vmpNoL8sOnzVD97cMDOe TVWSiHKlsN35KnaRtGMnpTc8sScZ6cVL90f
+ ezngccVFsHkkNnJPrUc0dWyelkIfkUMRz0znpUZ3Z P8QJ7dqe2ApVidh6+1R4USPubGcc1am7
+ rqSko6XBV+TLHgHtQeMlckgfMc0mVDE8njof0pSNifK2 T/EAaTlc0vZaiFiQo6Meuab8vz8+m
+ OafuUkZ4brVZztAJHQYFCvs0TFJ6j5JFEeGXnOD61DLJhQM Yz09hTmysW7GTjnio2BJBOMnnG
+ OlOLsEbbkLMyhj2A/WomJIU4PQcVLksXyNvOCCKrmTpubsCcUW bb6icbIhD/vgS2QTnHrUTcS
+ c4Ck8EipJGDSbeAP61AVyqqwPc7aqST0M5JrXYhYsVJYgYPXFKxOz jGGFMJCyDP3M8j09ajZg
+ pDLz754FG0i0uVXHKF3jduyDliDTiwMvUDPTNR5bJLDHYUAZTLYyD0FW pNi57eg4l/MHQL61E
+ 5bIAHyjnipC22M54FBcCQggEgUlUaaVgTTVkMVnk5VMHPTHNMfJX7w3HrxT s/6QCFJ9SDS9mA
+ AIzxmolP3hSdtCsC7Lt28k5we1QSMPL5UlurAVJJncSAxHIAHeoDgDoTxVx01K dmiLg78gjJy
+ BVNiRJlgd3UDPSrbLsUEA8jpmqLuASwDHI70Xv0IptJtdCCRz5uw4UpyWHQ1TkGSC HBLA5C+l
+ XsgIzMADjvWax3YUIw56miM0nsXDV3jsefgsdyqA2CMmrYIJBQ9OoPrVE8bemDx8tWou GYEHn
+ APtVSXKm0ZO+7Lkb87cjsT71fUdSCAx5wRVJFXaB/FjrVuHzDArEZOMKfWlOF2pJmr6P8y/ CM
+ k5G4ntUw7gg5OKrRsGbK/fU5zVsllJOMZpRSvqZuVxdoB2qee2R1q1Fnapwc96gQ9CQeCfxqVS
+ ehO3PSrUtGW4uSLIO1/mUN6CpF3NMyg7ecmokGIyZDnI5PvU+R5KgZzt59aye+w3Lp3H4AgHyN
+ vI 496OhRcEHoc+tIUyuDkDqM9qfGr/ADcBiT37U6clcUb8trljaXkYsjEHgVJGCMEABc8imAk
+ 8dV9q mCfLwGU+9EovUHK2iJV2gSEg5zzzUyqiDPUkcVCnO5SQRjJqUZDoPLPy8DIrNRaZpfrc
+ kxu43fLn irURJTDjPcCmooDcYbdzUyFVHGS2cE0m76WJVncevy52thsd6kT5W5xkjqaAuHI6g
+ 9falLELgYLd qqEuZWQpaj8YQ/KRxgUuFCjG4kfeye9NXeXxnJ6gEdalDqwwQWOMDFDlYfNdWF
+ QcKSpBPGc0+NC6 ttZeeo9KaC2wK3GetKMK+A4I9Kh33Qoxdrkq7gy57DOfWrKoFAIJBPqaqDI
+ GMHaauhFIyT8w6iql JrdFyjH0J02qSeemTzVuL7o+6Sc4/wAKpKMsArAnuvtVwMoIO3kdqfK7
+ DUL6J2LYKiNQCUYdQak4 bIVl3dsVCq7hllKn1PvU8S+UVbaWAGMUaJeZLvF6EyAiLYzADvx3q
+ RQ5Vdy7sDkAVBltxYLu5B9a urnduHLEdBScrPUavFkijg49cY9KeciPackn3pi7mXGw89frUy
+ 5ITIPHU0l7u4pNSWgg2hQvJ5pP 48EEc5Ap5X5/lIOeRUjACIc8HuaTbsKLaV7jVXqcc5wBU2H
+ BALIGbtiq4DBCSAQORzT18wgMWXP0 5FO19bml2KU2S7MnB64NOHQgAbS2eaaH/eZyC+O/ak3M
+ 2FwPU4FJvUqbskrgAPNy2ScVLujSMdTn oO31qJQyynOCMc5605l6HJOeAPaok13M5STdhPmbI
+ KgDPWmncI8feHsKk3nagTHTkmo3yF3D8QKa auJybeoE7iWyDTSV8gZbOcZA9akDAgnYQ2PmNR
+ Md6gEr1p7ly9NBhUFyfmGcVWJAdN3G5Sx9varT kAnad1QO6nICgYwCTVQeu1xOV9LXRA25lDt
+ yQe1RcKflbLHn1qct+76dT2qry78HC4xnFO7e+xSl aOpICokwcEc9BUPl7W4PGM5zS5AJCtjt
+ zSFl3lmPTr2FKO9kzNq5Ex2oFTO71J4NRlgX5baQ2SKl b7vp+FRyDcAcgYHJIpy8h82uhG+Sv
+ Xv+dIMrIQenFOJPlhQM49RmmseORgk5B9KTTaDV3EO7Chuu MZHalAKklRnHQ0jOhHIYkYyaaA
+ AM5LZGeDxT3uNu0dR2CS2c5x26VAyjJGCRwMelTDduy+BG1RBR v45GeTnpSunsRGdncQ4VQoB
+ LBehNQMWdQVG3Pb0qRwDyxII6EVC5XYBuG4ngU0+ZBZXuyJziY9el VpCpi3A855qeRT5hySST
+ 1qCQgDYwwM9cVUddAT8yFyoj3kc54yeoqq0iAfezj3qZshcHHI4qux3D sW6dKpIm1yLPRTkA5
+ xmlfDRDhT601gUTaTvXoAOpNRD/AFIYqwGfm56VN0nqU+e+iHdQGyMU9Xyh ABHPU1CcrMQeRn
+ IPajcxK7gADyMd6q/NuOUko7DiwAAI5I70mSI0zxnjB60qYxjGD15pA+ZMMVPH 5UOavawlG+o
+ 8sBCdqncO/vTH3gglgvHUU1nGNwJb6Um/12kHqTWV12FJtLRaCZOAVB5PXqKrswGA e/A4qRid
+ pAHocCq8m4nkAHvmtORNh53IflDFdxyF71Ucose7nkcVO/31BBBJqCVVI2Kc+nvRdpak ucb7l
+ Vy5baBkd8VRlYHhvlIORirjMSwAypA6mqLsvmY43E8H0pWQk9Tz8KpjJDFgO3pVqHAtx8pL E5
+ BzUChIpujONw4FW12F8E4I4rZWWjRKqPlS3LaffXH3ic5q1HzGrfN16dhVKHkxkZy3I5rSjc4J
+ I4J5A7VHLJbMq6jqSqQANox6+1XomDIN3JAPPUVUTmRQoAJHQirCLhlGCAOtFr+opJPUlBUqrD
+ Ab r7VYXPDsASTnj2qCM/OwwAM8ZGashQ5BcgqAfu+tN8yfkaaLUdGHkUZxnGasqo2jLbiQcKO
+ tRKmR hRgYJ68inqSsiY+7twKE+bWxklfW5YjJ+VcEt/dNTKMIG3Luzk8dPaoVBV+Dn196lUDa
+ 3t/D9KV0 nojWy2JMDbyQzHONtTooG4knHHNQJ83JUDPIOKtpvEI3Lyeg/pR0stxxajoJySxwT
+ zjIq1FxKuQe mMGoUwQepOMfSnqCZPXuSKmWq1Y3YljO6VVIKnGB9KtooE45xkY57VXUkyLkj7
+ tWgcle+BzUXtsT JvZDt+JTnJGe1TrzGVAJXvnqKrZ6dAOpzT45SBuyDn06VVna6C2mhOA20FT
+ vA9OopSxVidnBHGO9 RlnKrg4PAp+T0JyAODjrSk4p6iV7oAflUkdD3NTHbkA9AB0FRBXJUYYn
+ HFSP9xF3Ddijmje472ej JshpAoI2jgipRu3KVyQDzVSMb1O1stnJJqzGxKk5xxwaUk7jcGti0
+ rLn7uCOM1ZyXG48jPFUlYlF Lde4Aq2kgEcfqRyKIxafcIt3ukWs7Y8iTI3DrVtQfmzzz1qih4
+ AwMdR71dDkNwcelKo5dR2luiwM LwBxjgk1OnyKQoJPc5qIEFACBjP409VyMgj3oik9bDb93sX
+ EwARzxwOaUfK5xkIetRxklxjr+hqT d8w6An1ovyuxLbt5D9+F6cdD9KVgNw3nGRkc05mwoPyk
+ 49Ka+PL+YZ44OKEwjN20QDiU+XkjPc9a d8y5bB3E0ijcvAKnGeOhoEjPnkL+FQm3ctvqG1DI2
+ 44yOMUb1BAGSSOQO1DGMg8nIOevWmgHYflx 82QfarUrIIO6Ji29BjAwcc+lDEeQCOe4qBR+/w
+ C4X1p+cuMlR7VD3FyNvUXdhSPvetB3CPkYY8ke lIWZlzlR+HWmM2VGQSc4x7U1eW5Suh7PkAA
+ 84wfeo88KBgtio2YuxXBPTBpWxwOVApyVlqLlVtRj MzEHG1c1G2OR8pB/SnM/zYbkED8KjYnb
+ kcjuaq7W+hU1MhLgkgnA7ZFRMyD937Zp8o7gYOM4JppC qSxxnGckcVNnYhxttqRA5k6gEdc0O
+ +EVtpIPX2prHKliQD14FIZCUyFHsDVKydxxetxh6sXIOeQB 2pn8POQemeooOGl3g4bFMBONzZ
+ HfHpmktdWKT5XYT5WRc5B9QakPEHPXvTHIkIKnnoP60gLNuwVJ zTTuKb2YnO3HGO9KcKByOBk
+ H1qPc3CjB5546UbS7ZzznI+lVJdxTaaJWc/LyvTn2qsTjncCM9hUj KB909ByM1GwG5iAC23JW
+ iDgtCIciegh5UAj+HOf8arSKoZT0zz9KezNtzjaCBkU1pQFJzyOoHaqb lH4TTkb0WxGzNhQAC
+ MZ5qkfkQ5+YZ4PrU5kZyw+ZT6mqrnaFUnJ5zU6MXLb1GM5zlVLehqEswTgd uuOKeS6ttGCeB0
+ qs7Yf7rZ579az0voC0VmhpYEEpyAc47ioeWYgA46nmnOxDKyjjZ27mmBSGyCCc c4q5WWw7voR
+ lHKgHAyN2acmQoO7JOce1NO5txPUnJA9qe7jyVClSx7DtTdSTtZD57jix8tS5AJOS MUpaPYxC
+ nkY96rjO5QQWJ/KkYsEyeB3I70pQ6tEQsP6SgjB5GRS5bzmGR6jiogf3bHIB/XFMZ2ST KsOec
+ EVad2kiUkk2JJuaYOHGc9qa7Ycg84OCKc+DEoCkAdvX3qq0owxwPcGh2Za1WiB2ONxB3AcE 9q
+ ol9wH8QBzuFTOTI20KykcnP8qqMCAQflwOhppxvZkxtfXcYSqtnJYEdfT0qhNuMrbQCoHX1qy7
+ rsIRh1/Sq0j7V27i2eeB0NCdpbWEna7OBjLrcgyqT34q1GvKNtYk8cGq0QO/5zyw5J9Par8ZKQ
+ 4O OeFNaKKixSUlui1Ep+UYPX8h6VdiQjoG5PQ1SjOUwuR6k96vh2CrgjJ65FJyadkEd9CdOHH
+ zZPoO 1WFKYBBye9V1XEoKsGJ9quRgA7SoBx19aXNd6blOcbk65CBSQT16danwR8qg5POagVm2
+ qrLjaMA1 bQqqhiRzwKcrp6oIpLzQ7bv27Th8/NS/MzgNkbR9MilC7SCp5H86k2tkM4JyP1oTX
+ cjn7f8ABJUK mJwGABHyk09B8rBcnA596ZCRtA6sOOKl5Ac7eSeah26LUpSbRYiZSNjIV2jjNT
+ DhVPU44HpVYOS3 KHgdh14qbBaMGs0rajknFpskTJkPG0A9+9T4POM7SaroMRbG5HHNT7hyw3d
+ fWolZsaqST10JsgKN 3UcVKpCSYBPoc1WDYOeWBxUpbc20AMc9/Wrd4vTQcnZ2ROhyWx90HmhW
+ B42j1BHpTchF54Oe5pqn bHzwTVpXQ7LcslsJk5G09RSo75IBAye4qBWIX5sjihWxkEgqTkGs7
+ LZ7iastNS3ucyZJyMYGKAxW VlCsx3dahYgIdvXsKlUkrgHBbrSdkitbX2LEfJ+VgpI6Y61MgH
+ mEBlyBwPWqu4hl4yvX3qWJjjad pbGOByKLahHa5cQncp64HfvVxVBTIK4Pas9XPC5B9asoWUA
+ 44oS0ugvZdjQUgyEggLwB7VaQlgwO Ce+KqIyBgRk9yD0NWFbdHlQQSMginH3mRz3diynKEnJJ
+ 6j0qwvRMZyOoHeqiHG7nBzzmp0fadwPT pnvUSUraGlu7Lqv84HAOQcCpFTPLNli35VSUBRuOd
+ zcA5zVkNgYzk5GTWlraoU79CT+IqzAk+lOG DjPJHUCoULeYN3DE9COtSBwDsIAbqTRPunqTKo
+ 3uPBZQT37Ugww+Vc8cgUxm+fgHavSkLYOTgYqI pmlpJXH7gCFw24npSbgVIJOOn1phb5g4PX7
+ xNIpBlZ/4c4ApuKWqTJfw3bJozxtY5WkYgHK5Oema YDjIU8Z545oYFkUYJwucipi7SuxqUn6D
+ 95Em08EHnjv6Um853bScnr6U0H58nrjvSFvlBwVB7Vcm mKKtqOBbcCMNgc0yXnaPmzu9aQnK7
+ kYEHoaYzhXGWww6A0t3oHQa7EggjAA4Pf6UxORxuzt6mnkk RAZU46Co2ZSSQ4HajWzByfLYgl
+ B6McA9aYPvYGcEkYNSO4XGQSe1QNJlkHXvn1oTk42Y76CEcjJI I9aYx+ckEnjIx6U44bkHnJx
+ moJTtAABxkHNS7kryEcsWLAjoBjvSlMKScn3qMsN5zxk9D2pu5t5x 86YzxTk5dNAVO7JFPy9V
+ zt49qUYCYHPTkVAhJhztOfbtS5AMgXOd2atxSbQSbegrsAMLxzxmgFsj CnkZyD0PpTfTuBjJp
+ ruTFlfuk4oa6WFKAfMcsx9M8UwkdCc9uO9Mzt+UHqcZ9qiB2uTtNEvImEls h7FtoA5UdCB1qA
+ suWGC307VI0hOcjAYZzmq2TlyF6EfjSd+U1jGUU7sVmLP2UDnJFVnALYIJ460r 5KjqSfQVCzA
+ uCxGSOMdqmDa0Iv1GEnc2OhqBlHzbvl9z2pzOvGC27GDVc5GSx3Y4xVxWlx2k1dDM qBvHHPy8
+ 00u2SMKG3dMdKRSGB9M5xUJIOCTknk+xqFHmTZTk0tR2SsuQMAr1NIULOjA/MwzxUTZV 9ucZ7
+ ntR5gV9xJwea0SaWhE6bvpuSMWWTHLEelM8xS6gZzjBz3o85TICSMAHFRNIPLD8ccAYqEm9 ZI
+ E7rRakm4lmXCnnrURxj5/vZqMFcsTkEnkZxikkC79yt823nPNN6MnlvuSNJk8YHHB9aqM20tuO
+ 4Me3YCpN/wA7AgAYyc1XLsycYwy9SKuF9hW5XqhrFN+QTtHU5qvM3ICjJbgnOcUjEH5FPINRMQ
+ xA OVLcYB5qrK97jvHQgZPk7YI7dAaqsCsjgHB75q1ICq8ZfmqTjc53EqCKOfm0voEXdO7OOUj
+ zF+cE dh9KvoWZo2ypXb0xWdGSQPfjNXot4QKE4J3DNae0bCSdrMuwhS6hsEYyAK0Yvmj2kEkd
+ CKpxkM46 YI5AFXEUmIEMN4GM1EbvYybu9CULmJeAFBwR0NXN5BH8LDgk1XjVSSVyST8wzU0RG
+ 4koTim02+Yt O/QskZB2LtVj1z1qyhOeQox61XRsDplQcfn3qyAQxkwS3AI/rSumrMWpOQxLbg
+ Cc4BoX7+WcD19q i39CPqQaeOHPQkgcVlGLvfsEJ6PsTL94lTjI6irEbkPh0bb3NU1CrErZIP8
+ AFz3q0h3x8sc7uRVS lfRAn1LC8KDnHf6UsbjyyRwM/eqKM8HJOOQOakT5QCOVzihrqUlH5lkN
+ lV6EYORilUkgbSD9KgDB 3AY4bGB71LnfGAg24GBUxTiU5cr0Jy6CNskAjoM0oBOSGHTg1CcEB
+ iMPtAyacSQgZSG5xTdltcLt 6LQduywyS+DnOaejfvgx5zke1V1ZuQBk+1OVgHJUcAHNNyfUuN
+ +m5cJJZRjnufpSYJCsB823oKgD 5YkkA4596AxKHJwAO1TTk73YlK3Uubl8vkbGyAMmpI+ZgDk
+ ADHJqmADLw+H9D3qRXy4HIyOppyin 1J3TLSkh/vcY71KrfviRnnrz1qupDfNnvU6fMVGPmB7C
+ lJdy+ZblxXQsDjCscD2q4gG1MsDxjGf0 rOU4IBx1/KrcQIj5OBUPR2XUlzvHVl5MmTDcY6GrU
+ U3zFTngZGKoxv8Au++OhA61ZR8OuVx7049g 1toaMUihQzKST61JGCy7eAR0zVIM2Bt+bnGM1Y
+ DAxnIII75oQ4rtoXg2FI9B19KkjI8xx3H61VQq eTnb9alXaXwThQM5HehWdkNJJWJvNcxgsrA
+ 5wD6CnqGWQh+gXvUcZDKFB/E05WzknIIPQ1T0dloR a6FBUgDPvTyxzkABQOtQh1VjgcHOBSKw
+ 8woT9PpQrvVjlAnRlCHHzHdTSPkLcjB4/CmsF8vjOAOv rQFQIM7sEHPNZ8yvuCjyq7GrIS2Oz
+ HJp5kCH5mHFRLsU4zyBwfamsCAG3AnFaJxbsXzJsl+8MFgx BHSnbsSAEEjHGahRv3OSQWzk4p
+ u4MSec9AM1m1d3uNkiyElRwo7DFMO4jAIZsZBo6qG4zSAZdWJH XnFXzaaEuKirsRuyngk5zUT
+ OhdCFIwMZ9akdzvPTj261AwO08HcOnHFKyW+j9SkrLcj5YAqRjFMw pXcGAYHAqQ8n5QNpHFQk
+ FkBQAE8nNCm0gspDWUFm5xnmomBO7d0B59qlwA5Qg9Ofr7VCTtRiSfoe 9SpXVkyXroMccEHHT
+ J4pmQY1XcFwMk0u0MysTheuKiPRiULHGMA1XLrqxqSTSaAyeXFxjrT92Y2X bgdSR2qDBO4MNo
+ B4pwZmwQQvrx1FVaSsyZWtckLZTtx6VE4PlkZwueBipN4VT0AHTPpVZiruMucd cE4pRm7jWm4
+ 0nbCQxIPIFNZgB8xzwKQMecjg9CaiLDCknGOufSm5N9NAlKLsx+4spG4Edj6VE7bl U5LAc5X6
+ 0jN8+ARsx1A601G29BgHORUpNq4p8rSbImIR2AYgHqPSoWJ3YAPrT3YEMRjrmoCQS+GI JpXuh
+ +VyNiS/ODt4+tRZwgxktnk1IwYsF3Dbjiq0nB2c4BBq4S3QOXQa/CnjC54yOajI/d5AP5VI ZC
+ FcfeUtkH6VBJJIXDBTt9BUpyetiOW+vRDB8wCk5HXIppAMTBSODhaQvmQj7vFQOcgqeBjPHHPp
+ VR10RXs5Rt9485wSBgg85/nR5gORj5VqKNS0LKTgHpzSNjIXJVfX6UNO+opqO7HeYfMACk57+t
+ IW QuScHA7U3eABxyehqsTlAAGxuzk+lOe3mRTS5idpBtDYZc9agbhz1Cg8Uudm4daiZgAoYdO
+ BikuZ 7lp+RDkl8bSoHrUL7V2u2ckZ4PepGV+X5yemKruchQykHr7YrX7VhcuuhHIwzlTgexql
+ Ix2Nhhz0 Bq2xBBGRuPI4qhKxDk4yB2p00kyVHXU5WIPgA/Kd3TFacbHYM9CeMCscvm4AVmZFP
+ PbNaceMgsTg t096cIppSe4oQdveZfhU+YeeM8CrUf3mYfdPvVeIncdw4Pp2qyqho9xOPl49qa
+ 31B1d7F2LiIgkZ xjHerYOVAY4XntyKpL9xVzweCcd6tDccLwAG5JqJKzUuwKVtbEqn5gUxgcc
+ +lWI/u5LZ5yBnrVcB t+Bjk849KsIACUK4zycGkpNsi2t7DwckqMktzmnD5hnkc4564qLJLjgh
+ exFPBbaMnopyfeqcXpct Su/dHsSzsNwx64qxESyBcgKO4qqB8vUcDp3qdXAI4KkfeqbJ+Zb8y
+ yrNsYgfP7VOrYUBmJY/rVJW RQ5JIG4Y96lWQPGNnDdBnvUTve2we0didWywC7SM5b2NWFbMi/
+ 3vSqQLo42kBe4I5NO4JZt25uow ad76kc3NKzLgIIO7JK/ewabuwMgg5GcHtVbAC5IIGRg5608
+ Hk4H+9mk7l1Hd9hxyuMnGTz9alU4P tnJNV0O1/mDHJyfapI2UrlclT696mV2hy12J94BYYLEn
+ I9qkJTyzhtxJ6f1qtyZSHODnqKdgAk5z 7UJx0voJSTdkWS6hyQcfhT4h+9y2eDwKpqGExYsCM
+ 8CrCtgEEng9fU1F7PRluLtYthgpcjIOe9Tx bsh0OCfWqZYspPXBH4VYVzs+8oDD8jWqatohpO
+ yRfUlQTkAk4GatRnIznORgVnIExgszccc1Yhbk FidvQKDWTjzEWT2NKNmO4NxgdRVgHLrwTn0
+ qipWRv3bDGOc1aTy9u1m556U3ZO1hve5cUtuKqOCc VbVSAehPQ1TiK+UD2xVuMk9hgqKcm3r0
+ QStLoTJklSwzxjC1Lg7lAIVQOM1ArFMsMdOB7VIrgxqS ec8j0qk29hWcWTpkKMMB6e9SEneDk
+ HnBqEFdoK9hkCnZLLlVbJGcmm2+oJXWmhPG+ZmIC8c8iohg yjcBk9KcG2Rg8ZxyPSkDqcDgMe
+ pFTFNXaCKiODYO1jgLwR60KN8h+b5cZ+tMYswXaBz0NMLFc5Bc AYODilyaaGyjpZMmZflxwoH
+ eoxjaf4mB4pT+8AIOBtzg1AASCoO0g9SaairE+9Lcn2gcgH5l7dBT VZApbhiDjg0zls5bjouO
+ KamDn5WH9fepkronybJfMAk5+Yd81FuYEkHPPbpQW/esAMHbQMFkG5Sc Z4pOKaG1FbiYfYGYk
+ U08qPmKgDn2pWJVflHbpmoHLAAjPv71XvSe4K+iYhOGyOw6CkyMhiMEjqPW lXOct1DcZpjDL9
+ T1zUpp+odbXGM2wl1OXPvUBILIT09aectk4/D0prAbQOrY6elKLj1FfR3ISylT jJCjFRMGVXJ
+ OB1B9amcEJtUbsjJA9fSmHLR/Mp+lOUtdBR03GphHLHJZhjFNdnyNoAH0pGKonck/ 3jSO7Lyg
+ GAvI9qbb0E0ubUiZ8SHcSQTgYHFNJBBUsu7AIGKc3zbWUnGM/SqxY88jIPHHJzTSG7dx zNghi
+ eAOMDrUTsrSA/eHTA707dmM7uD0AqEKxXIXAB60PVXvYmLsrLccVUIpOc9jnioyYtuBk8fN z3
+ pxb5twO4jjHamEfeIGB/OhOyTuDhciZsqQoBNMJznO0euRTyPnycAZ60xuWAJGAfSnv0KurWKs
+ jFcYPA4zTGwTubn/ABqdii43Ju5/CoGBEXHXGcelSlfQr2t0QSMVGMAsB0qEO5UnIz6YqTlosD
+ rx zUMshUkDH9aqTSurERtuiOTb5YycMDjNV9xwTkccHFSuysfm7rkc4qH5TDwCCPyoVt2Cm3o
+ MGQD1 BPIJody7M2cLn86YzKYs8l88AGneYwjw6YHvROVlexXNqMZkYKqdcVG+WTb368GgqPu5
+ wB3qMD94 R3xkGtoqDs0wtZWuPATGMtk9CTTC43MAR+NGAB7fyqJt2PlGQeR71D1l7pDV72Gqj
+ LIdxyR0qJ2O MLgt/EfSnklfvZHrmoyDypXhl7VMoO+pm5JNMrsMxkrgtjvWZNkQspPzAf5NXi
+ eOAwYDnFUJs5Zc 5OevrW9JvpsXFOMrtHNwKAFBYMScjHetCF87Wbam0YOfSstFO7cSPp6VfVs
+ NgDdu5OPSs4qLd0Ja q5qxvldxwUzzVhY2Mqsp3DHOKrQqRlScjsPWrilhGQN3J/Orje/MiG0r
+ 8rLqgbQeQccg9jUiFjuG CTuAxUMZwuHwSefepkJDll4c0ScloKCstSVMA8ghgeecUoblMZwR6
+ 9aiAORgFjnluxp5faQGG054 PrSUXzaDdtEiZT8uPm5PQ9qkz8uN25h1+tRIMyckZzjJp7IpkL
+ Ek5ORipvFsttbk5kAx90NSugd9 24rzwPUU0FQG3dB1NM3dGB+mO9KnJ3skCT3uTO2U2EfKp6m
+ pgVUgJ1B4J71W3YOMFjn9KlKn76hv qe1Nyv5BBPUsB3Eg+RmKipN4+V8HHt2qDLBSQQSOpp4I
+ P8SjPNRFue5UoqLLDOcKu0AkdaaNzcEg rjFMUEDcpL8/L3NOyrEhmAA6g0pNJ3QLmXmKWY5+Y
+ eoxTg4C9Qfp61CVBG9enYCpDgKTtx9fypXY udOKDzB5I5781OHEjEoM88VWiKbyB+JNTK+0n5
+ MDoDVW622CKS0sS+ZsfdkYPQd6nV2JGcdPTqaq 7htUnle/t7ipVQkKwJ2rkY96W+5bk3HYvq3
+ yqdw3Z5WrMcnzdm4x0qjGNoBYjGeo71bDDIUfpURm m7WEloWlbDYxwO9SIwyrAH3qBSNp3ckH
+ oPWrMZ3rgYJBzVRnZbBbsi3HgkscYPAAq2iN8wAHHVjV FFZl5YDvirIBwu3K/Ng5NEn5iXxWu
+ aSsoVFBwvcCrKEAtnP0qggJjKnbkDOcdasoDsUg9PWnN3Wj HZbEjt/CDwecjtVlSBncPmA5NQ
+ BcFQVOegp4Y43ngHr7Uoy7FxkraEylcHHBzyDVne7TFVIA61VB woB4PBI9alz1ZelEo66slyb
+ RNht+5iMY4wKe+1o9xPscDqarA7WQYzxzz3pzEhwGUnjnFFpJlXF3 7DtByMjHrim7ShJHIJzz
+ 3oY7WG7oB6UhfAIyNmeDTadrIcJauw7cDxtJUDk/0qPy24xnceoPamBg rHJy3UUquTIxJx7+t
+ JOUdkEnLuOG3zfmDeoxSfLuY/NjPODTfMCyHnIIwaY2GYkZLEYIBoctrkPV 6inAYsMkAcjPNI
+ rKr+uO9NHQEEbv4hmgFsDcoGGyaHJNWKkiRmQrnkccGmE/KAOVApCQCoYgZ7Zq PI3YXcDnj2p
+ uy3Jd2tGDZGC2cHrn1pgYgZPrn1pc8sXwwzjHpUXykswOc9qS5Wxy0Qrk7QFYMSOv qaiJUD+8
+ 3c04hD1JJzxg9KRVz0IwTyPSiVSCTfYhy2ZDIu3ocE8ioy2EAY4APX3qRvl2AfMp61Vc HcB94
+ fw+1SrSdmXGLkxjjdIkhODjilIOCScHOPanHBAOctTWweeDx+taXb2JmrvcR8KjBcg9OarF WB
+ 5wf73tUzMSncc9fWoGJJIUEnHBPem/dVib2dxkpbegUAn1HekYZbkbexx0p45kUlh6sMVHklWO
+ Cc96zUrrQtSXYa0ZTJUg4681EWcOA3TsfWnO+IgT1xyKaSdv3SCeMU+l5BJNIRmAYgjd6H1phK
+ rG ASMAdCacQPN5YEY4xUUgHBIJA6e9OEbbmfuN2sV3zv4OVJPJ7Uxcqwyy7duQcdanLNvZSpw
+ AOKrn DBsrsGOAaOddTXV+hXZgcYYFgckAdagchs7Rk9Qo61OxCyA8MT1qttOeCO+D60JrbuDV
+ newwg7SQ Ac8cjpUBkJypAU55OKldiqbXHU5qqQGlY5znGMd6laXRiovqIwKuwIwAfvAVCPMAI
+ fLMTkGldynG QADgmohvZevGOfrWkNVdm0bN3X/DjxzGcMAT3Peo844OSxHPtS7ioIABY8gYpA
+ Bt3Akdcj3ppva+ gOpaLQ0BgAOdw6/jSO+F2hTkcDmlLsY1XK/722ojlZgQMjj5vU1XNd3fQzu
+ t2DszSnI2gDkY61Dy Mj+NutWS+ZCRjOOpqgWBdwFJOeTQouXQatezRC5LHpgDg8VSc7Fz6nqa
+ tyA/Pk5x29aoykkHd8oL fLQuW2oSbscvEXB3N0HUHvWhEAATjHGV5rNgceUzltwPQetXomAUM
+ Qxz94U3F3aasT8PwmxGSU3b vmOMe9XEfaoHKjOBjnis+3G4ggjYq45/lV5CCwyvy579qFCysh
+ Xu7dS6Dld2R14NS5CEHIJJ4qoA d3UYHBFWVIbjac4x9fehwT3ZK0JFdgeOFzyalCKTnDH0qDG
+ Fznv0qSMFgRg8HjNVL+6Va71Jd7KR 8wJPQVKWAYAdCcE+hqujqJRwNygjPapAGweMemai10KS
+ 8rEgQK2c8Hkg+tKPuEkY5qNVwCS3U5NS bCBgklQeRUu17FRaHn5lxzn+I+tSoSFIBwB6+tVVf
+ ORjBzhfcd6njUM5xuwf1pTWlnsOcblnduxu UghsfWkOBM67h97oBSKTGCf4u+ecUiBmf5uAe9
+ EYJq/RCT0vHYsLuIBJKrgjPYGmtIwTG0DI5OKX DDojAY6noTTWfcF44OaElo0aQemmogB24BO
+ Ac/SpWHDBWzznB71Dvw5wCASMj3p7EOcjIYcnHpV2 bJcFLqSEKZAo4HU49KcBtAcE4xzk0wMh
+ JxnJ64peUCYO5AORWEZu9gndLyLOAxwMFs8VLGWwvXB6 1SXesi844I5qaORiegwBjNU4y5bvY
+ bskncuq245XO0Hmrsajdy3zHkc9KooPlJ6BQB9TU8LZ5IJI bn3FTvpHYpu0dy6rkoqjBz6CrM
+ YbaT3PcVVTZ1Ugk9u9TI58xQcL/e9qpXtYWltGaEO5lJIG7PQd atKAY1bsRwKqL9xQA/Xr61Z
+ ickfwlQeKXJLdMSRbT5ARnag6GrAJLp/e9+9VUfeCcHnpxU5diBu5 7Ad6qFuZ33HFSUtiz5jb
+ 23HhjwKeoVQwLfMBjGelVjzjJwOwqUMdw45NJJLbQNVt1LQZWIODtI5N P3oMYI56k9BVNZOgJ
+ z3x61MAShYY+hokrEyunqywHUD5myCeo705STjHQAgfWq2/quDjscUuS020 EjA5Oe9EdtirW2
+ Jzgq28YYjjPpTcbcDdxjkVG0mTg8sMA+1EcqLMSQxAPr1os+XQlybdkOZwJPmX K5GPpTGYLJx
+ 1I9aN5KjcFVevNMbeu45UjORx+lCkirJhkiHaMZ9x1pCMMmWGTyCO1RF22MxTPWgP iE8AEf5x
+ R1KvfdKw7euCcMMDFMySmwsGJ6GoSSAOhXv7GgsxYlfl9eKTTuHLtaxMSWy5IJHA9qYC dzHID
+ DpxUO7Oed3c4oDgxkkgMBjFVJMlwWo9m3AABgAevqaZtYMTn5c8UEqDhSffmoncbM5AGMcV F7
+ 6WFboPXBlHOcGkLnO3BxUGSGU5Bx0AphdnYnjj0qlRT2Y4au9ywx+U8/w9PxqNiDEW6cimLKS4
+ LbdvUkd6Vj5iuSAVJ7UJNOwnF3SYwcg/UEVGzH94QdpzgDFPHTYDt96TehfO5cdTQ3dkR0ukVi
+ Qs Pynvj3pN/wC5YL1zg56ipyEaL1btioXVgzbSvXoKlyj3EktmyAqythTzjmkYsiK2Rk4PSpM
+ KEOSd x/Wq5yHAAyB0qU5PU2u7DmZc5wCfWo+THubnBwMUH+7kDI4yKR2xGAOfWr5Laoydo2Eb
+ arBiSCeM VCzFmQHsMfWpc8c9e2KjfcTkOARV3TKtdDcgjdzkngVUk28lsgnoKmztnbecY+6Kj
+ b5n3EcEcmpb irNAlZ67FVmUM7gDBHeqzE+XnrjJwPrVqZx8uQSuB+VU2fdJgKetJy5TS7b22K
+ 7kc43FTzzVd5dq Y4Ge+OtTSBQ4CnHt61C5VtpAwAeAeorRxW7J5o3vYPl3hB+vvULZIC7Tx1N
+ K2dmWyQDmkVkZujZ6 4rNcqRN1zXEEny8DvlcjtURkG4b/AJVPb0pzkYK5yWGQF7VCS0jbSR26
+ itb2d7D5Fa6HEkMSD26e ntTS+dwB2kcn2PpTG2iUjOPm7/zpp/iLE57+5qtRJXsmSFsBGP3jz
+ VRzk+hxyfWpBvaLocVE5UPG OhBxTp3TEmkRuwE2Wxg+lULggN1wMA81ckZQxU4K5+UiqExOeR
+ uQjrjrQtdNrgpdzlLcLt2nue/r V5CDcKT17g1TVmEYBKqSCcEdKnjyp6goRxn1pxmpRV9xcrR
+ sqSIwFwcHnFWY2YcBwQKzF/eR9Tw2 G9quptC5Vvm4xnuB1qFbW7ErPQ1g4OC3c4qZJcSbkPAH
+ f+VUIwSmVPfuasglUzgjjOe1WlGWjMuV bssmYmFRgcYzxzUivuIXlXHBqHLBjnAA6kipQqgoc
+ 8k4J9aJuNti+aNuxOFUHO3cR90jvUse0yY2 sGAHWq4DKxDAlQ3B9KkQcr13Dk1k0mm2UpO2g5
+ QDPySd2fyoO7eu4nBFI3EhOcL0zT2zxyDjnFCY 7dmPGDjAO5epqQf3s8YxioEJ8tTzyOR3zUg
+ OUwzYHXP0qbu9rijpLQnX7xLHginKwMeXGCTUKA7Q Mg7jU552pj5geBim7LSxd3fcUM28bySN
+ pwBSfOCAD25GKjLOW4+THHI6ikjU/OfmLA8VUYpapgpv XUnBAj+YZYnj2pW2FxtIA24z60Fw2
+ Mqxz/FTCVYZA2gHPX9KNXrYyTe9iRGRdxYEgDjFODqUyAWV uuOxqH5ZGbAIBHTPenKxCJzg9R
+ 7Vnyxi79S0kt1qyePGdrA5xjJqWNV285GTjmojk5crtxUqkG2Y k7z14FP2lxxeti4BmE4557V
+ ZR8EYZeOnpioIyFOBgAjBz3qUDg8AjHbtWSnG9jaLi4l8EBV6MT/d 71YABbI45qjCFb7pIGOR
+ 6VawAB97d654q+ZNkpR2vY0oxmHk8dRnvU0WDJllwM96pRD5gCe3HPtV 5WXCBlOSOue1Jc1tD
+ Nx6LUn8shixb5cZODUytmIdMDoTVZDuBwdwJzipWKiLGCCCCaHZ9Q12LHmf N8oUEAcmpFLlmx
+ jAPp0quDuUkDL5FTLkPgZG73pylG4XdrJC5AHzcMD1/pUqOcDLDkflUDZMzIeA uDk96RDgtxw
+ OgzV8nPqXJpKzJiWLAnB9cVIZG3HLA9s1AN23Bz/hSY2p3Yk5Yil7t0Q5XsiZD++D MwPHPvSg
+ nBGB1yCahyHwRkDHQinA/MduSDyD6CqcmU273uSbsyKcbgTSOxy2eFx61Hu3IMHJA59q iJBY5
+ BIPIJrJJp3C33Em4mDryp549aZuA/2Wx0Paos4yM9emKXdn+JQQOBjvTkne4pX7A0uZXB5R ue
+ D0pA2EBDHA61Az5IIQ5Jwab0Hy/Pxzmh9UEGm9SYuBk5xjj61BG4JI7Z/E0zcMckEn17VCXKue
+ wzwR0rS9o2YOyLJkG4rjn69KYGCtlhnnj0quXbeSB8vGfembgWVhnJPc1HvdGOfTm1sWgyqAc5
+ 5z SkqVDD5SeeKr72YLhcADoKRSzsRnr93j86Ol7hJR6FjLAAgDIyMAU3ewjQFSF7ketRlmwOm
+ WPIpy ksm0YUA5we9Ta24003ddBWcA5/p0NIrAKC4UkjsKQkLxtLHdwR6U6QgQBgCAOORVwu1a
+ xKvIYGHz Y5HsaiJcyZCkc80rn5xkgITkgdqjLeYCoODn8aU421Y5JSkrobI53YZCee1QgrwcN
+ 05Ge9PYqsIU /M/cUw5ZAQRVr3V6kPYYxAm7n1wcUEgRMU4U+v8AOgsMKWIzjkVE6kng4Hfnis
+ 5PYbswZioBOHx3 FR7gz4Gdx7D2okKiJuTnoTUAbEhAPAAJwOa2i9DSUdE0xzk7y2OAeMjp9aq
+ vuZAM9QMAVYEm5Sg6 EZqAgeVgj5T3704zsmmjL3272sRHgnLDaowc1VdiR8hHP3j3xUrqC2GY
+ sSc9euO1VmYZZh8ox0NZ JXC+mpAzjaFK98Bqru2A5XGCcZxxUhy0o+7jGM9qiYbot3Bwegq52
+ SL1lZCMvzjJ+UDmoCxzkrgA 8mlYtglgctyfbFJIVXf8w3HHFZp9EJaaDfNydwPU56U1RmZstt
+ PPXvTQSFHzDI6D1p24hgGGTtNW 9CXGw1SAdg5XPJpPlDFEJPuaa20LlGPH60Fj8zgj3x2oe+r
+ G72uMbgkk5QnjFR5DjAPQ9+tBYHcx Ofl4AqM+oPPc+tW5XVzJS11K8u3e/GFU9aoyktje6g5/
+ MetWJGYI2RuGc/Sq0pVTlFJ479jVOTlG yZautf8AI5SMtKRwR6A96uIMRkvztzgVQi3ZJzkMc
+ DHarS+aCfYDGRSg76tk2cn2NGDLg46Edv4q tIWBIUFRjlj/ACrOjkOGz94A4CjGKtJlgNxIXP
+ AzTs7astrlgaSEh1O4qDyCatpuEqh2Jfb09az0 KeV3yDgjPSraksDI5Jx6U4prXoRFrYuId0o
+ BJDHOM96srgJt5z3HrVIK7Mr5AYcirEeTnuTgDAqb LdBbXcsB2aZfmO2p1yIjtOXzwB1NVeQw
+ QnA5O4jgVMpwwYn5QOtHu9LBp3JlbABCtjHPFNLfJtJy yjkAc89KGZxvAPyjHNIoOZGLBvXFV
+ G6V7jSVtCdSuxcNtcgcmn/dIUAMevFVQ5WLOFHapFLBtzDg 8E9vrWbpu92aOF47FknCphfwBp
+ 4bMYf5s98/yqJPvbicANjrSNIFyAOjc0lByZMVJqyROwVlO0nr gUAKrhQ2GJyMmo8hWYtnO7j
+ 3pw+djgEknjFTy9wimtCQFi/JGAccjrSgkvkfdB6VF1YAnjvz0pFP +kDdwo6Z7Vak0i5OKehI
+ h/dEcbi3GO9Skny2+XjcMYqsXCttB6HginjcioSCV7+9Z1IpPzGnqtCy WDMx3deasBsY2jO0Y
+ PvVPzE3jIyuKmhYs+Typ9qhpJik4vVFoHJ4ywPbNXkPyYbIU/mazyMNuALD 2PWrYKkDkqfQ+l
+ Obi2mNSfQvRHPygMBmrCu+/DLuAbg/1qou/goRluoNWkIWXG8EDgiiKT95jcW0 y7ESQSPvdDV
+ xQUVQysR0+maoQPjb1GRmrMbnbnJGPXmh/F5Ecr6F4Hy4Tzk9x3pUJfGSMDqMVEGx uZj6dafv
+ XaQrAgcHHWk0gu7WJsjzRtJI9qN37veu4E9aqpIN2Oq59amG5WY7sqT0rT4dRqNupYST OflJx
+ xUhZVjU4JzwKqxHI3MduTkink/u2wQcdDU/MTUWyRmZioDdjg0FyBjBJI7VArDcCM7s4p4J bB
+ bIbg1EnYNVp0LGW2bgdxxzx3pAxL9QGPXiq4O0sOeT60iFlLbs4/h9abcXqg0ZMZSIsdTjBIHF
+ DkeXGD90Dj3quzjeQVYgjpS5XYDg5B6Z6Umr6hyajzIWITjIYYNM5MjAnoRzTC5eY7SMk/Lx+V
+ Re Zg8tznJq3toilFW0JJCGIAb5hy3tVdnKMGwfmPTPanMAZHYA5LDAFQOT5uCc4PGaFq7MqPY
+ eSDnI ZQOBk1A7nkxlQMdSP1p5ZjGSRk96hZ9q/Njbjn1xTjduxKuhWcsgBK9QRjvSbgH6Ajoc
+ DvUGcqCO gPSlyhcBj3zwalvoyZJvRonZgAAThgeSPWlVgp6E88c1XEuG+ccnue9CnccqRgdhT
+ vrqwTsixE37 3kEHOeTU/mZyEA39c1TEkaz7gWz2p6kFgUYBei+tDjZ3a0CUepbA+QkMCB3xUR
+ J3krx7Z4qNiyFl PcY49aUMhUZ4yPWnDmtcLisQVLEdeMmmvnawXHHem4DHhjjPAph8wgYyfXF
+ S9VqNOMhJCSEPAzxU OSHb+HnPPSpSNpyPTjPrUEj/ACsAOeBTbVhJdENdhtTZtb+8RUO4AlSe
+ e1K4ZITk44qDqSw+4OAe tVpYdl0ZKzA5yTioy2NwyMjofam9X5yQR096iZmUsTjavAoTjazHK
+ 0WrClhuwOucZHpTHD7Gw459 ulIz/ug2BuIOAOtQblRck/Nt6ZqeRPYUY3I2l+ZdpUgDP4VXMg
+ MZAUnBJBpAzCViFyOhpu/94eR0 xjFXyRWlhcq16jH24Vty4B6Adc1UkkI3JhQAe4709w/PKbB
+ 3xVZ93m/Ng88E0RjHVFQ31FILxgZA 44GKrF1D/Nz1GfU1K+4jeO3FRsBtAce+aatorkyu3zDQ
+ Dx37kU0LtZiSSccc08ZUqOQDn/8AVTXO 11BGPX/ClOWtgc3JaC7gykg59qiB4yWUKRnn0pec4
+ 4A6moZWG0EAYB5I6CiGl0jPmauhQY1QYPP6 VGZG344A9xTSf3TBCAcdD2pj/KQcj0OPWnNt6W
+ uNxtqROWxhsbjxnHBqnL5iySE8j+H2qVjtUt82 BkcnpVZyzEnO0gZwTUrnirrcqk3FnMoFZSM
+ nIPPsam3MqZyc5G6qyFDtByB61KSwK7fulT1Nb02p PkbFCfOXBOD1YZ+nXFXbeUmND/F/Ks1N
+ vm4PzY4AFWomKjA4IGeaThFaW1JaSdjTjZSCFHG45zzj 61MjgySDJOOlZsMrgeUFwauI24ggB
+ eOR60rSStYIxcXroaCy5QY5bp+FWklGAOQV61nAbnyCQe5q VQiyHcwI4ORRypPbUtST3NMN8h
+ JU5wM5qRJS3yMPlAwDjrVNGBJ35H49qkRwQevqvtVciT0Kv5bF osQNucZ7U9WY4I645I7mq6k
+ EAntyc08PlvMAxyciqTSexE2nsSxmQjgKTnk4qT5sNuPfp61WL/IM kkjsKkTax3fMWHbNZOLS
+ ci2tidB90BiQRkg1YVhjoMZ9KqAgAqM5wATUwRRCGLcdeKzlNaLqTKLW o/zMR53oVye3NPDNs
+ 4YAbutV2UeWQAVwRtzTztb5NxY5HT+VaqXLG44vlSaHnaYiQQW3cnPFCnbJ nIJ9xTPlRDkHIy
+ eaQujfN944zgVF27dhqzW1yQyq2xM5OetOUzSOSBhe31qDzNpBMZweoxzVjcfL BQrzyBUybto
+ S5vsSA4ZiSd2fwqyPmiYAhXBqiCS+Xz6nHaplY4CgEg1EU76Cv1ZfRxsIZieRg1bj dTM2RgDu
+ azI5CJAxPy9GJHQ1c3Hy02+mC2M5ob63K5tDQibDgEH5hn6VZjI6Lwcck1RR+FUN8xGc elWVY
+ 7h3GODRF2K23Reib94MOMhcgYq6shBIY8HI+tZiY3jHLY5x2qwrbiRnqauotnuKW2rLgkYf KC
+ ckdDVhjj5Mc7dwqisgG2Rh16D+dTGTc6nhvQ09biU1cs+YDtBHG3kjvThKMb9rECqxZhHy6g/T
+ ofSnBm8plbkg8epNFkDkm7ItiRWdgSMgYwO9GflxkYYVAM7gDwSckgU7cAcFx0NZKLvfcXXQlO
+ 0K Q3zD245pDNu2kcHOPoKrmRQwRiXGM8HrTy2JMAqAOuacoq1maKyWo9WZXBYgnHSpeDzkjnH
+ Xiq6u zA4BJHT3pWkURnggZ5FKdrabmUtWmTl03D+Jj+mKiZsPkKykng9qg3MjoY1Zwe4/lQST
+ FvOd3YZp xSTsy2rInQZJzjI6c1FvVAey+4/Wog5zvORk460oCnJORgc+/NUopJpkKbUrsVpAE
+ 65weTUb8kk4 B64PWlDYBPBXd0qu7MA2GXI6Aj9KlRUmXFtrQC7MeCMY4pCcsSQ3A7UbiAASOn
+ YUxuFBJ4H605WS CU76XGNjgYO3tUeflyGUYPzE9qCQyEbu361BuInA5IAzz3qqabjdFNN2J8/
+ KOdx7VIoB+YEe1VlI LKd2MHjPepuTu4ycZPNQp3eo5/DclHI5OR1z/SnoQuGGNpORVeMHAJwC
+ vT2pzMwj3ZJx6Cq5r3I2 67lnfucjIzjOD3pPMbLJ8u/OR/hUO5XjQ888/WlDK7Es2DnPPYYo0
+ S1JSTeqJTy69jjioSSzHk4z g4pxfMTHjp8opm44APB6kd6Oa6uO3RBIQGGM4Ax1qBypJy4Az0
+ 70rP8AJ1yQ3PtUZfJwAMepHein a9hra41i+0Asufeo9xMJXIUDBzimtJkAnp1J9aY7Mq4JB7c
+ d6bTfQGlLqIznLZ7ntwTUDOC3cknI 9qldm8oAKC3fFVVYb2JIyetSo21auIWORTKQWzkHmoWw
+ ASGzx1NTOyiM44B6MB0qoVAlJzv+Xrmq VrdhOXYQsHQ4wr+lQH5lY5A5HSib5ckZBGDTVfgsS
+ o6np1qpNdCYySdrETAFT83IPrVZidzNkHtj FTtIARhSQc81A58uMOgJJ4ORRdtFvfyItwYspB
+ OB2prL/o4LHljnGeKU4YnjBx9KgYqI+pLA560J pNJEtdEDsfl5AGaRtvmMfmPI6GmuS6rgZwc
+ emTSrw4O5cE/NmrltZEpaNtjQ5MZPJB7Y5qFlxGzI cHuD3pWYtKcDJPSq7FgwQsGIB6Ci2uhX
+ PZWHO+bfI4yO45FRO2QQx4OMmnsy8YOFPXNRMwwU424y OKm7a02Jb6WK7kqzBsHBwM96rvgxb
+ jnrxUhO6Rh94YBzmqUhbywp+bk4Naq0ZJX1Hfn0OaVmBw2O OelSCUNtKHcvIqrHiQ78nKnnnr
+ VxDhfkwPTIpKKb5uoc0bNrqTq+VYDO4kfjVrBJzzjg7qrKSrkM ue4x2qwjbshc4PJJ9aG1cSl
+ ZbFtWCyEg5weoqyGBjHXLHNUlZFBAYccEVZRiHxjgD07VD91Dbk1c vrgOBndyCMVZYgMB1GDt
+ OKpoTgBCCMfL/jVgblxlgSDyxHFCfVCk3uWkYr8p5JGcjoanJwPTjnmq qEgKQRycCnqxySxUj
+ 2FUtWJa6l778RAYcenehThixIK5waqKQV3Ipz9aCxymDjHUUkpWZcWnp0L2 7AZGxjtx1oBHlk
+ LgHpzVXeTtZ+BgYAp4kB+bgc85ocJLYOVK6SLqkbwrNyVxxSkjYOcDIyKqiQ7g WxwMcDvS7+j
+ jle4qfZtvUtKz3LIYgkHLAHpUuWRC4AA3DHvVQElcHC89TR5hGd2WA5FW7WaQra/o WncswLDA
+ waiB+fPPHpQZAzEHAGR/+qotxwAMH09qhb6ExbtZqxM8hwzbScfrSJJ1G5Rxx7VXL/Ju H5e1K
+ N4fGzI7nFTy2+I2SsXQxfcSdw3DO2pEdhls/TFU1fYjKDwx9KsIxchPup9OtTNSvqZKbitC 5F
+ Io2jgluevFXRKTIEJwO2Ky4yAcKQVBJJx0qzC28lyDg5/Cl7jfMLmcmaQbCb8jhueeatLIzHrw
+ elZqksSCpVeM81ZSQKMD5gXz9KSSvoyoq2xphwp8wn5c4wOoqZWbPDDJ5OOlZccn77BG5Tzx7V
+ dV wMEcZGTU8/Ky57abl9GUjG7d7elPWTqvJJ6Gs9HbqBxg4zU0LuwUEEgLkt6GtL2V2yaitZp
+ F4N9z JJyccVMCobnkY79RVIvhVPTj73anKxKgnJ5GeetO8rXuKMVqjQV/lPzDJqMygwg7SWHB
+ qMSKof5c 5OMelJhd+7kDvRdaihZbkhZty52gZ9Ke0gyeRjFQAOJSeo4xn0ppYYAP0pWTVjRS1
+ 0LBmIfCHHAJ JoeXEhLdevHvVbdjnHQflSblKN1YD35NDjGJPLbct7yqqPfJpTKM4PUHt3quGV
+ iOchvSmJgH7rHa MHmp5Y31E9XqtiXeS5JXqeMUrKzSAbdox92oXZnU+x4pu5mdxnGDketXEpw
+ 2bH7nLjO3AFNdSpPB L9aYWGGJPWo5pG2BOcj86Uua+hV76iuRvBz0HamBizNwRt9+tBx8rbXA
+ 2VGwKsPmOxuQauLurDbW 0RJCBIpz82OcUAEHhgVNMBO4MT14z6VG2clFDDkEZqeS79A5+ZrXY
+ mfgLjv+lSEEREr8pz371CCu SGBwB0qMj90x5GW4PahxStYlSu73LasScMw47Cl8zJTuuccGqw
+ bDh8hmyQRinq6iNhkKO+exqGnb QN3ZFkkLEMck9famkgEvkF81XD8rn1/KpGkVkJKkDuaTi1o
+ Db2JCB5YGTu6ioncHaeQT1/wpuT5K tzwcnJqMOzBtykKTnPvVJcruhRl+A5mC8DJz3pm9lG11
+ IZumaU4AJYhR1x3qJ2BbeMgD5sH071Sd 9LCnNRFYKVCg9Bgiq7MFlIc4HT6Uu9ch+vHTvUBYt
+ IzldyZArWF0rkpNK4M5KEg9DnH0qENmDpli Tx60skp35UYUcEe1RO4weOMYX1qJPsK1la2orE
+ BNmd2R3/hqvwAVB4xkihpBkDn0YUw4ZGyMccYq oRvuNq71GHexJHzY9aiJf5m4PTIpzMyOCAW
+ Ug/yqPK+WE2NnBwc0QulsU7Kz6EW0+SwLZ9R+NRs3 y4ZsnPBFOH3/AJQRxyDURwM78Ar2pv3m
+ J2bI3I3MAQA3vyKiLHJC8sODTz87EEHnmo1IBw3IzQ2k tR8sVqwHACBucc+1Rs23O0n3zTcru
+ Y5O4DkZpnmbc56+9K2l0Q5/IduZW+XHGSeKruQWDZySOR6V MMKBkj3NVzlpCSyZI+UAdKFbuV
+ HRa7jG27cLxnrk1CThWHJwOQKkcbtuMFfb3qu0qguFU5HynFXL
+ l5dTOd1qxgYfaS4U4KY4NUJA23byzdMDrVl2D2+5c9gT71Rk++cB85yMmiEXF2Jb5ndH/9k=
+UID:934731C6-1C95-4C40-BE1F-FA4215B2307B
+END:VCARD
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/AFBB77B8-0438-4825-A1DB-A75D76B6C3A8.vcf
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/AFBB77B8-0438-4825-A1DB-A75D76B6C3A8.vcf 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/AFBB77B8-0438-4825-A1DB-A75D76B6C3A8.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,25 +0,0 @@
-BEGIN:VCARD
-VERSION:3.0
-N:Here;Custom;Fields;All;
-FN:All Custom Fields Here
-NICKNAME:custome
-ORG:Major League Co.;Macosx server group
-TITLE:QA Engineer
-EMAIL;type=INTERNET;type=WORK;type=pref:custom at example.com
-TEL;type=WORK;type=pref:777-777-7777
-TEL;type=CELL:8888888888
-item1.ADR;type=WORK;type=pref:;;1 Goroku St.;Mountain Top;CA;99999;USA
-item1.X-ABADR:us
-NOTE: Many customer fields are added
-item2.URL;type=pref:http://www.example.com/~magic
-item2.X-ABLabel:_$!<HomePage>!$_
-BDAY;value=date:1999-03-18
-X-AIM;type=WORK;type=pref:custom at example.com
-item3.X-ABDATE;type=pref:1995-05-21
-item3.X-ABLabel:_$!<Anniversary>!$_
-item4.X-ABRELATEDNAMES;type=pref:Aho Sak
-item4.X-ABLabel:_$!<Friend>!$_
-item5.X-ABRELATEDNAMES:Sanma
-item5.X-ABLabel:_$!<Assistant>!$_
-UID:AFBB77B8-0438-4825-A1DB-A75D76B6C3A8
-END:VCARD
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/AFBB77B8-0438-4825-A1DB-A75D76B6C3A8.vcf (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/AFBB77B8-0438-4825-A1DB-A75D76B6C3A8.vcf)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/AFBB77B8-0438-4825-A1DB-A75D76B6C3A8.vcf (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/AFBB77B8-0438-4825-A1DB-A75D76B6C3A8.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,25 @@
+BEGIN:VCARD
+VERSION:3.0
+N:Here;Custom;Fields;All;
+FN:All Custom Fields Here
+NICKNAME:custome
+ORG:Major League Co.;Macosx server group
+TITLE:QA Engineer
+EMAIL;type=INTERNET;type=WORK;type=pref:custom at example.com
+TEL;type=WORK;type=pref:777-777-7777
+TEL;type=CELL:8888888888
+item1.ADR;type=WORK;type=pref:;;1 Goroku St.;Mountain Top;CA;99999;USA
+item1.X-ABADR:us
+NOTE: Many customer fields are added
+item2.URL;type=pref:http://www.example.com/~magic
+item2.X-ABLabel:_$!<HomePage>!$_
+BDAY;value=date:1999-03-18
+X-AIM;type=WORK;type=pref:custom at example.com
+item3.X-ABDATE;type=pref:1995-05-21
+item3.X-ABLabel:_$!<Anniversary>!$_
+item4.X-ABRELATEDNAMES;type=pref:Aho Sak
+item4.X-ABLabel:_$!<Friend>!$_
+item5.X-ABRELATEDNAMES:Sanma
+item5.X-ABLabel:_$!<Assistant>!$_
+UID:AFBB77B8-0438-4825-A1DB-A75D76B6C3A8
+END:VCARD
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E1.vcf
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E1.vcf 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E1.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,11 +0,0 @@
-BEGIN:VCARD
-VERSION:3.0
-N:Thompson;Default;;;
-FN:Default Thompson
-EMAIL;type=INTERNET;type=WORK;type=pref:lthompson at example.com
-TEL;type=WORK;type=pref:1-555-555-5555
-TEL;type=CELL:1-444-444-4444
-item1.ADR;type=WORK;type=pref:;;1245 Test;Sesame Street;California;11111;USA
-item1.X-ABADR:us
-UID:ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E1
-END:VCARD
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E1.vcf (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E1.vcf)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E1.vcf (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E1.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,11 @@
+BEGIN:VCARD
+VERSION:3.0
+N:Thompson;Default;;;
+FN:Default Thompson
+EMAIL;type=INTERNET;type=WORK;type=pref:lthompson at example.com
+TEL;type=WORK;type=pref:1-555-555-5555
+TEL;type=CELL:1-444-444-4444
+item1.ADR;type=WORK;type=pref:;;1245 Test;Sesame Street;California;11111;USA
+item1.X-ABADR:us
+UID:ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E1
+END:VCARD
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E2.vcf
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E2.vcf 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E2.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,17 +0,0 @@
-BEGIN:VCARD
-VERSION:3.0
-N:Contact;Mulberry;;;
-FN:Mulberry Contact
-NICKNAME:mulberry
-ORG:Apple Inc.;
-EMAIL;type=INTERNET;type=WORK;type=pref:mulberry at example.com
-TEL;type=HOME;type=pref:777-777-7777
-TEL;type=WORK:8888888888
-TEL;type=WORK;type=FAX:5555555555
-item1.ADR;type=WORK;type=pref:;;1234 Infinite Circle;Exampletino\, CA 99999;USA;;
-item1.X-ABADR:us
-NOTE:This is a contact created in Mulberry.
-item2.URL;type=pref:http://www.example.com/~magic
-item2.X-ABLabel:_$!<HomePage>!$_
-UID:ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E2
-END:VCARD
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E2.vcf (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E2.vcf)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E2.vcf (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E2.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,17 @@
+BEGIN:VCARD
+VERSION:3.0
+N:Contact;Mulberry;;;
+FN:Mulberry Contact
+NICKNAME:mulberry
+ORG:Apple Inc.;
+EMAIL;type=INTERNET;type=WORK;type=pref:mulberry at example.com
+TEL;type=HOME;type=pref:777-777-7777
+TEL;type=WORK:8888888888
+TEL;type=WORK;type=FAX:5555555555
+item1.ADR;type=WORK;type=pref:;;1234 Infinite Circle;Exampletino\, CA 99999;USA;;
+item1.X-ABADR:us
+NOTE:This is a contact created in Mulberry.
+item2.URL;type=pref:http://www.example.com/~magic
+item2.X-ABLabel:_$!<HomePage>!$_
+UID:ED7A5AEC-AB19-4CE0-AD6A-2923A3E5C4E2
+END:VCARD
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/F0A6918D-8E09-43FA-9684-226810B8A96F.vcf
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/F0A6918D-8E09-43FA-9684-226810B8A96F.vcf 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/F0A6918D-8E09-43FA-9684-226810B8A96F.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,2010 +0,0 @@
-BEGIN:VCARD
-VERSION:3.0
-N:Inc.;Test;;;
-FN:Test Inc.
-ORG:Test Inc.;
-EMAIL;type=INTERNET;type=WORK;type=pref:testinc_sf at example.com
-TEL;type=WORK;type=pref:777-777-7777
-item1.ADR;type=WORK;type=pref:;;3 TV Street;San Francisco;California;99999;US
-item1.X-ABADR:us
-NOTE: Company with picture
-PHOTO;BASE64:
- /9j/4AAQSkZJRgABAQAAAQABAAD/7QA8UGhvdG9zaG9wIDMuMAA4QklNBAQAAAAAAB8cAVoAAx
- sl RxwCAAACAAIcAhkAC1Bob3RvIEJvb3RoAP/iG6hJQ0NfUFJPRklMRQABAQAAG5hhcHBsAgA
- AAG1u dHJSR0IgWFlaIAfaAAEAEwAJADEABGFjc3BBUFBMAAAAAAAAAAAAAAAAAAAAAAAAAAAA
- AAAAAAD2 1gABAAAAANMtYXBwbFYcEOZVYuhIRg5LwLIi62wAAAAAAAAAAAAAAAAAAAAAAAAAA
- AAAAAAAAAAA AAAAEXJYWVoAAAFQAAAAFGdYWVoAAAFkAAAAFGJYWVoAAAF4AAAAFHd0cHQAAA
- GMAAAAFGNoYWQA AAGgAAAALHJUUkMAAAHMAAAIDGdUUkMAAAnYAAAIDGJUUkMAABHkAAAIDGF
- hcmcAABnwAAAAIGFh Z2cAABoQAAAAIGFhYmcAABowAAAAIHZjZ3QAABpQAAAAMG5kaW4AABqA
- AAAAOGRlc2MAABq4AAAA ZGRzY20AABscAAAALm1tb2QAABtMAAAAKGNwcnQAABt0AAAAJFhZW
- iAAAAAAAAB7vQAAQXsAAAJL WFlaIAAAAAAAAFYqAACp0AAAFF9YWVogAAAAAAAAJO8AABS1AA
- C8glhZWiAAAAAAAADz2AABAAAA ARYIc2YzMgAAAAAAAQu3AAAFlv//81cAAAcpAAD91///+7f
- ///2mAAAD2gAAwPZjdXJ2AAAAAAAA BAAAAAAFAAoADwAUABkAHgAjACgALQAyADcAOwBAAEUA
- SgBPAFQAWQBeAGMAaABtAHIAdwB8AIEA hgCLAJAAlQCaAJ8ApACpAK4AsgC3ALwAwQDGAMsA0
- ADVANoA4ADlAOoA8AD1APsBAQEHAQwBEgEY AR4BJQErATEBOAE+AUUBSwFSAVkBYAFmAW0BdQ
- F8AYMBigGSAZkBoQGoAbABuAHAAcgB0AHYAeAB 6QHxAfoCAgILAhQCHAIlAi4CNwJAAkoCUwJ
- cAmYCcAJ5AoMCjQKXAqECqwK1Ar8CygLUAt8C6gL0 Av8DCgMVAyADKwM3A0IDTQNZA2UDcAN8
- A4gDlAOgA6wDuQPFA9ID3gPrA/gEBAQRBB4ELAQ5BEYE VARhBG8EfASKBJgEpgS0BMIE0QTfB
- O4E/AULBRoFKAU3BUcFVgVlBXQFhAWTBaMFswXDBdMF4wXz BgMGFAYkBjUGRQZWBmcGeAaJBp
- oGqwa9Bs4G4AbyBwMHFQcnBzkHTAdeB3AHgweWB6gHuwfOB+EH 9AgICBsILwhCCFYIagh+CJI
- Ipgi6CM4I4wj3CQwJIQk2CUsJYAl1CYoJoAm1CcsJ4An2CgwKIgo5 Ck8KZQp8CpIKqQrACtcK
- 7gsFCx0LNAtLC2MLewuTC6sLwwvbC/MMDAwkDD0MVgxuDIcMoQy6DNMM 7Q0GDSANOg1UDW4Ni
- A2iDbwN1w3xDgwOJw5CDl0OeA6TDq8Oyg7mDwIPHg86D1YPcg+OD6sPyA/k EAEQHhA7EFgQdh
- CTELEQzhDsEQoRKBFGEWQRgxGhEcAR3xH+Eh0SPBJbEnoSmhK5EtkS+RMZEzkT WRN6E5oTuxP
- bE/wUHRQ+FF8UgRSiFMQU5RUHFSkVSxVtFZAVshXVFfcWGhY9FmAWgxanFsoW7hcS FzUXWRd9
- F6IXxhfqGA8YNBhZGH0YoxjIGO0ZExk4GV4ZhBmqGdAZ9hodGkMaahqQGrca3hsGGy0b VBt8G
- 6MbyxvzHBscQxxsHJQcvRzmHQ4dNx1gHYodsx3dHgYeMB5aHoQerh7YHwMfLR9YH4Mfrh/Z IA
- QgMCBbIIcgsyDeIQohNyFjIY8hvCHpIhUiQiJwIp0iyiL4IyUjUyOBI68j3SQMJDokaSSXJMYk
- 9SUkJVQlgyWzJeImEiZCJnImoybTJwMnNCdlJ5Ynxyf4KCooWyiNKL4o8CkiKVUphym5KewqHy
- pS KoUquCrrKx4rUiuGK7or7iwiLFYsiiy/LPQtKS1eLZMtyC39LjMuaS6eLtQvCy9BL3cvri/
- kMBsw UjCJMMEw+DEwMWcxnzHXMg8ySDKAMrgy8TMqM2MznDPVNA80SDSCNLw09jUwNWo1pTXf
- Nho2VTaQ Nss3BjdCN343uTf1ODE4bTiqOOY5IzlgOZ052joXOlQ6kjrPOw07SzuJO8c8BjxEP
- IM8wj0BPUA9 fz2/Pf4+Pj5+Pr4+/j8/P38/wEAAQEFAgkDEQQVBR0GIQcpCDEJOQpFC00MWQ1
- hDm0PeRCFEZUSo ROxFMEV0RbhF/EZARoVGykcOR1NHmUfeSCNIaUivSPVJO0mBScdKDkpVSpt
- K4ksqS3FLuEwATEhM kEzYTSBNaE2xTfpOQk6MTtVPHk9nT7FP+1BFUI9Q2VEkUW5RuVIEUk9S
- mlLlUzFTfFPIVBRUYFSt VPlVRlWSVd9WLFZ6VsdXFFdiV7BX/lhMWJpY6Vk4WYZZ1VokWnRaw
- 1sTW2NbslwDXFNco1z0XURd lV3mXjdeiV7aXyxffl/QYCJgdGDHYRlhbGG/YhJiZWK5YwxjYG
- O0ZAhkXGSxZQVlWmWvZgRmWWav ZwRnWmewaAZoXGiyaQlpX2m2ag1qZGq8axNra2vDbBtsc2z
- LbSNtfG3Vbi5uh27gbzpvk2/tcEdw oXD7cVZxsHILcmZywXMcc3hz03QvdIt053VDdaB1/HZZ
- drZ3E3dwd854K3iJeOd5RXmjegJ6YHq/ ex57fXvcfDx8m3z7fVt9u34bfnx+3H89f55//4Bgg
- MKBI4GFgeeCSYKrgw6DcIPThDaEmYT8hWCF w4YnhouG74dUh7iIHYiBiOaJTImxihaKfIrii0
- iLrowUjHuM4o1Ija+OF45+juWPTY+1kB2QhZDu kVaRv5IokpGS+pNkk82UN5ShlQuVdZXglkq
- WtZcgl4uX95himM6ZOpmmmhKafprrm1ebxJwxnJ+d DJ15neeeVZ7DnzGfoKAPoH2g7KFbocui
- OqKqoxqjiqP6pGqk26VMpbymLqafpxCngqf0qGWo2KlK qbyqL6qiqxWriKv7rG+s461WrcuuP
- 66zryivnbARsIew/LFxseeyXbLTs0mzv7Q2tK21JLWbthK2 ibcBt3m38bhpuOG5WrnSuku6xL
- s+u7e8MLyqvSS9nr4ZvpO/Dr+JwATAf8D6wXbB8cJtwunDZsPi xF/E3MVZxdbGU8bRx07HzMh
- KyMnJR8nGykXKxMtDy8LMQszBzUHNwc5CzsLPQ8/D0ETQxtFH0cjS StLM007T0NRT1NbVWNXb
- 1l7W4tdl1+nYbdjx2XXZ+tp/2wPbiNwO3JPdGd2e3iTeqt8x37fgPuDF 4Uzh0+Ja4uLjauPy5
- HrlAuWL5hPmnOcl56/oOOjC6Uzp1upg6urrdev/7IrtFu2h7izuuO9E79Dw XPDp8XXyAvKP8x
- zzqvQ39MX1U/Xh9m/2/veM+Bv4qvk5+cn6Wfro+3j8CPyZ/Sn9uv5L/tz/bmN1 cnYAAAAAAAA
- EAAAAAAUACgAPABQAGQAeACMAKAAtADIANwA7AEAARQBKAE8AVABZAF4AYwBoAG0A cgB3AHwA
- gQCGAIsAkACVAJoAnwCkAKkArgCyALcAvADBAMYAywDQANUA2gDgAOUA6gDwAPUA+wEB AQcBD
- AESARgBHgElASsBMQE4AT4BRQFLAVIBWQFgAWYBbQF1AXwBgwGKAZIBmQGhAagBsAG4AcAB yA
- HQAdgB4AHpAfEB+gICAgsCFAIcAiUCLgI3AkACSgJTAlwCZgJwAnkCgwKNApcCoQKrArUCvwLK
- AtQC3wLqAvQC/wMKAxUDIAMrAzcDQgNNA1kDZQNwA3wDiAOUA6ADrAO5A8UD0gPeA+sD+AQEBB
- EE HgQsBDkERgRUBGEEbwR8BIoEmASmBLQEwgTRBN8E7gT8BQsFGgUoBTcFRwVWBWUFdAWEBZM
- FowWz BcMF0wXjBfMGAwYUBiQGNQZFBlYGZwZ4BokGmgarBr0GzgbgBvIHAwcVBycHOQdMB14H
- cAeDB5YH qAe7B84H4Qf0CAgIGwgvCEIIVghqCH4IkgimCLoIzgjjCPcJDAkhCTYJSwlgCXUJi
- gmgCbUJywng CfYKDAoiCjkKTwplCnwKkgqpCsAK1wruCwULHQs0C0sLYwt7C5MLqwvDC9sL8w
- wMDCQMPQxWDG4M hwyhDLoM0wztDQYNIA06DVQNbg2IDaINvA3XDfEODA4nDkIOXQ54DpMOrw7
- KDuYPAg8eDzoPVg9y D44Pqw/ID+QQARAeEDsQWBB2EJMQsRDOEOwRChEoEUYRZBGDEaERwBHf
- Ef4SHRI8ElsSehKaErkS 2RL5ExkTORNZE3oTmhO7E9sT/BQdFD4UXxSBFKIUxBTlFQcVKRVLF
- W0VkBWyFdUV9xYaFj0WYBaD FqcWyhbuFxIXNRdZF30XohfGF+oYDxg0GFkYfRijGMgY7RkTGT
- gZXhmEGaoZ0Bn2Gh0aQxpqGpAa txreGwYbLRtUG3wboxvLG/McGxxDHGwclBy9HOYdDh03HWA
- dih2zHd0eBh4wHloehB6uHtgfAx8t H1gfgx+uH9kgBCAwIFsghyCzIN4hCiE3IWMhjyG8Ieki
- FSJCInAinSLKIvgjJSNTI4EjryPdJAwk OiRpJJckxiT1JSQlVCWDJbMl4iYSJkImciajJtMnA
- yc0J2UnlifHJ/goKihbKI0ovijwKSIpVSmH Kbkp7CofKlIqhSq4KusrHitSK4YruivuLCIsVi
- yKLL8s9C0pLV4tky3ILf0uMy5pLp4u1C8LL0Ev dy+uL+QwGzBSMIkwwTD4MTAxZzGfMdcyDzJ
- IMoAyuDLxMyozYzOcM9U0DzRINII0vDT2NTA1ajWl Nd82GjZVNpA2yzcGN0I3fje5N/U4MTht
- OKo45jkjOWA5nTnaOhc6VDqSOs87DTtLO4k7xzwGPEQ8 gzzCPQE9QD1/Pb89/j4+Pn4+vj7+P
- z8/fz/AQABAQUCCQMRBBUFHQYhBykIMQk5CkULTQxZDWEOb Q95EIURlRKhE7EUwRXRFuEX8Rk
- BGhUbKRw5HU0eZR95II0hpSK9I9Uk7SYFJx0oOSlVKm0riSypL cUu4TABMSEyQTNhNIE1oTbF
- N+k5CToxO1U8eT2dPsU/7UEVQj1DZUSRRblG5UgRST1KaUuVTMVN8 U8hUFFRgVK1U+VVGVZJV
- 31YsVnpWx1cUV2JXsFf+WExYmljpWThZhlnVWiRadFrDWxNbY1uyXANc U1yjXPRdRF2VXeZeN
- 16JXtpfLF9+X9BgImB0YMdhGWFsYb9iEmJlYrljDGNgY7RkCGRcZLFlBWVa Za9mBGZZZq9nBG
- daZ7BoBmhcaLJpCWlfabZqDWpkarxrE2tra8NsG2xzbMttI218bdVuLm6HbuBv Om+Tb+1wR3C
- hcPtxVnGwcgtyZnLBcxxzeHPTdC90i3TndUN1oHX8dll2tncTd3B3zngreIl453lF eaN6Anpg
- er97Hnt9e9x8PHybfPt9W327fht+fH7cfz1/nn//gGCAwoEjgYWB54JJgquDDoNwg9OE NoSZh
- PyFYIXDhieGi4bvh1SHuIgdiIGI5olMibGKFop8iuKLSIuujBSMe4zijUiNr44Xjn6O5Y9N j7
- WQHZCFkO6RVpG/kiiSkZL6k2STzZQ3lKGVC5V1leCWSpa1lyCXi5f3mGKYzpk6maaaEpp+muub
- V5vEnDGcn50MnXmd555VnsOfMZ+goA+gfaDsoVuhy6I6oqqjGqOKo/qkaqTbpUylvKYupp+nEK
- eC p/SoZajYqUqpvKovqqKrFauIq/usb6zjrVaty64/rrOvKK+dsBGwh7D8sXGx57JdstOzSbO
- /tDa0 rbUktZu2EraJtwG3ebfxuGm44blaudK6S7rEuz67t7wwvKq9JL2evhm+k78Ov4nABMB/
- wPrBdsHx wm3C6cNmw+LEX8TcxVnF1sZTxtHHTsfMyErIyclHycbKRcrEy0PLwsxCzMHNQc3Bz
- kLOws9Dz8PQ RNDG0UfRyNJK0szTTtPQ1FPU1tVY1dvWXtbi12XX6dht2PHZddn62n/bA9uI3A
- 7ck90Z3Z7eJN6q 3zHft+A+4MXhTOHT4lri4uNq4/LkeuUC5YvmE+ac5yXnr+g46MLpTOnW6mD
- q6ut16//siu0W7aHu LO6470Tv0PBc8OnxdfIC8o/zHPOq9Df0xfVT9eH2b/b+94z4G/iq+Tn5
- yfpZ+uj7ePwI/Jn9Kf26 /kv+3P9uY3VydgAAAAAAAAQAAAAABQAKAA8AFAAZAB4AIwAoAC0AM
- gA3ADsAQABFAEoATwBUAFkA XgBjAGgAbQByAHcAfACBAIYAiwCQAJUAmgCfAKQAqQCuALIAtw
- C8AMEAxgDLANAA1QDaAOAA5QDq APAA9QD7AQEBBwEMARIBGAEeASUBKwExATgBPgFFAUsBUgF
- ZAWABZgFtAXUBfAGDAYoBkgGZAaEB qAGwAbgBwAHIAdAB2AHgAekB8QH6AgICCwIUAhwCJQIu
- AjcCQAJKAlMCXAJmAnACeQKDAo0ClwKh AqsCtQK/AsoC1ALfAuoC9AL/AwoDFQMgAysDNwNCA
- 00DWQNlA3ADfAOIA5QDoAOsA7kDxQPSA94D 6wP4BAQEEQQeBCwEOQRGBFQEYQRvBHwEigSYBK
- YEtATCBNEE3wTuBPwFCwUaBSgFNwVHBVYFZQV0 BYQFkwWjBbMFwwXTBeMF8wYDBhQGJAY1BkU
- GVgZnBngGiQaaBqsGvQbOBuAG8gcDBxUHJwc5B0wH XgdwB4MHlgeoB7sHzgfhB/QICAgbCC8I
- QghWCGoIfgiSCKYIugjOCOMI9wkMCSEJNglLCWAJdQmK CaAJtQnLCeAJ9goMCiIKOQpPCmUKf
- AqSCqkKwArXCu4LBQsdCzQLSwtjC3sLkwurC8ML2wvzDAwM JAw9DFYMbgyHDKEMugzTDO0NBg
- 0gDToNVA1uDYgNog28DdcN8Q4MDicOQg5dDngOkw6vDsoO5g8C Dx4POg9WD3IPjg+rD8gP5BA
- BEB4QOxBYEHYQkxCxEM4Q7BEKESgRRhFkEYMRoRHAEd8R/hIdEjwS WxJ6EpoSuRLZEvkTGRM5
- E1kTehOaE7sT2xP8FB0UPhRfFIEUohTEFOUVBxUpFUsVbRWQFbIV1RX3 FhoWPRZgFoMWpxbKF
- u4XEhc1F1kXfReiF8YX6hgPGDQYWRh9GKMYyBjtGRMZOBleGYQZqhnQGfYa HRpDGmoakBq3Gt
- 4bBhstG1QbfBujG8sb8xwbHEMcbByUHL0c5h0OHTcdYB2KHbMd3R4GHjAeWh6E Hq4e2B8DHy0
- fWB+DH64f2SAEIDAgWyCHILMg3iEKITchYyGPIbwh6SIVIkIicCKdIsoi+CMlI1Mj gSOvI90k
- DCQ6JGkklyTGJPUlJCVUJYMlsyXiJhImQiZyJqMm0ycDJzQnZSeWJ8cn+CgqKFsojSi+ KPApI
- ilVKYcpuSnsKh8qUiqFKrgq6yseK1Irhiu6K+4sIixWLIosvyz0LSktXi2TLcgt/S4zLmku ni
- 7ULwsvQS93L64v5DAbMFIwiTDBMPgxMDFnMZ8x1zIPMkgygDK4MvEzKjNjM5wz1TQPNEg0gjS8
- NPY1MDVqNaU13zYaNlU2kDbLNwY3Qjd+N7k39TgxOG04qjjmOSM5YDmdOdo6FzpUOpI6zzsNO0
- s7 iTvHPAY8RDyDPMI9AT1APX89vz3+Pj4+fj6+Pv4/Pz9/P8BAAEBBQIJAxEEFQUdBiEHKQgx
- CTkKR QtNDFkNYQ5tD3kQhRGVEqETsRTBFdEW4RfxGQEaFRspHDkdTR5lH3kgjSGlIr0j1STtJ
- gUnHSg5K VUqbSuJLKktxS7hMAExITJBM2E0gTWhNsU36TkJOjE7VTx5PZ0+xT/tQRVCPUNlRJ
- FFuUblSBFJP UppS5VMxU3xTyFQUVGBUrVT5VUZVklXfVixWelbHVxRXYlewV/5YTFiaWOlZOF
- mGWdVaJFp0WsNb E1tjW7JcA1xTXKNc9F1EXZVd5l43Xole2l8sX35f0GAiYHRgx2EZYWxhv2I
- SYmViuWMMY2BjtGQI ZFxksWUFZVplr2YEZllmr2cEZ1pnsGgGaFxosmkJaV9ptmoNamRqvGsT
- a2trw2wbbHNsy20jbXxt 1W4ubodu4G86b5Nv7XBHcKFw+3FWcbByC3JmcsFzHHN4c9N0L3SLd
- Od1Q3Wgdfx2WXa2dxN3cHfO eCt4iXjneUV5o3oCemB6v3see3173Hw8fJt8+31bfbt+G358ft
- x/PX+ef/+AYIDCgSOBhYHngkmC q4MOg3CD04Q2hJmE/IVghcOGJ4aLhu+HVIe4iB2IgYjmiUy
- JsYoWinyK4otIi66MFIx7jOKNSI2v jheOfo7lj02PtZAdkIWQ7pFWkb+SKJKRkvqTZJPNlDeU
- oZULlXWV4JZKlrWXIJeLl/eYYpjOmTqZ ppoSmn6a65tXm8ScMZyfnQydeZ3nnlWew58xn6CgD
- 6B9oOyhW6HLojqiqqMao4qj+qRqpNulTKW8 pi6mn6cQp4Kn9KhlqNipSqm8qi+qoqsVq4ir+6
- xvrOOtVq3Lrj+us68or52wEbCHsPyxcbHnsl2y 07NJs7+0NrSttSS1m7YStom3Abd5t/G4abj
- huVq50rpLusS7Pru3vDC8qr0kvZ6+Gb6Tvw6/icAE wH/A+sF2wfHCbcLpw2bD4sRfxNzFWcXW
- xlPG0cdOx8zISsjJyUfJxspFysTLQ8vCzELMwc1BzcHO Qs7Cz0PPw9BE0MbRR9HI0krSzNNO0
- 9DUU9TW1VjV29Ze1uLXZdfp2G3Y8dl12fraf9sD24jcDtyT 3Rndnt4k3qrfMd+34D7gxeFM4d
- PiWuLi42rj8uR65QLli+YT5pznJeev6DjowulM6dbqYOrq63Xr /+yK7Rbtoe4s7rjvRO/Q8Fz
- w6fF18gLyj/Mc86r0N/TF9VP14fZv9v73jPgb+Kr5OfnJ+ln66Pt4 /Aj8mf0p/br+S/7c/25w
- YXJhAAAAAAADAAAAAmZmAADypwAADVkAABPQAAALA3BhcmEAAAAAAAMA AAACZmYAAPKnAAANW
- QAAE9AAAAsDcGFyYQAAAAAAAwAAAAJmZgAA8qcAAA1ZAAAT0AAACwN2Y2d0 AAAAAAAAAAEAAQ
- AAAAAAAAABAAAAAQAAAAAAAAABAAAAAQAAAAAAAAABAABuZGluAAAAAAAAADAA AKPAAABXwAA
- ASsAAAJ5AAAAlQAAAEwAAAFBAAABUQAACMzMAAjMzAAIzM2Rlc2MAAAAAAAAACkNp bmVtYSBI
- RAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA AAAAA
- AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABtbHVjAAAAAAAAAAEAAAAMZW5VUwAAABIAAAAc AE
- MAaQBuAGUAbQBhACAASABEAABtbW9kAAAAAAAABhAAAJIjAgAqqcBCT4AAAAAAAAAAAAAAAAAA
- AAAAdGV4dAAAAABDb3B5cmlnaHQgQXBwbGUsIEluYy4sIDIwMTAA/+EAQEV4aWYAAE1NACoAAA
- AI AAGHaQAEAAAAAQAAABoAAAAAAAKgAgAEAAAAAQAAAoCgAwAEAAAAAQAAAeAAAAAA/9sAQwA
- CAgIC AgECAgICAgICAwMGBAMDAwMHBQUEBggHCAgIBwgICQoNCwkJDAoICAsPCwwNDg4ODgkL
- EBEPDhEN Dg4O/9sAQwECAgIDAwMGBAQGDgkICQ4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4OD
- g4ODg4ODg4O Dg4ODg4ODg4ODg4ODg4O/8AAEQgB4AKAAwEiAAIRAQMRAf/EAB8AAAEFAQEBAQ
- EBAAAAAAAAAAAB AgMEBQYHCAkKC//EALUQAAIBAwMCBAMFBQQEAAABfQECAwAEEQUSITFBBhN
- RYQcicRQygZGhCCNC scEVUtHwJDNicoIJChYXGBkaJSYnKCkqNDU2Nzg5OkNERUZHSElKU1RV
- VldYWVpjZGVmZ2hpanN0 dXZ3eHl6g4SFhoeIiYqSk5SVlpeYmZqio6Slpqeoqaqys7S1tre4u
- brCw8TFxsfIycrS09TV1tfY 2drh4uPk5ebn6Onq8fLz9PX29/j5+v/EAB8BAAMBAQEBAQEBAQ
- EAAAAAAAABAgMEBQYHCAkKC//E ALURAAIBAgQEAwQHBQQEAAECdwABAgMRBAUhMQYSQVEHYXE
- TIjKBCBRCkaGxwQkjM1LwFWJy0QoW JDThJfEXGBkaJicoKSo1Njc4OTpDREVGR0hJSlNUVVZX
- WFlaY2RlZmdoaWpzdHV2d3h5eoKDhIWG h4iJipKTlJWWl5iZmqKjpKWmp6ipqrKztLW2t7i5u
- sLDxMXGx8jJytLT1NXW19jZ2uLj5OXm5+jp 6vLz9PX29/j5+v/aAAwDAQACEQMRAD8A/Ie6uZ
- LZoIYXSSFWZhjkgn1NZS7Y4I4yQS5woHUnP+Na MsLq7qIXfaMN22nrTWmbzIWeOMzBmckLxk+
- gr8+hZLRH9Z18KqM3KhBK/wB339SvFbQrNGluD50T ESbj96QHjHHT29qeG3hT5bIQ/QnoQc5+
- lLayrFJuBUqSWb1PsD606OJriMIiFwpLZA6nv9aqTd9T PDQlBuNlyteX5l3yTvExmWMFjGjbS
- wORknionheOeFHDNHHGVyvBIPpke9SWrJJp7XJjdCj7QrHI DdTx79qhfe8yF43TswJ556CsVe
- 7TOujOMk5T1b2a0/4P4knkhYYpX+WM5wpPLDtiro+S2AEqMpcP tJ5OOPTjipZGtBYxwLcKzxK
- FZjkhjnkj0A6VXnaFEa3KjeAfmLHll5OPwrJSctzahOM1zTXlr26f 0hk8iCYCVGcRqVBU4ySe
- vTp2pYrkRWwhijf/AFo+Y/wDqc+vNVgPtE8QY5LMNuOy46/gakjAdfOa VDGSRu7buw/OrcY2s
- zOc6DlyzfL2/r/gFuaWzkib93I85kZyScDnv+lRbBFYi6iuVE44ZCvKhuo+ tU3SYNC/lERwhV
- nXbycnOKfI5iu1yI2Z3JAK5XCn0pKFtEzCbpxg47rzas/LYfbWtotvHbxyOpLk M3ufT8KhY2q
- zpGqzOrEklTyxHA7dKgaaJpt5+RnLeWoHOewq8lqy6bFNNxuAxn+PP8Q9hVyvF+89 yeWMZKHM
- 9dkun43IzEtveq8b5IXdDIP4hjmlDKvlvETvRQoOPXmmTpDC6qsglOSAy5wABn9al8pX 2BiI/
- l3Kw/rS0aTZ2wjGcWra9dOnn1J2kkmuVeY7nmYvImMY/uipmVv7QYbvNlOQcDA9DT2SBZys gE
- gLsQ2/G4AduDVgNbLIk+/y1VVDRHJZuOSD+VYOVtkd9GEqauruJUt3QXscZieVHbaXQ42rj5jU
- gtiqQBra4WJUZt2RwW/+tUhVTEv2dgiIjIzH+Jc/zzUKwSN8gcyHesZBORuOTxSb1vsTWu5c83
- Ze fT0CK2NytuFOCAoYMOpJOP5VVuYLeLWY4nR/JbkgnlsjqPbNaEUUn9pBY5Nq43ynHCsoPFU
- mE11J GDJHLM5CxqUx26dKqMnzb6HHUjJVZRcvdtuPsZrpLee7eMwCOQ7kdeST0I9qqJMG0Z1U
- Au0gOQeB k/NT4nJmW3uGZU34I6cY6n2FVot88oQwi3UDa7MOhxnn61ooq7bOCNOFObdR6rrbd
- FrdNDFJCu4h ZMwZHQdzUxluC5ijmiETHccjOCKbaW1ybR2mjwu7y3DDLKx6CoYU3/ZpJy0aJE
- yOgX5icnBz071L 5Xc7FHC1YP2cb37atefYltoZbefa0sKJJnJwTkZ+lOhSe1v7a4BDRupywXj
- nI2/U0yGTyFikJVZj 91X+br+FaG95VjXI/wBWcpj/AD9aU+a77M6amAnUklF+71v0+5Ic+nH7
- RB5S/Zl+64kOTHxwD79a 6OGOKz0xGRxFLCqou/n5S2effr+dV0s2uAkCy+fEBucr958D7wPoK
- fLJcqvmQ2zpmUK5lAYFuAhH tjPFeXVqOpaNzzZJSfspPZ+Vv839xZuLO2llZyfMZzJsVTggdS
- D796ykkjjihZIZVlaMbGJ4bDcM B6ZrQeeSO2l+0OivDdFXkC4BYentg9Kz/wB297AY3zuG2Ln
- gj2+nWlSUuXXYKUavspKb93p2fr0L Cwtc6oZWmj+3RAxt8vBPVjiqAkjltpc3cCJJIX2Eck5+
- U5rcmtjLB5UkkeXk3l4/lOPT61j29hbq zxoj4VsBmbIUkH5frmnTnFq7exyuLk71Hr2026a/8
- Ay/7PimJlW4WUcg7ONp7VlH7WI0AiIWJGXd jtnk/rWtdpcNNbm3QwTR4S6VunqPyNVru5JulJ
- ZZHYnzggwGbjkDsK9KlOXr+hrTryvda22XbzMq 4tJmuEKkxwqWXLdx1p/2SNba0eeUwPKDIwY
- ZJIJAIx2qaRNt8n2hxIkjE7o+MnNSfZX+feduyXY5 bqpPQfSt3UdlqOqubXn19PyK8kCKiyS3
- MbTkEjjAb1xSwRSecs0Ubl0BGBztLcc1ry2dssrm7R2O wl1XA2noO3bvToVaDT51j8uSZ5Rhg
- Oox1+lZe393+rDlUUYuPK5X6tW+/cyhCfICSyxxXG7ALZww xx09ahgjK2oiDrEHO9lYcjjH8h
- V0Q3Hn7Agk3bXDAds4yKlubf8As+/iJQXAZHVW7Nxww9uf0q/a K9r6nXSjTU4xb5uy/rVGZHA
- Yrf7xYMwKsBxjHI+tWHUNbKwR/KVgScEjd9alhVo7dI5M+Wq8ZH+e tRT53feI3HLIBgLVczcj
- veHlBLmh+bFgsYLeKMl8ySSZCg8px1NSwzz2lzHInD71/h9P/r0yN3ku HVG3SEEhcdcelVgGG
- wZYZOWbr+P0FJxcr8zuYuh7kqcndPtbT5WNGa7+03Jk2vnYRIw6Z6k1qxXl ta2kA2s/7pmyD9
- 0HsfcnpXNWl00drJtVJfMcktjtjFSi6aRSoYDAC4C84/8ArVnPD306Iy9gqqip ydlt/wAAvLe
- 2UlpHbLBLaOxG/wAxs/vAeD9KoXN5IU8yUhJ5Zw7nH3iARx+lN2ESxCSaPC8hthO7 3HGe2Kpm
- G4nMD3JO4hmjAGMDkf0rSFKCdzjlhnKahZv1/r8CHYgiKMwUkbhk+nUVYxJcQqszIRt+ cKOuD
- nj6YFPjhdrYZXcqMAOmcH39KU2zGVUE0RILK3BwR3IOOlbOSNZS5VZp66WV/wAfIqBQqBYZ Bh
- cgMeg3dqgjU21itvHIAryF5UIyRxgc47DmtGG3fjy9jKcn7oIB/KoVhgMNtGyuk4Y5JP3ge9Vz
- o4sRgnzRm4Wts1/wxLGyyBlk3Pt3AFeAw/hxUi3ziwt1kGXRtkQAxlfX8DVUKxjjEP7xdwDYHQ
- DP tShVLlQ4TnKp/Fn/AA71DhF7lV6PtUnNv1/y2LNy141xG926uWRmygxnnBNRW8+9nTzRCjF
- d4cZy vfFSSTb512QuVOflYA5AHHQVGro+2d1j2wZAAQAMzDAz6ikl7trDqKcafs6b/wDbX92o
- 15GMwC4E THAyONo6dqRY7aOSBo4pZCrBmAPQjmm7oxb2ykFZo1KsWOQ2TRE4K7MFFZwVJ64HX
- 86trQmUE4Wd /nv+RHOZGYhV3KuD7nnINJCwi/5dJ0yCSSQce9OdpQ2duOPm/OpCkbuzM7uEHB
- U8PV30sdqT9ro3 fzt+bRRfyxlllUlgMYHfpUaOzop5aUZ3Nnj2GKsSqYn8wxOQBlRxxjHP51U
- SR1kDKm4nOSBxn/Jr WOqPKrt+2tK6S7X1JXvGjgCyfvS45UDsTTPs6i7kiRAPLyrHOee/8qHT
- McYuYsbV5IH3v88U64ig Z90TlCARjPfvmmrLYhqrUfNUs1HZPf7yqi+ZIFOYgSCM85wDVyNFL
- EBxtHK5bt6VT8lpJ1VXwcjy lOeR9anWSObISKUuQRtUe9VPUww1aCk1PRrXV7r8iUw70kYXCg
- LtzgdyOn1qtHGk1mJVnHmqzOR7 4HFNDq4fZG9uORhzktgf/rqHygVRh+6KtxnvTjF9zPFS9q0
- 4bLXrr95LGt9IV/eRKVBLHb0TuPr1 qw1zI4kXzV2bsKSOoqKNlAPy/KxYg+uOMVZtI2kMqYjG
- Yy4LDjCg1M2lq0Vh7r3ud2e+rexBHKWU +XukPm7sg8DHb+daUkMbM2Uk4J8rLDG09KywrySwF
- nS3TPI2/eXv071LFvGNisMhiwkOfWlKPZmt CX71Jpv8n8tSREkS2FvwIgwLE8jpx/M08WpQEx
- sWIX5lU4NNijaWBsFiv8BA68frWgIklbyw5UMh ZnXtjms5TsddGjR9nJvRLs9eu1inLKGg/eb
- ywA4H5imvPMTkElWAx8vbuakVf3JlVMZkAOQG6DIP Paoo3Kyp5hcbpRuKgfdOeOnftQrEzxk1
- Fzd7eW5RnEk9wjZSELgNuHOMVLC72+/5FkweWHY4/wDr 1OYraR5GxK64BHPTsM/lVa6WMzliG
- A6nB6mt4yTtFo5VGUE6yer8/wDgWR0NzDKtzEqGRwpwxwQP XB9+aHTY0KSJISI/lAx8oOTz61
- El6r7Y2lDGVioYEgA+/HepfJDSKiv5jlQWYnjODwP89q4NVa57 6rQdTmVtfWy+8diFLfeqhxt
- 4PbJ6dqVHk8uOFoyjpG5XaMFgOpqv5DmGN5oXiMZ5wcZ9PypIJriJ ot77pIQyL8vr1BoautCa
- 1adRrkitNr7E7TPBB5UaFsKAXA+UKcEk/wAs1I6tPcbIgCXOTg8rjnB/ lVRblJIiSRgHaDnrU
- sckksJkASEliNpGTgd6ORo3pQp83uy37L+vzNK9juJIBcT2wjhdyVUIARk8 /liqUELXNwsfll
- XIZyS2dzf0yMUhlaS4WNlYnJ3dunNX4ZYvPE/ktHvJJZSAuPb2rF3hGxzV5+x0 STfSysvu/wC
- CQ+ZNBIj20QjVkztdQSTjBwfQVloI5ImcYIU4JU4y3Y1cG1PKywnjdt5APLYP6CnG ONt6b4kQ
- kyBVXb8x4A6dquNkVSpxcuZQWu7/AK1M+B7q2X7KXzFJMWm3c845Ge3FTsXi1AGNcsSf LDYJV
- MdD7+9TvZPDLEZhmZVcMo6Agf8A16bYgjLNLF9oKBF3rwQepH04FW5Ra5kccYJNuO3boIUk vg
- w8qFHV1KsqAYAXrVwPc/ZYkkClPLPlnbxyen1yMCoRtW6RpWAZNygKcZLcDPtV8xxxXgidyypG
- RIQM4bnGOOnNYyltobYOnL23M1a22jbX5FORrpWe2WEcnJURgsoC884qSK2d4kkDAITsBJ/i7D
- 8a sl7gRIU2RKdqNLt5Ax606dY0+0I86zmObKmMYDLtwMD1zUc72R6MvbRqfu1pfXTX9EPSxii
- nME4k bGWV93+qA+8G9TUc5R4cpaMkZbdvzznHH4e1SmeGWO3ktzMJlUocnOSRj9RmhIJjLbxB
- dmEypccL z0ask3e8maU1Je/zX8u3ys/1JnNx9ntwyxw7oecxjghsntUgKyXKvPewIjOXGFwdw
- 6HgdKinmSRp VEbInneZuJ3bTjGD7e1Z7/Z4SHjDuq4G1m5G7t9RSjDmW1mbVKEpU/fVn3snp5
- XW/wAydFE9/JK8 6Rxl9sjZwCSDg1lMpjjQEPxyDu5HpV2DdLbPabdpVvMXPXj1PeoI2228hIP
- mAEdM7hnlvoK6IaNn LOkpTbe9vNW83p1IFKjzGZwjhsktzk/lW5pywSwj7QQ+3fFIqcHcT1/C
- qyfYjGY8CSUuQX7NjnOK nfLmJ1gKOYZDKqjq5Pt+FZVpc6tax5uIpyxElGKs++zGuzCaexBMo
- 3bQUGOR3oWKI2DCPDO21pSf +WbZIx9O9VYYdluJhKHCuoZQTknB4q99me60yN4X8yRVyFQYJX
- uT64pStHqei6c6XKtEu67+Y2K3 SV900tuAgYhiv3wDgMParM9my2SzC6tXULtXYuCecfzqtBD
- G9ypuIplZZQqIMDcCM+noK6KOH7Pe Szq8XltIpEbrnC45H171jWquL0ZvUquDvGV/u1/yKMMN
- 1azxMtzFG03zcj7pBxt/HNasst4yCBQj G3baAF5YZBP49QKrTOJHjYSRbURgeOpIzkfSnhkaz
- WWK5V5JkJi6/dHGT9O9cc25WbX4f10OOpyy kpVlr3tp82tSW6EE1v5e8O+M+XnlvmyxPuBUds
- PL1k3gh3RIAsW0Arz/APXNR2k8rt5DRb5UnOJV XCuCOCPbrSztLZeW/mqwRHUccHBBB/Wps17
- ncy9rzxdBR+L8fQS5upbbUSkzoyIQ2AuCxX0P1I/K oIpDqdvcTMjIyEoqx4G8sfvfnUpSwYXT
- zzK1usoUEtk54IGf50lxcSWz+dCY3G2RWVEwMn+orRJa KK17luMI1oRowfP+H4GVqG+P7RF5i
- J5LxhmP8TYBPP61nS8Xlw8ULOfMbew5HzDqPQA0s7b4YYQD vjTLHOS/vVgRTNHZKh+0bY9s3l
- 9ixyM/hXfFckVcqvGVGznBJ32X6/8ABK1rYx3tpGEbLNCzKpOS WBxke1MeEQrFGRLGshP2jzD
- nJ42kelalvttryDzJY8GIlAo25HO7FVkjEbRyRMXAUlt3OHHQUe1k 5Pt/X9epxc85y5VK3a2m
- vqI00bM0cwIDL5cbt/c759TnnNZCRrMojMpEqNjcCQCfUe2BXRefI1rF LNZ8gFHZkADMeePSs
- xLfc6K1tIZBtDhBggnOCaqlOyf+ZrSUuTnlt6poIQLQh4tw3gsm9s47gfh/ Wm/apLlvNkgkmX
- cF+X+DIwR+Oafd2c8W+JQxij27mPc4zkegqrHPcJeBomBmfd8qLxnHp0+lXFKS 5luUuRw56du
- bv/W3yBpm8+EzJHt2FQMY3EAqPyphWNre1aMOx2ESknOSOc1OJw6mLCTwsrNlV+YB R1B7DOfy
- qmsrSBBEqqWLKBjO4AcmtYp+h00qsZSUpOz+eo1EM0wliO11IfIHXB/lVpHS8nkkZ0Tc TuCr0
- P5YxTJBHFEoCSxuQAVJ7cEGpfswM7bWUBHCsw4BJ5z9M0SaeprSgm+ZRs+uln+oxRmLyYzH Fu
- kHUckjsPwqeSJ7edQI0CliXjIBZCOACa0bW1hlTzDFIH8wsqhumDwPzp32JZHmVRJGWcNIHfJB
- 6msHWXNqW051GmtF0M1YVmslQQyi6DYBByOegxVEFTkfPmMlFDHpnk1uJauLjz7aGZoSGaIk9N
- 3A ye+Kz1GYAtrt2jmZ25GeRx6Zq4VEOKjBpL/N/wDAI4EaZZUWGRwSASDwuPWldpRegSRxwsm
- UbKjO WGM8DoK0vLZ77yYysUjvsUjox/x60kihYZI2ZfNaUm3BGS6fxNn8OM1HtE2YVKkZ1OS/
- 5XXmRyxM LY28c9lIkRysoj+9gc9u5rF8qdbjdMgZt2WCDGwZ5B44NakqolqwKywxcmFnPLEe/
- pUDKqR2srF2 Mineu7lZBxg/zq6bsvU56kacIr3t9NLfjYqrLBDc4kXaG37EHHTpmnW5865hia
- AEODkouCCRgc1M scTQOrgxyrMIzvGSCTyfpTolKSPFGrozNuScnC7VB56d6ttWfc5+Sm4Nyf3
- 30/ryC0+zlhBNG5Ql dpDfMAMhuf1qGe0iacFAUhZG8s5zkL/F9DVqRoRLbuCJ0jiKDbxtPO4H
- jqM06O2As0kadCVUDC9W UAk49OtSp2fN3KhGEZqb+F6aa39Oxm29qGuIlwFG3fhurY6D8aI7J
- pLnzImSBgzbUc5IHbPFXGtT PbJNbF5WwPlXkg4yf0qncvvKJCSY+gI6kdQa1UnJ6MJ041Obkf
- y/4BXFv5bKHky8ZJAYE8f3SO5N SYkjjjWMKu6PIUg5FSwzCGaS5kiWYyHaoZchgeC34UnS+cS
- MDHl48hcDbj7w9hVOTvqHtYU3JxVn bp/V/uZlssghVJ2EkpUkOnAx6fhVeWWTciIhcZ+bA6L3
- q7HdvLAZEaEBSYz8mcj/AOvUjwB7IzGS N5A4UxgEELjmujm5X7yOOUeaFqcnZ9df+D97KE8v7
- xkSVX3NtVj0bHpVULJAuwsMMTnIzmtjykct EHhyFLA7euBn9azpYj5ce4kblDDn3q6c1sY14O
- Tb3a+RXkaeZoyVyBzlONoz/iKcYZ3BPCHO4np3 qd/JiLhldweFC1Ahjhk2K2/erKMnP860Um1
- ojlqUlGTcndvfX7ug2VIphGGkMuwll2tjk9c+tW4Y vMhYBh8rZJBz26VF9mSLT7Zt8UiHPCDB
- B9+KWCJxs8ncAPv5PXmlKScdGRg1FyUuT1SJDGHk2ylU VQNuxeTnmlW12tF98jblSPTPf8asG
- Jhcuj7mKnBKjgelapvJpLe3VPs6rtZB+7+9g5z+PQVzzqyV rHdOmlUXu3bfUznjUlhOhV1lCK
- i8E8ZOP0pZ7URyLHcDEinDkZGD17dulaNtaRbC9zOvLqSmfmQ5 7n+dTXEMPnrsdxIzl1aTlWQ
- cbunTNYe3tKxq5e+1O935afO/+Rl7pgv7t4oI1m+8yjAY9R0qREba 77CpmkZlz2Vei/nQ00Vt
- cFFT7SN3Ddsdc89aspELmWLadiJueUsPXgKPc05O2ttDGWHSrKpb3eun 5ozZo7iWaMzZQyP5g
- TGOvf6VApVQxjjJ3Nkkjg4PUe1aSxzpOjeW6FZdnzgHB9KrbUQsSDGM/Ip5 IBJ4rSMtLG1ClD
- mfs3p8tBDG+2TEbRHuOuDUE1tsQCSJsYLo2efx9avWscb2SyPMytliw9QOg/Go 3s5nijZSXVQ
- Bx2B6/lTjNX3DmpV1zKO3Vr8hwsxa+Vb+X+82szHHUjn+Rp0Mkf7sCKWTcAF57k1X GJCsiyyP
- s4Bzzk9/oKuRguD55VG88YKJtAU8Htg84pS2u2drxCov91H3enW3qKuILh4Rwwcx4zn5 h0P0N
- QxnbCfOkCS+crDKkhgBgnilkSWOTZhWfqWxwcUwHfPE0iYc/KBjoewI9am10XK7ilzbbdEP Eb
- GPcDCEX73y+/H86mDBbxVSSFWdiqq6574xVJFLkAKyEA8N3NXI4h9kb7s2clW9MUS03NFzTg4x
- W6/rqh8cjtG9w4GwuwXjsOM/nU0RcwDJUoGAQkcKCOatgFLTyDENzMNy4yBn+Hp1NQPH5byBV8
- r5 tzK3X0ArDmudOFjiPdVTp16/cCRqVaQSBpY1JxtwB+GKZEF+/JltykjAAq7bxKw2yRSO2AZ
- Npxnn oKkEcflwq1tOjMhYjd0yeD9MVDnZtFRTjUd2/XS/5lCJNojdMMPLLEZJyc4x+NaElnY7
- baMLKrYK ctyG6kfWrUdvBJNFMFEEEbEAuf8AWcHDD29qYGFsSdyNKiqN7cq2T6euM81hKq5PQ
- 83EU6lad4dO 2n5f8Eo2FgzXzbwBhfmLj7uc8mrcFrHHO8UiSuyx7Btb7xIJ3fnV9JSzXaxMqq
- kvljIyWUnAOfal jkt5r3EkcskuDsZeA+DgYqJ1ptts7E6nvSinGy/pt6fmOtDbRrHHbwvPcOo
- ZyxDBCvVSPXrReQKs LxxWbKJvmQ8bmAOBzWeUul1W3iWCSVcsrNEMc9zn2PH4VoPcXtldW7yR
- 73+ZYlP8Kkc5+hNZSi1J OLu35nL7KSq88Xd+r/zKB0+G3+zuZ/tEkUg3JESCdpyT9Of0qS4Fx
- cWqzzI8aJuO9OA+48H6cUye bz7iLYzCRR5hC85Zf6Y6itOSLzdPee4BVS22NAcAh8H+daSk4u
- LluelSw0qVSFSUU9db7/cY6yKt oDNBKT5jBwpALcDJ6fSniNGkSD7K6xjO1jjI6ZJ9eKrJbTT
- 3CRKQZmIyp/vdMfXirNxHMk1qof5j HLKMevdffgVrKydk9R4vmg1FSu797WMmRoortpEDBVOI
- wW556Zx1pbaB5YHlkkjk8qUZSMYJGMkV YkkQwruiGXG5D/snGPxFaGnxOlmZBEZ0dyx2d/X+l
- aTqcsLnHWnKa5pRtfTff8SvC/kyLIjQxM8L ja65xnv/ACqJpWlhkBjZrltgG09Ox/E1It2SYL
- nckDueY2XJBAwOx4rQlhkklmR0E4wG8yEAfdAI 9KycuV3aNVUSp891213/ACV/vJVkijS4Rts
- eydMggHqMUxYxa3H+onit5MpG5bgg9R+dUrBZL9Vj YeQxYtM7/wARwSD+VbxtLifTEjVPMR8y
- Y7gcEL9a5qjVOVm/U8+M6cJW773/ADRAwu54t62xT5i3 QZQ4wAfc4qQXrLoO+6s5VYS8oxG7A
- x8307U4JJHDClw/Mo3SFeNhB4U+5pt0Ymt7m4hhuEkBIdZH zsyRgfXqax0k0rdTohCNRwUYvf
- 0/G4ly9u9wXhjeZH3bERudoNZNoXN0yMjOoDMqqcFV6gfh39a0 rnU7FLOaILvZ5FYMnGD6fTi
- qktyHnit3j3ZDI+zAPJBz9Pat6Sko2sd1KVRx9+DS23ennb/hgmvF GqfbIZ0MbfKoU8Zx0qe3
- gi8pmuvNmRGCvl/uE8bT7nrWWI0lnWHymOWEhKnA+XI4+tSwec8zR78v PiWRD1Vl6fpWkoaWT
- tb+v8yMRTlVk9bNL00JLmSG3ln8sLHtZYyj/N1HX61SuHAiSOJZkaNsYds4 9j645zTZZJZZ5G
- nCrlt2x1+9kcEVIyxy2cM4LjHE565Yenpwa1jHlSudCp8yi5X8rf0iO3hCSs15 kxE8Opxux6H
- 0qe3ke10+OdyJY2I+WM4O7ORz9K2ATdQRgWrS2rRF1IwNgHG0++ax3hmMiCZY7WIE Id4OMseP
- xrNVOdvmOJ0/aybqvRb69P8APuMuLiV5YTcSQGRZnkwkeCORgVblZ5ba6khXlZywKL6j PT86W
- C2tlupYZ7232wvtEjKTuBBzViDy7ZYonu4XiaEMyDruGcc/SpnKK+FbeXzMXRo0rcsL211T e/
- 3GNMjzW65L2ylvMBckggYGPrzmrVvZpuVJXklaQMAyn7/Pyke1PuLW4khgRTstjbGQOwzu29cf
- jUTXoSM3USZZcRkHoCw4I/WtOaTjaLIqxqxpr2Oj9dCdhZxRSW4eUMXVWd2yNxXg/QY6Vlrata
- Sx BicSvlZB1QKCcH3NLJK0izxFMSkneCOm3p+OKrO0kiS+Y7NGzq+M9DjitadOSW5UcLNpcrs
- 3vcWd 43uifOjiHllAuMdV9h602PdbWFsgik2BS27HO4n1pscoLtsWN15Xdtz+PIqaKCN5UVhI
- wPLKD0Pr 9K2doqzOicFTiqmjXp/X3jVndoyskamQ42uUyWXByRx0FXQ7xg7fuugVNycMT3Hqa
- q2bBp4w8Duw RxheORyD9ParttNvKbpYoV4Rd4zgnms6mmyN4zVKm1ZW31T/ACLtttggZQDdN5
- 5QtGcYHGPxp9xD Ld3riNDahGJZnPQAdD75/nVRkQ5kil+RZCWjz8xx1/Si2uJWlLgtHbysXO8
- 5JA4PNc3K78y3HRjF XlGV/v8AwRfSW4tYFkKfIZFj24+4CCSakCRQvLJK8NwrcKsa7cjsaZNe
- faLVXe3fyVdsHONw29fz xinWcMQS28yeOFBGpbzed3JJxWT+G70NVFSpubXrotV2Wl7kE8UbJ
- bhJogxYs3tjk/jWbc4F0rvE YoJmYoSM4A6jNaE0LER3EAZvNHmtJwVB3YwKzpXnN1biVlABBG
- 5eAGPP61vR8mKlF04t0no/v/D/ AIJbmu3jVRHGkkboz/MARzxx6VmWYW1uobggSxpzIeoLNxj
- nvin3EKeepEu7HIweOPb0J7VJHujt Y55SBM6syIF4PbOOnrWiilCy6nFiMDCUbKHxfL7+q9SO
- Wa3MyKqM8bI24Dgk5+XJ9afBbO8byS3N vACQ4BzlgDxj27VJbi0EEZO5H+z7SDnJPI3Z9Paqp
- aWRkhDBV+X5m7EDHp/nNUr7LQinQai7tqK7 q/5ofPIkjj5GQzHz3YdAewwB0pkJuJJGtzJGq4
- BGV64BPHFX/IWXUCZImjhcuzRg4aIgYwT7dadH BFbtaTupEUkBIlBOOuMn6modSKjYxq1MOo8
- kdX0/q1yhNPch42lUQu43rsXbjPGeO2KS1ma21pXV YbhWOxiEGD2B56cVchghclppGjBIUM2T
- 8g+8PqT0qNp7Z2eFYtkLHzowPvDH8JPfvT5k04pDqqMq UowpuzWr2+6+7KbkLem3FuyKXH2dm
- wcIMk/XNMktLq5d5ZlW3t5SSHcYAHccVoMsFzcREsyokLBC OPlJ6/zqt9octDGq4VY2UFjkMS
- fl4qoyl0WpwRoVXdKNlbqv6X5lX7DbWV1Bu8uRWi37kHynAOeK ijkY2ashtk3MA6NHkn1q2Fm
- iKPcwbT9x+ANue5+nFUp9n2sx+YGXcQJVGAcDrW0W5PV3G8LHnfJK +nay/wAiG4jS0uh5DLNb
- fOiPjPbIyfxqmsiGHe0bsVAR+RgN+XpV0pF9iRFjYmRAyHcBwCQfx460 x4UVy4CoAwkwec4re
- LVrMKeGrqnZO6XyRXeGH96RMGBm2x4GT04H4/0qiI4hp7+aU84TqyHB+YYw fpg1dVF8yFzllQ
- ZIHBPOalMa/ahmFmLI+fqe9aqbiclfB1JQbntHW/f16fkZkRCiIRE7Sh68jrkm pLZJJt0bzxx
- sCWBI4Ix0+uasMR5sEMZSJCn7xmHK56j86cUnhvCjLG4VsEoNvaqlK5goR9ooRbT8 l/wdSdXl
- +x7/ACnhm37SsmDle/Hr05q7bWjFyTE/l5Ii56joaqWMc00u1tsKLJ8zscgAjgfU461t W1xFH
- YC3jVpy25vLJ+YZGOD7d64603HRHXepCCs25d30XfyEFxPbrFAPIjiKjLvHuJ/rUy3EYhlL we
- cUxGj5wFwM7envmp5tPmexEm1vPQLCseMlivVvpioV024W4khCPKMBl29/b6/4Vyc1KSvc7I08
- O6ammpd9bfcKytNZmRbeABJAOIxl1xlselI1tcNpweOHKSMXJUY2gfdB+lVIrm9hlPAYFhlNvc
- 5A q0t3cnfD5oCGZcE9MHjP0puM47WMJYerSkpU0rLXr+WpmzWrRXcQEgO+RSVHJJA61DIWm8r
- DRhWT Ifb05/xq/cLAmqrcCYqYZwqjOQwYcEVmxb0CJcIxw20qDgqBnNdMJNq50Rr1IzlHldml
- stf8iqlr FHbDe0jyIxKhScEHv+FXbVCjw3QYtFHG3G772Ov86r2sbyPCkTAOww5c8dTz06dK0
- ZbWQsIVt5pG SIquw8A9wfU1pVnrZsxn7Hk5NEuqf+ZmrFiRY4AQuCXyMnHer0i5ulitz54ky4
- CjJUYzg+pqOWZ0 fypRFHKVwy7cFjknjHTjFTo0wthdTR+fHIefKAU+gYH0HpUyk9zor15U2nH
- 7v+B/lYosS37wMVXh T6564qxPsnufM3BXaQnZ0wVGcj2qEmN5dkYMg35ZwcA46HFXkgWRSYla
- 6lYCTKDhSeCKJStZm8W5 JSk7K+t9iASpHeCbiXcCxwf4j0q3b/Z4woLuWYlXRV5z6gkYxUyQw
- vMYGi8s7dysewHU1ONOhawt Z3k8xtzPiPgt789qwnUhszSdoVUotpvt/nqWRa+fdokaSGFYjt
- bPLvjg5qt5ZXTp5HurdyZAsigZ bd+XFaC+ZFe2rElIS2+MZ67Rjd9PWqmTc3DbEAj3h5ZgPkL
- Lnbx79K5oyd/I6ZVZqs/eVl2/zf6E cBuIzEBC5lAV3XuoU5z+VaNwGup3nWeJpGLNEEGPkBya
- maZPtab2Te4cttX/AFfy8qfx6VWkDTab GwuIpHjiSIBE27cnv7mo5uaSdrDoRU5+05Xrpqrp/
- gkMjgDOkiSQ4Me5sDoD2qBZp4oZETyywkCb SgY8DIrVYWK3lxHuMKCQLhj904x+XWksYrO3WQ
- efHclGyNvYYPB9z2pOro21+BWKxk+b3o38mv8A gfqVI5pDf77qaKOLyysi7QCWYZA4HXOKhUb
- bS3EJRpFQqx287geRn8aFjtpLfaI5nkfawTdkrg42 n3pLdZf7Ut4A8QdnLkEfcGeVPvWmiuwl
- WjD95JWXbYm868ljg25LoqeWijBbrg+/vVeYyXFgZpLh ZJEm3LjPzgDqPYGtGa5kN3tAS3ihl
- LAleYgBwGPfJ6VWia3lkaNgBdyzeai4yBx93HvUxl1t/X9f 1sc8KylO7gl/XdrYRbea8ZmXy3
- dnDZVcZ9SPYirElpI9zPbDd8qlzluM8FcVXGousKskP2aRiFO4 cBTxVkCO1u7ppneVi4SMoSN
- ydz+FJ86f5HpUZSbbS9F/W5Xulijt0Rfmnum83cp+6y9vx5xWWl4X hJBH+sznuOxH5VrXIdrp
- BbxO8ZbIfr+I9KorOXhUpBHAoZ1eMqMtwMHOPXP5VrT+HVXJ9k5Plcb/ AC1CztlfUBEWKQ5Yu
- SOVUcitezjlRdkcTxxSDc8h+7uHQj0HtVG2aOQB7iKSRiQp8ttuT2P09aZP PLatJahZfJLqW3
- NnGPQ+45qailN8o8TRdWTj0SW9/wAFfX56Gnd21rCYhK0c7spYqnGQOSRUD2sR kSaKK4ZZUDC
- MSc9CDz7danZ7K7WRo3y7ThQuecY6j0FRW0Zee1iZnclBllYjGScj8MVzxclHVs8h OolJp7fL
- 8CS3keA28cKpKoUHIHbsf51pyTyXEf7wtFG0hwqfL9Bx6077IZ5GiVT5bEspXgsp7j2F Qm1Js
- 4Wk3PCV2OQfXowz71zylCTv1Ob2dPmVRuz9LsqwRPE0Jud1urxMVaVsg7Tiq7yTTTIZZEb+ Jw
- owCemPwrRFpKkCR3SSXHlSbZcHgDjP07VJcJFKyiGNvJBYNg8s7Hgj2q1VXNc6PauT1fz0sUpI
- 55A0VrBBKBIfnMYO3GMZPr1ouLeQx3O+DzFSRQ5jUAn3FWU0poneRJZYvKVo5kLnl+uar2yoE8
- wX gZ1QxyR5J3l+jD2FOM1unt5GuHc4vmUrpdtPxK0lvLJAq2s8AVVwqkfMFA3Ek49aaIJbiyt
- naWJb jDKSOM7uc/0p8HnWt5aoJFCMuZWcZCODyD9c1anJniMUqCNDIxhIABJxx+FaOTTSHLm+
- sR5tt7+v y1+8zLSwWUW0ckUjyNESvzdgTk/hU8kUK3EkYAjfysBmOVyR6Y6mrP2NURRcTlZUT
- CsrEDb/ABVk 3apJFHIgZVX5QSc7we9aRk5y3PXwik5Pm1vtuv1sWDcPAYltn+7DtcY4ORnI+l
- NWbzdQsjdobiIo wQJxuJHDfnVY5t7oeW3mqA4YnkEAVJFHvWLfLGSsY2Y4IXOf0rRwja4sRha
- c73VvO2v4ECRRrbJE 0qK4ABDDnOSTz9KtzLasqeTnzZZPMXLZEZA4U/UZNO+zRuiXcJQxrPsc
- MMnpgn6c1QjJ+1IwSSSJ GYqQCASM4Ge/pRfm1T2PMrynOFqU9F6flpcspLNLexwREh/LZEjbn
- aoOTn8Kbv0+2tmWVtrSTAWw Y5JGM/j9e1UFb/SbabftZlLMcnII6A0x1aa4gLI1xDFExDKMYP
- OOSK1dJXtfQxxmHnN8t/d06fkN 2oXSRkcjAJAOCRnnJqKYASptGwIpXkfe96crPJDEpI2rEck
- d25xUs7eYib0L74wcJ25/xrdaPU9K Eabp2m7P+vuIkSXzG2hQ23cPlGOnIx71YRg0EnnREfON
- p9BjoapwFn8xmjliZSVYk9M9varMtzMU ghVMrIQM7fTvRJO5FoJxqJadO463kmiYTKECsD8+w
- cHHAzSfafNtcCFWllbexAA2npxTB5scLjIK LJ6ZAB/rV5/tEVuDFFEsYLbiEHBGOPxrOVr3tq
- czd6ilKKTv94uniJtTEbqYxjBU9TgZzmtWJpks 7dBp8kplXOQAAOTuFUreS8lkTbAJG8t1d1j
- 6hhwfwrW+zXIs83TMYfKby9nBU4wM1x15Lm1NK8m6 nv2T7f8ADMryNO8u0252LuCxgcsRgKfp
- mqLzyS3G4oowAJAR1Oecela0XkLHZKI7g3MSYK7+SoPJ /DrU8ULTSIoSMwYZiwXnIGefr1FZq
- rGO6NIYunST54fozPupC2kSrJEYsTD92ODnr+AHpVCKVDcX TmLzYyQc+4GanlYOiuFJCKeG5z
- u/nVeSRvs7wFfLEeA7Yxg4zg/lW1OK5bHTD2UIJOOj316FdTHM lu9wdptlAQKMbwxOSfpVe4l
- ZnUGUbI49qjHJGeoojbzwH6yMhby06jB5/TmmxTxSFYpo82zOozxu Az1z2HrXWo2bfY5Jyo0V
- KUE35ak0MkeZGby92Rg7Rhh3A9OKc8quXkSEDzNpiCjHA4z/AJ71TeOb MqymPMbkbQMZIPX9a
- lku3jRFhhA6bTjI2j72KXJd3RxpSlH2jTafT/h/+CSyXRQb5YpYpJZc89Gy MMfoKcbSeQJbxu
- ziJBGnPDLn7w9qh27L6QIfOCEomRn6Ef55q2i3l9GSWX92WVSo2gk84NQ/d1Wh MKMoPm0Ue/b
- 5Fh4YVtnt1kEjeb5ipnJAHUZ/Ws6QK1zFDAylI0ZY36A7j+vpTUed7UxbWWRCAMDJ I60p+zyw
- l1byWMi+XlumSTj9KcYuO7uEYypx7t7Pt52H2kUlsJUkBjSRWBLDow4xn8c1DOY5LKxi t4XZ1
- ADuOpOcA59K0CZLQyCQNcoTuGOrMRyR7Cq0lvC0DCJiZkZUCdd64+99KUZXldhGc5zip6Jd tV
- 8yocbRJMs7gMyStv43nv8AhxxTbiS0MUdtFGTKmUZ853txjAxViaAWdvGVYOrSkgEcYA6H3Pr7
- VVaO6BVjEwkLgYAAJYd/yrWDT1uc1OKac29Lvq7fdchMLiKQLbywFWKhJDk8nlen41XKFrtdrR
- gv NhMjjZj+uKs36m4lS7WUu0U+0qODjHB/SqyRRC3hVgytt3Id3Pet4bXOaeGqV17OOke+uv3
- lkJFA HABYNkgdwPSqIQP5aDckRXLORyuD3OKvgqLXauzcQPnY/mKpyJKbqWPOIweo6DvVQO+v
- QlKiotNJ bWtcj/dHa3mLkqdpwPWljVVijaR2YFmO3dlgSO/HeqBNulvEqo0pYnc2cbD2zVm0W
- JI3nfFwqnYN vTd1/lWso6XPKVeFSpF8vvL+t2vyL4hHzS+asfQOGH8eOFq7EBbiWRJbeVklVN
- q53EdTjj8KoskE lmrBXjYjeGZ8hsHGcU44F0G3M4znA7DOT+Irnacla500ZTqQbk3u9NNf68j
- X8uNrm3X7aUadZHjj YklPVT7miZpreFEX7REduSS2SrdMfX/Gkjeymvo/OkYbpDJE+cYHdavy
- XFvLYi6f/SVdi8oQ4Ikz wPpjtXE5NSSauZ08RKNRRnG66K36r9SigvnhEks0LNJltnlgMdnHY
- e9OvDNDfwybUysbKPl+7xgZ 9acLuSNoRAFGVPmbhnPzc49KbJEkt6R5NxHAJP3bl8/L0z78mm
- rqV2tA53CV6isu3l6f5alEcm2R 4WaR4/nPoT0/HHSltZrWJth5ZiW8x+QrjOFP1ovnBZYY4mM
- inAkLcEDio2mFuohlEZAY4UKNwKnl T7nsa3tzR9SMdVahZ3XlbX+vuIrOTa8dysTFEkCgD1OT
- ileSG4VnzLFIzcgnqTx+n9arW81tLdLP skSLzGZ1EmQrngHoOgqKP9/CgK/Z1TO7J5fPQ+1bO
- HvXegQU6tS8lq1pZrbz6GvFFE0cklyqtLIN 0WTycZyelJ9ilktoGIKLnCNuwqg9c/U9Kmjl+0
- QPLJsklLYAjGAqkY6e3WplsPOYQxSbmbHlyk/I VHBPTPJrndSz1djtlXjOKU2k46/CvzKJtTb
- XEiuMXCOscfHGW7EeuM0+FLWS/aRRKlsofG1iMEDG M/Wp4o5IL9YiDcoW2xrjlmAJU59qo+TM
- kFukkBgm2sJMjAbrjA7VSfN1HQlKVXlet/67r9S1GzND HMCUeNAkasMkoc5PvV+2hU2tlFIk5
- dly5D8dSQoB7kVnpHJJLBGIGeRY8bB/F74q9GYWvkdUaJwf vGQlcnoMY49qzqbaHozpTjsrSX
- W9vvS1/M6GSa3ZYEdPs6eWzs7444+UD60RLCliZZQqhx80fcn/ AOuMmqFxcWElmu6XzH5ZlUH
- tjFU2aP7P5zQXEbM52s8ny4A5HH1rz40W11Rw4ag5zV002/k/vZam J/syVxEAIpcwuy5BRgMA
- +tTobUKGkCnJLOkfy+UQMBD6n0rLaVZpPMtnAkPDRMchmA+8B0wBUwu3 ijlNzBujdgVZV28Fc
- YJ7n3rZ03ax2zg+XkW67Wu/vf5MkdZFtN0sahAVZWK8yEcbvoKczxPpbFxs Mh82EoNpKrwc46
- 81nm+jMIhmk8mEYVA2SSepUH16VOpguppNolKgiNFB5j4+6feqdNpXaMnCjTs5 PVav+r7jUuT
- au/kRlmLqised28ZJH17VCt5NOiReULhVkCymMYZjzjBxxU1zJbx2Qs4cxRRzhiXO 53wMBgfQ
- dxT7SQwSCQxh4mV2aVFwsjAEAj0Har05XLl1CU5Onfk19df69BzXIZZo1jZAXy7NzgqM jP1qB
- Im8tp/9ar4kV0HcenHQZxVa0ZLiNVtw80+zDqB1cHOfoB1+tXfsssUzfvY9oYOQM8LnJ/wp 2U
- XY3w8ox0ST+X+T/Mkjltb54zJMsMxZvMkIyvboParvliaWVYVF6SwkRVHKY6g8d/xpdOtIbbz5
- ZXgD7225HAPUDFWItQSI2/CSSzQ7n8lMFuTkj0xXLOer5NbG6dveUW/L/g7/AIg63LT75k+xwy
- Rl 2yvbODiqKafFbS/eS6tG3eW45ywwRzirE+2GysRHdLcElxIxJPDHr9Paop1+zRQxSSC5hOW
- ZI+CC Bwc9hSg3bTr+hVOcqkE1Gzbta35dn53K0QzayS3dpLbSSTCTOMDaOoAHTNOlNzNbQyIy
- PG43QIEB cgHvxzj60599wYlKSsVQbjnjI54+ucVatoWbTVmhdbhxIFaBOGUnrjsBWkpKOrMMR
- VVCC9o7u+z1 X/AK8QK5lcRw3KSqjnb8uepbA6VrQ2aILqe3LEzsZEJPRB1x+dY8KzxzRTMVtk
- jUoWlGQ/PLf0rc VVluPLWZZ1IwFjODj1Ht2+tc9dtbM893jUdRS0W9rteny9RcC2tIlfzSzQg
- xtuxgZ4H54zSuYRE4 a5T7SsgQJk4PGTx9avGWC0y8LKYdshIk+YrnGOvY9qzphfNBb3cDW8gl
- iZjGIvmVhwQT61zQbk+x eHqutNRS18+v3iR284h81JCzyjzPmOcBf4ceuajSNXggmlmCOFDOq
- 5G3JyeParP2KT7Ku6R3ZIvl CkjgY/nmrMkyqkkRCK+/93Ht+Ypx19ap1OxvKpUWy2/rzKSTxu
- ZE/eujtlpFbhyTxj6in3NrIoI2 R28Q3I7sv3WYgKD9R0rWSSC2nkZ7JsKzonTGD/8AX4FVZp5
- 005V8n55CDscAlNg7/Q4rJVG5LlRn GpUdRcqt266nPGG2+2tbtdKDCGjZiSdxwTuHsO9GZoYL
- FXkRkcI8cuOFAznP1qY2sBtt5mElxMFk lAODnODj6iqk9vsD+W/+iLMIclskZ53fQV6EZKWlz
- 2pRhJcs5aro1+Rblle5V1itnYvKF+U/dX+I fWmynT48JIjARM2Bu6jI5/Kp9rCSdzcRSQ+cpP
- lcc9x7cVRP2qRxNtWGFV8phIuTkkkfj0qYJGbi pPlUvdXqtf68iu09ldmZYfkgDERjv07n271
- Elq/2dZdwdfJ4IGBx1FaVvYQPq1pCwGDGQ+3gEjOT VieCW2t5Y2ddvBicD5dg6jHrWrrRTUYs
- 3hiIUpKlT95rX7+3mYlnPHBJCBGZIirRsoP3i3THv/hU l40IsIoIyDtx0OOB/U1cRIDI00iNJ
- iNgI4yFOOzfgOaoMIfsX7u2mXDKvmO2Qxz249KtNSnexk8P TlX5nF/kr+jf5FNba2k05RG0xu
- dwZFBz8oOCKlKFbu48wbRvwUHGCeg4qyzrCiDyg6OCTEgAeNc9 z+tUrqGRYxKQ8WJQAGH3gRk
- NxW0ZXerFRcIcyau+m/53/T5kSCORLYQ5FwM4GcjaOp/DmqbMHkjV B8zKc8Z3YJ5FW2VwyyKB
- kjAK4AUNkEGpjCsZjVA4kH7tDj7pJ+nvWqkkzKOHkpTnzrRdX/V/Qpu7 CYrEpJbqCOMcYP161
- IFltpxLFNCzLvAyM5HQnB9c1OHhkihtMjcCxMv949KiQPIYklgk2qjN8pAP HWlzdzk9tKpU5Z
- K3l/w7/wAiXzJTakzwmNFHlxqy8kkZ59TVyWK4mhgkkyFiUqwTAJcjg/j/AEqu giaWykWG4eK
- SBiPnz83IzWhYi4iEG2WMIsbb/MXIbGcEficVhUlZXQ4zStZXaelv+A2MSG4WKN5x IGfBGw4D
- Kv3jQ7xxXwkVLs2bSfIXc4P9DzWtAwEdrJdyxSvEhiUJwDuHP+FMkmE2nLZxxH92/lbC MsnOS
- T+Arm9q3LVBh8TUlV5asb33euxT8m8EE7tguW4bpgnqv5Vntc3ioVVWR3cbmBwMjgCtSFd+ po
- QXVjbsyBjxgevqfeqS3EUC/ZmI3MQwc8jI6D861g99LnfKNvdSWnlb7xN1xPcKkieWrg+awXjd
- nH4AVmY2xlZFke6kbEW3ptHUmthrgs82UYTuMMOMAngjFQyW8M0zRRxyRSLOq4c8qP7v145rSE
- +X dWMKuIdLr+GnzM5FJtIdqsswT5COAVLYz/n1ojhjOnzRxyQNmTJGPmUjoM+laRithfOlncR
- kM25N xztxzj+dOmFnHKZ12t03qhxlT/F+FP2t3oeXObcly/l/n0MiK2JlxIWZv+WhU981sQW8
- P2gSCSOF 9sikOM4UEc/TNU4GtVeQRbxznc7E7mydmK0bS3lFqs8zxowOWVhzycE/QdxUV5vq7
- GOMrVeXlba8 rb/mTRxQT2StIV8/cdxXjJPRh6DA6VjrC80aLbxyywuAyFT0Az19T1Oa3IQwin
- id4UMMwiA2+vQ/ lUkUQtL53hdXBXb5S9ieN30A5rnjVcb2ObDV504Slzarvf8Ay0MKS1X7L5s
- iSQlFKoCceav94e1M t4hJ5DRvEspiIbcuQG/h/Tiumh+0rai3eKOchRl9uQ2OGIHYYrJs5Ekl
- VAUWOIFUbbjKnOOfXNaR rycX5FrFVZU27Nta3vp9zRHayTmBbe7CyMs6NtAAZB3BqvcOm+a43
- KjSox2Y5ByBj8OtWTAQkW+K V2f5iVOMDoQffNZsVrdjWIZnhaCJUYnzOR0PP8q0hy3cjOlUVF
- udt/Sz/r0K8xMcbxorg798xfnD EYIHp61EFWOWErKTcZIB7HsCMirdhaXV5BAgG1Sh3O4yCSe
- /4VpNp8SXF2qIT5NxGQp6njgA9vet 3WjH3X/XQ7VXpezcUtd7Xute9+nkcrBbiaaG33GBXbd8
- /Odp5z+tXbpITqNxJ8oCy4iRBjC4/pXQ XFoilfNjadt26UwjbtJ6AemaomLakl3JbneGO1Wx0
- BwQfehYnmdzmwlV+2UpL0WhhNG8l7Dl0jjY cs4yvHJNOu8C4HlkbJAWDHoR2NaTWbSSy7djbC
- qsRxyTxj6jis6a0uA9xGbeYMk4TaRkpz90+9bx qRb3Omo2q0pup8tLFK3tlkhjnLQywxkiTYu
- OcVGJhEYkiURKPmCvzub8qsMpS6kjdNpUlWXbj5um KesYQrJNE02wFVAAwMD5gffmt+bW71PP
- +r+z95Pmb/ruRxXM/wBsjlgMTMqlMFMqAe2PWrtmZBIz Twqux/njdASRjBqg6qpDoQGDYDKMA
- jtx61VkeO4uEKzDOeSWzlhSdNS6EVKLnJzmkm/vN+Ke3MjP NaSiBcquGxtOOn40sU32p3jZfs
- 3ntuDEfKpVTx+NY8UUwgdDcIq53iNsknHXt6UxvMW4jaKZDEWD lcenb61HsU27McY1FTk4r3v
- np8n/AJGr5scqwzufvxElAcfN93P046U0eU1hFIrTLIpKMxYlQCM9 KpzRzSSlslNzl3UJjA74
- 9BioxLGttNHvXLykxtnqOAKPZ3WhtWjJxip6d3/X+Qj+btj3FZFcH5mJ xxznp+FNQN9n2idEw
- /mAMcnd9aTyn+VA6yMUJXC4GM/1p25IzIGUPwMAKMj1GfrW/Qv4qvLPX1e3 3f5j4lX7PIjsFI
- +YHsVHJH1z3ptuPJkjmmYSRfdcDtnlQfrVYqk88Kq+QB7/AHh0H4014pXVCySh nYn7uACDiny
- 30b3POxMruUYz+9/18jp3kWaQ/uQVlQunlrjgde3rWlayzskUslxAjMykJswSB6eg IqtLHHK9
- vktE0RaMoAQV3cih0t0tmWfzEkldCDnG04PFeVPllGx6VdqtS5Ze75dfxJkETass0roY mV3XZ
- kEgdKyrhrhdhIJBHO4Z5z2+uanlh8uKOMSNiFSu7P8AF1P+FQMJpLZJ4Y3eNADKSc4brW1N Ja
- nqWi6XPUfvPRfIdsdpynmbHXKBCuSR3H1q9A1zBC2yIRr56nzZFyAx+6DkemaoiOfyY7mRQjOr
- FVz8ynoM/XNWIA94nlSloZhIBMjnqQPvD09KKjutdjDFV6dSmrbLrq/vWxoTrDBDgx7xvZMgYK
- rk bgff09qdJa2ksRgtGkdDKzJls5GB830ArLuiJSpy6w7Mx7j6kA5Pc1UlX/iYeTE2ZtxETBs
- Ar+VZ wpNpPm1M40aipKftPhd/I27RYY7RRJeWse+YOx2+33fxFOknnmMETp9ms2jMmXTOecVi
- JDGzhWmK oZBtAPL89R7dqvTSlNTkbdhHVwFbnZjjbSlSXNfcy9lBt8z5uv8Aw3n6oqIr5VmKM
- ypuUFOw789a dHalfKMTu5MiuGTPK5wfxz3p4SP7AjXKSlAzKhVsZA4qRUCusdufs6Bm3PK2QC
- P4a1c9NDrjiadS m7QSW1/6epYuZo2S980pNtIUYGCSTnIPYDGKyreIJeJCvmnLsGXd0P8A9f8
- ApVhJlurdZLiLaytl BnGFB6H1Oe9OAtpNRaZo5WJjYqySEAnHXpSinFNWJWHrSpX5bX22X4bl
- WNZliVJZ7dJI5mjGxMbg er8DnkY9a0Vim2F5C8sgmwdvRsr0/ACqkMP+grMkQniU4Y4ycn39j
- WqlmWkRWDJIyu0gPQkHAYeg 5pVJpdTWhenFKUv1/KxtRW9td6OptldoU4J3ZJ29TUEbyS2Bhi
- tDCrsCrsBlcD196itre7+zkNHJ FEiBNq8ZOcdvzPrU8kMcUbeRFchoWKySM+V3g+noc15+ibV
- 76nbQahNJT5ne68isZ5Xs4RHCIwRg hhn5egPTtzUjqs9vEk+54oodsUsZxnnjOeuenNW72QDS
- kSRV4DIrhQOMjp9c1T3I1hDE1vMdn7p1 U4KkHdz9P61UZXiml1PRjiYSoxaWrbX9aXJ2s3WGd
- mmEN4swxGTzgDmmxTyfZ5P9GkRJJUYEcc9j 9Kie4me/kliUsJJo3IbkqT1X8RUV0MEpLDO1vG
- +yMqSBg9BnoTSUW9JHn4mPM7TSdtbu1/0uWzbM YLVrhiiuT5ise+7GParwjtbeK8tkSa4/eM5
- eJsFsEYCnqOKy7V76MbPIllPyZZuQmeD/AJ9auvDL l7dbG68hA2FVhv4Hc1lNO9mzyXRl7ZRb
- 09bfk0XYY1kndyrWSzoWQXHzcD/OPxqxaTPPLLLG0TSM p3QquPLOMH/GqkTWsVis12ZI2mIZQ
- 7Z2jGDj271E8lsmnLcRCSRVYKGRsbgep9655Rcm18v6/wCH NIRlJtX301S/Pf8AE25ZrhHjLy
- ReSkflEeXyCRgkn071UgLSXiFZ7e42RnYyJ26H8T2qH5o79ZIp cIVLBJMtkqeP0Jq1CWMjfab
- SZUT5AUO3JJ4P0HesuVRiarDuCklZ6b6L8P8AJkl5cTWkFvtMe2OB hEWGdxyMk+vXFZsswXT3
- BYyTp/AD8xQnJb6DvWxcQkaYzNDJhm3BmPygAjcPxqjPeCYrHHCsscpY 741ALAfxA+nalRaaV
- kOjOm1GSXXfb+vvI4HRdTu2u1WO389Qvy4OcZUZ9Kx3u72K/QosbGYlinlg hevBrVMv76DzZE
- jQJudnHAboD+VdT4i8Q3/jHxZFf65FpkDWkEVsiWFqtuAiLtUMF6uepbvWvtHC fw3TWvy2sut
- /U6pXpzc3FO6/q3Q5ZYYpkg3zxC5dAzBeFceoFRm6d52QGI4beFC9SOp/CiSTN3bC GeBh5Xyq
- F+bAyM59M1lXE9wCkV7GHSSMuTF8pY9Mj29q1hTcnqKnH2tS1r+vT00RYvPtKRxRTKQX OfNQb
- Qy9Tj9Kou58yYxszTAmXy2+bb/eHPGO9RwSzRs7NBJOGkOHHRQvWrEV3K6TsQsgKbnVUAJY 8D
- nHT1FdcYOPY9DD0eXmcEml1/q41ne4t7jyImn2sE3x8A7un4cVUihvVgkjXC+SwCRtyWz/ABD2
- FRJNtj2tIYycq2DjeR0I9u1NEi3VmrfaPNmGAMKRsAzkHjk1uoNLyM6vNLRNfd/X6DSN0kj7wF
- En lgs2cf5FV/LfyJICSqGUMDJnjAOF+prWjb7bvgMPkOGErsehIHyjpxmobYeZdxy3qhJGd2Z
- TwMBf T2NV7Syd+hhXxajG01drorFQrAzRqqSR7o2KKxyeT096favJaR2+4qrYZmMq5Bx/Dj19
- 6cwEssrN Iql3jKEcAEjoPSrMk0T2MisFndZMsmOVPcZ/CiTurWuY1o3tTjC73+/8DOMwMTK6R
- 75QpUouGDZ6 UjtAjHG4uEIY5wc5ois0KxP52+Jst5YPz4ByRz3xVi2U/bk/1ckksLMAy5GTn+
- VaOUVex20MQkpK Mdkt1/X5BbRPFYyyfIdj+WqkZ4I5I/OtGW38rT4EM8TQZDK654H8Q/HikUN
- NeQ/aIZJ4kQKwh+X5 m4H5ir0KhLQ2UUbW0gJJW4+Yrg5rkqVHf+v6/A46mLrJRpu1k79NP1/A
- pz20K2MhR3Yh94TdkqM9 D/OrFxaSqUEUcgdonkJz6c8+9Mis4LnJM26V1ZwVzjGM8ipYo5PtE
- RacwMSI1aQkgKycnFZuduuw 1VlCEWp6pt9fyX+QqW8k9jbF5FWCaIOnHzAjgjOO9VntIAj7YJ
- jJJJGQWIJGc8D8qswGaJbYSq0s PkeZGUGPlQnj8aVryZmjOwQQvDvLuOxPUVKlO+g3iK7Tell
- 20X3LVsUrFcakIyyRyRMqoD1MZHzH 6jjmnXJtZroQNbS27RyHZuf5mAXqfWoY3ni1aQCNLiJZ
- FQyIv3jj5QD71cYvaaotzdYTcjK+R0cE dPzrN6NWfTQ4akFzys29LpJ316kUEUD2UefJWZURW
- ULyCW6H3NVTbPDeX108JeIuUVcdeOMe1aSL Msk5a3Z9rEZUj5eOAffNZ8mmKloUYXMLljuWR8
- 5I5/rThNXd3uc9GvGDak7J6ar/AIYZDJCIgJrV R86bNqgEjPzfrWxPBFdWoCoVMgxkHqWOAw9
- sDpWRAbgXkm6LzJMqWCjox4WtG2tLpZwv2e4gVNzT lzkK2CcD04qa2jvc5selTmnB3fS3+TGS
- abc2czxwSpJlXZyR95h8u76YzTbe4RYLQ/ZpHMdpIEYH mRT/ABVPHLcLZRSeW7RKilNxyQAck
- E9zin3+pNIs8lvp06ySI3kEKCAhwCQB2qFKTdpK/nsZQ5qr UatrPrdL7zNigjFtbsxuHC8RMs
- hHmLtzke2f5VCkc6G3uSYVE0Bk+5gADjH+HpWjZyRWFnE7I00a ryueSR6eg56Ut3uvltjKVSF
- yrIqDBeP+LFaKo1Jp7EwlKDkm/del97+SMCGR0vRFAx5ZWjEhJ/dj P6mrQjhmljt3uVHmxMYz
- k/KobOD6k9M1oLPDDq4RZLUwsrC1cr98YyD9M1E1sIrEzPC0sjgeYU4w cY49BWjqXfYqTU6qi
- pWb2/qyKCW0RvJoWuVt4ZYw+4k4hbsh9yaty2c62P7uRVLSbpQc5VwBx+Iq 2sYVSuxduSjhgN
- 28Abfz5IolmllK20REUTYkkLjJ3LgYz71DqybVjSpKTaUXtve3+VyKG1uPtVxk E+bOoOP72Ol
- STW8htPL86JDtZWZkz83p+NSPqFxFDMUMcryOXLBeFY8AflSCzuHjSOdxK3zBgowX C87x7Vm5
- SveVkcVab5lUmkn007eRTeWCNHcsk8oYIkcYwX5yD+GKzJ7K+utQW4mKwS3SvJhsjDDs evJxx
- Vy4LzXtwi+Tb5dE8wrwWYcEelQz6XcSPKnnSSXUXMYVjhwp+Yiuqm1CzvZs6G1F25kpPurq 3z
- 0RhanA8X9nv5ixNJC7sHGfbH1rP87zYlXz43iUlSdvQngZrqbv7LfwurFYplAMG9v+WZ5YfXjO
- awo9OMO1YQZxKPMBjUk7R3PFd9CquS0tx4NX0qaPXW23muhRnja38t/LDtuKsATwfy9KSCzU3q
- RK 0WcsWJXvjIH1xSvL5IgguonVnmXMnQRqMHJFW7iJjdA28UwV5XZmAzkge3TAro5mlbv1MJ1
- 5TrOm 9+70/DZjBaxMiFpViATDE5Jk75X2xS21rbPKZJX2RYKgZ/iI4FXbWKGfT0kuHCTFcbW4
- yp6EfhSC ezhv4yIGdFjbzCG4Jx8p/DpWLqSd0ric6klJQk5SW+iRHaWVwdQWUxlNwI2OM5OOR
- +FULjS33Isc MrIEz+NaDwzyNCAjhERUc9dhbqTn9aIiI/tQWUyCNwkTA8eWec/Wmqkk+ZMuTk
- m+eV21tbRfO5if vE+0bFaMNMp5PTGcD6VDsZ7uNFDlmfn25rTktRJM8M0iwsiliTnsQapSNBL
- f4EjxIkvX269q6ozT 2KxKp001De26RIYdt5ttiJWCMQ4HDL3I+lQLDLuj4LKEIGO4PNSnzRLN
- dW6uI5C4HooPUfrTreZx ClrIishkDBioyBjpReSRzVJzilJw5l1t/Vjoo2WFZRNDLPIpJZkOO
- x6/n+lKl0o05UDI7FSU3ruK jo2ffuKWJZYZYxZzQ3SKGQsFz19c9T6VGVZd7CDJztI9Mfxf7o
- 7+tedaLev9f5HqNUue+7Xz/P8A yKH2OST7ivLtycqDg46896kTYSoEcqpySmeT7j8K1bKFbV5
- TvYxk5jyc8Y+99O1QxNH5m8XEO7lm JQ/MMcAenvVus232O7629Woad7bfqjPe4VYY1w6AnCZP
- 3u/6VE5jlvYIbeKaWQuwMitx0yrU/Y82 9vK+8+MkfdOMkD3p6Ru/ARP3hVSduBnoBWqskP2F6
- XxKPX/h9dRi21wumZEE6qhA+fPOTkkflQyc tIJI2ZgXUqvKhuMVrS6eVmERkJEX7uUjpuHIx9
- adDHYpFlGIlSI7AxzvB5NZe3W61MqfIkuV3+Vv zbKAhaWz852RCZTuGOd2PlUemetNRFmkPnS
- eSzjIdhwQBz071opa7cyAbWaQDY38Bxnn37/SorSI PMkcEYkVshsjdgnkip9qrPUKUlaT9orL
- 5fduZMgMssWJgwALAAHAz1qytuHcRI5MahpA56YyOTV4 2U6sFW1Ko7MyuBwoA5H41FBE8a7wj
- hFAzuPO09vzq/aprRmsOSUHyavpbo+4WvmQC58qS2mgaQfv dgKg54HIyAfStBLRDqrH7JOCJN
- rMCNoUjP8An61YOnXMt0IRAURiTKw4UPj5f0qUtFbW0c3mkSCN g6s2S2RgfpXHOtzP3d2cc5q
- o/Zp+8+3+dyGOO1kvmtYLaZI2/eKQ/G0DJNPjt3EZuVlDSvEDAjdR HnBz60skUz2hiSRAY22L
- tHKpjJU+pzTxCstrD5LFInjY8nJTkArn261m5ab/ANf8E6acvZpWdk9G nr+ZYjdbaOCNmkkR1
- aRcN+hPtUd3fGaZY7VsNu3SHqrYHBotrNBqcoecRbJESOR+VAPUEepp89s0 GsRfKixO+zdjv2
- /Lv9az9zn8zVKhGo03dpXSWiKpu4pLCWafMkjMCyLxliOCPQVG0s0dpiQGCdDg hhnb7H1LZ6+
- lTysgvxE1v5hRC6hfp0I71nG5ma8C26b5ePkZd3AGST64reEU+nmelCFoNyVra2bW v+ROZLgX
- MjxvEpEnOV4YnoR7CmXksL3scEXmLsbYCzZDZHB/nzVy2e4XEpiDo0ZwwUYXJyQff0qm BBMI5
- VIgwmArDJOD14pxaUttjlnyufa+3LqWx9qh2Rf6xHKEt/eCnPH4VoSXCxXUoEc6QOzNlnyR 3H
- Ppms4AgQRxP5mCrq/93HQH69fpV6G2mbUpI3zKjNwR/Gx4JHsPSueoo7s55UotynK2i9H+JLNa
- pcS6fiWNoxExkBHEpxwV9ADWnDZW4s08vdFIIh5ivyAMYPHrWMj6p9sWWONYUkUvGjqDtUHbnp
- 07 1agsbydSJxPGwIThsBsd/wAa56iaSTnoJRcbRdXTfT+kTSQQ/YvJjkKW6jaJHOccZHP51It
- y7Jh8 iSR02g9G3DHH0qF90sgtzNFCHbeA4zn2H5VdunumUvBHGjRnKhkBJPHSs30TO2jKo4KE
- bX6Xdvns UpftaxwWqymQRwlmQ5JXB+YGsZjapcQyvDcRxyFWQq/3B3B+tbRe2iCefcDznYZwc
- bs/0NVJoLFp mDzbreL5AFODnGR+Z6V0UpW0aOuE1GMoyjb0X5bDnezyrBXdVVmZN/K4OFz9Ot
- VJnH2yIMJ5AsJM rI+N7HoPxHSp5bsKVMcaNM8ZkPHB4/lWU6B7kvLdxRK4UHk8Ajn8q0pQ6s5
- KdKKg5u67Xd39yHyQ TxQWrW8MytJ/rGLD5WB4PsKkiu7d7j7Net5tugJEkfB3AfLg+h9Kg+e2
- tt73a3MI+RyufmZjxj8K bPEILxEGyVYlAbaMZx3/ABrdJS0ZFKopR/ePbZrf7/8AMkSSBUR4h
- Ise07gWyQT3NSiaTbAITBLN 5LD5ExvP/wCqiO7T+0vLlEUapDLtAQABiOM+tQRW3kadHcQybn
- ZR8vfjHI/Ok0upM5xhpb06/wCQ k0MyMjq0EXG5mKZx/wDXOahkVSWhsYMSiTMRx95F9fU5zQR
- fQSCCQ+YVBK/KD781FDZ33l+aI5F4 YhvYDkVqkrXbRqqa5b1JJN7Wf5rqbNhCqvLdX0yW4zsb
- K4y38B47VlmW1ktlaRvtjq5wYiU5z93n 1q8siR6DA6p5zDHnBju+Y42j8R0qoklsbM5gWJ496
- IwGA248Hp1FZQT5m3f+vxOWjGpXq3ld306J fduR3MsX2TDhI5pGDbMYKEev17UksU/lykmLNw
- N5A6qV5btxioZUNvPChKOpQhWxncM849/etdl0 2f7NbJM6LIS5YsSenC/U1rJ8iVtjtrupQgu
- TVd0k7HNSNE3lFElBYMwYNwoPH9K2IYX/ALOuUlgA Pmr+8AGY/UZ9D1xWhaWtlFp6+XPFG+4M
- vmnJPoKjF9b272UkmXBiYTYPDMCeMetTOu5aRW3/AA55 31icpJxhd+e/4/8ABKcDPBOtz5c21
- GA68NjIP5549K0Q7XVwqySLC0swXBHzFtuV5/n61Etw7okk M0LRoNgiK5O7OT+dRRy+bqMcty
- 6w7ZVZVxjnJx0qJJyu7am1bmqRlVWkkun5WLkENpcyRI/mwZfM yq2CWPQD0HtVojT1iW3Nyok
- 3CRy55BXp+GOKQ3FnJLJvI+0vJn5eNvzfrxmq8UUMetXW+VIVWb5f MG7dwSPwrmd3e91Y5XBy
- fM1LTp1v/XkP1I28sLNB5iDEjDL8YYA/l/jVeBUazjEV5BGpj3kOu7y2 B4Qn3OaWK1tWhQzzG
- QeWNwVsdG4/StV4Izdp5Nvtt0RhIMDOTz+lN1FGKj2O5csaai09Nf6v/wAA iLw3FjKZ5IiJSZ
- AkY2lZOyfXjis1YruCWFJZo4z5DPJ567xGQeQffpWnHst7ASwQm8uVKxIV+64b knHr6HtUBkj
- eKJ79SvkwmNU7shbkn1IpU5NXSWn9dDKivdajDT5X+4UTi4sXjSTzD5uAyt94dd34 d6zgt3Nc
- h1n2xuoYGQk84p4FtHE7RhobJpdsTMeoPv8A561KtzYTT7lL7zJsgTfjKnqfwx+VWly3 5UcU5
- uCbjt3texEYI59IneJnL5CmTccOc/LUstrLb2zol64mllJKsxJ4A/TtVuRryN2jIidAyliq YB
- brkegqbzWuZ5VIQs0jYbb1XjJHoKzdSS9DzcRiK0ElGWi1/r/IQDybKaVCzsx2qM5CqRjp61mN
- p1uWiX7Yw8iFlPzngjBA/Gt2K3DObUP5bZZstzuCjlh7VB58clgrHy8SHc5C92GP5VnTrSi7xZ
- zx xtWPurV/13RTSVDv3xtgkGfn+NuRj0AHUVox28kkcSYUMbeXqPu7e/0PpWWl0rytawotxuc
- bivVS B1+mBUYtGW9h86eSOCQFkBcguvRiD6VU4edhzhzTT+Hl1tvclhsYJ7a2aZWYKoZGU42B
- hgA/zqGC 2uLTyh88ttvKyEnOWIOOvr2psFxC0zRafcCNgQEEmWAC5wD+taSu11bRpcN5CMQQR
- xs7Kp9STVzn OOj27Fz54NX+5p3/AAI547f+ybOQ2t0ZGg37RJySvC/iKiu0uX0qNoF2zDBkBG
- S+D1HoKmNy6acI GUCXcpYMOjE4yPRfatW4kWRIA6BmS3dJzHwNxIxWPtJRa0vqTCdWHI3G+vm
- 9PO/QxozJHcySXMO1 SWymANxGMEegFQzJLN+9SO5S4L7Qd/G36dsmpQWuPMjmkWRWxKu3g4x/
- I4qyVuis0ovbZAW4Taej Dkf19q05uV36luElNyej/roVioe2AUrbXBlDCKUZMij+MY7AVGsar
- MxDvEsyFkmckrg9h6Z/rWi2 nXCabJFHjyovlSRlyVJwApPqajiihMslpdB43J3MWPAK9VHpx2
- qVUVtHc5rRgubmv5X/AKv6HOrD LHJGd0UNsnlj51yfmBHWqsFkvleXH50siy7XMb4A2tyPoRX
- YyW1iZxGzGVZlDhA3Me052H3rDl0u 7N5G9gsjrL+/YZztC/wn3NdNPFKV9bf1/wAObTxUHzJe
- 7deiOXuYrd9akZFZrUBypJyW9D9M1XIj /wBHdTLcKIyJdrEbH/HsRWx5MCXMUTXUeAGEJVfvL
- yT/AFFLbRRR3RWK6gLcuQUzgqO/rXo+2SR1 1J03CLT97qulvWxVtQG0iGWCWMx+YyTlueB9xR
- 6HBp94k8iaYosZWLQFGMQxnH8Rq4Z0h0qNE+zy 5bMaIuCFPUH1J5we1QRXFu8Rmj89XTPlKZN
- 2E7/lWalK/Nb+vwOLD08To46a9X/TQQWs8GnpdR3S 8qUdG/vZ5FZptoVuIlMjmRVDPg8eZn5R
- +XatWJ9Rlt0DeT9mYhg4QYw3JX/ePrVOeCC1MMyMW3vv VS3JX1zVwk+ZpvXyOiKqTj7ObXM9r
- fkyvI9xBqVx5iDzBMFbeuduO35dqpMVcvLtYj+FUAB29d3I qwytOymabLq6KBnllPLE/QdzU6
- pYvcXBG7yon2RsW4K//XrdNRW2prVcabsrp9bLf+vIwS0gjVd6 hd7sx7EEDA/SrME86P8AIgV
- iMfdB2fXIpzAb3aOARxsC2G56f/XqCU2yGMs7NyG2huWbNdN1JWsY xpclOdlpvZu977nV+ey/
- vjCbeRWw644OeCMeuMU1LZpXUyXUaoke1m59eB9TULGaZTLJDIY2kfJH GOBkH9KsW2nwskZkn
- 8tZIi4BP3mB4/AV5ztGN9vxPYp07U+fl+e6foxxt44Y7m3hguUkeUMUkfJQ D7yn86pysn2ZY0
- 2sm3ggdw3StMhRCnmXQllizuYA4Oefx9KgaQKqzC1ISZc4OOcnGR6AH9amEn1/ r7yqClF+/dJ
- bX/4IkbCOe5SaNoUaQMofqBnn9KdcRRhVS3vIZ0dtyRqp3DBqWG1Z7g26ROAhMcnm HJDt/hir
- C2lmtr5UxaN1kwWz+QqJTipXKqOmqi017KzX9fMyY5JItzqsqSyTfNubgHnK49T2qdLC Z75o/
- KcquUAAGV2jp+HetARRRlEuNsaBC0bHtzxn1I9aoXSXcCQkM6IhaOVgf9YzcnH6VUajb93Q Kc
- p0pXste/6alo+QLCOOC6UOpG5mBwM/1NPtxE5lMit5MV0kYKNg57UyO032FqwiZ1TdgL1ODzk9
- 8fpTpd0V2LqFGW3Me5Q3Ic4xn8DWWjukxQhOrTkqctXfdobPdXIubl1LCUXB8tQM7lI+bA9OlN
- gQ vBgAh1mX5Tz8uM5PtxSCCK50yCaORmuIyqRop5zz19STz9BVh/OsQXiCh5lBfK5OQcZHsc0
- 9EuVb m1Kh7vJ7ql6b+r7F+Nbi41WKQMxWWIzvsOArAYIP86ovGZLWFEdJCrRh37A9vzqSNY44
- 5LjModJF hChsde35VcNoq6vMbe0n2Rt5TEtkMSRj8RXPzKL/AK/rqcdVRUpS5tvS1/1KUpeyu
- yLcGbe7Z74P /wBfNbUy202gBEXaXdXjQH5sAgH86yJYb2wV38vlZdh3DOxs4Cn3INXY72eG8u
- FdoGaORYV+ToG/ z+dRUi5JSjrb8S3S5oxmkm12/UqTacvn3X2iYv8AvWYIpweB15plrF51t5K
- M9u0jCZTM24v6sPYY 5q1LcGaSaORhJGjlXCDBBPyqCf1qhJPqfkJbJCryoTHKQnOTyNvoMdq1
- i5yjZs6I1JWSlq/w/wAi pJBKwlVJBJhhIze3QCh42guy8wZZWUqxBxtbGT/hTZYHHm+ZIseJA
- oUn+8Pl/LmtNU+xOIgyzMbZ pGLc5YHtmt5Tsl1O6eIaS5NfJ9f0sZ6xyywRP88Q2qPLz90dCx
- /GpJY8vFbBg/kxhCyj7xJzxx3F IuoTDMgjX5Gx0zjcclT71bSaI3SGSJ5ZDKCY1O1gV6H6c0p
- Oad2jnnOtGSlOndLVa9flYpKftF4E UNHC2CyA8kdCwOOgrZP2y0nKQSJPHEjCNgPvA/eaoolH
- 9qecAsK7hjI/hwcj86jkWGSBHlEySu2A Q/ypkYAP161jKXM0un3i5ozq6r3X87f18x7yyT6dJ
- P8APGhQqi7uQuRmtWG6jjtXlR3ZhIDMNxON y8/TFZsyvYwxqtxGfs4MWCvXuDj9KjjSaWSONf
- 3jyxSSlV4C4HQ/SspQjKPkdMXGVNc2iTeu34WN CyDfbE8ueKdI7faqheQTnGc9TQ2o3NvJyVe
- FU+UEfxH/AArIgSTfZyEOiOo3Nnjdnj/CpXmtE8xL mKcPJligbBjOc4P1qnRi5a6m8nQdVqKU
- o2XTX8CXD3jW80sLzgR+WqxnbuOSQ49h0qC6n8prmJRv 81g+7HA9f/rVZN9eRNE9vD5cQVsbl
- BwT1qv9pVFtpHmguyImXCJjGDwDnv8A4VUVK+qFOrONT3o8 3az/AKRTgvIxdsXBEewqrE/dUD
- gH60eXNN5TwvHc4wr7V+6cdDS20PmyvEGgjRW34K9R1Y/QCpFu YY4XELBVBJVQOSc8HP0rd6P
- 3VqYShOVW1Na9nt+S/MSOdfsxhnaM7ZVdV2YLHPX8K0poyUkaYBv3 oRGXjI75/GqqzPvmmkjg
- lRZUCzKoAX0B/Oo2e7acLHmSSFWDei/XNYyjd6aHLLDSjV9yKsvMfOYD BOwifzFbYwJ5XjnPv
- UVn5n2CadEcohAiOeCpzkfnV20t02r5sggjnchRIc+YMYz+dKYfsf7h0e4D KC5jOApXp+FL2k
- UuVGk8VFRcEua3m/8AhiB7G4uGt55LS7CheTu6nufoPSnyQTiITRyB/wB2+zaT gjcAPzodVkn
- MdxfGNXDSSDJ+V+m3jpx2oQWbWlsod32QjIVyNuW6H370Xlp/kKMKianKPytp+JGI 0uJriJoJ
- UeNkVgGwN3/1qrhHt4UTyGdJXDsx5Abv+FWJIrtpXhhglUxSBBzySfU9zT3tmguJUnlM NzGxG
- 1+ijAzke1UpdL/I2s2/ea9N7fdr+BC93JLFBJ5EbLFCyA7BkE0QiOSGPcnl8rIHHADDoPp7 Va
- kM0wjQRo+JMII1xuU9G9wMc1Qe5khnYh4nzvIULkegFOOqskHKnDljDX5/1+BKJIJZDhZJpZ0M
- gCHG0joPxqOFQlrEZZraIxtsCyLkgNyW/lSB4Y9OUKcpImFVRhg4PBz2HtV57KFrSBZ8xyMuAM
- 85 B5B/CiTS3Oeoocq5m/69V/kZlvagsY3u4Y5Fk+TKn5gvJNMVDPdIFni8uQfIzDgnnA/pWt/
- Y9tIF WJLkKX3I2/kDPANE1rJBptzIwV44rkFNgxz7HtzT+sRb0e4lmE3VXK7y87aencdHCEtT
- E0LtKG38 /eIX/CqcTyvdxzzRtsZQ5PYFsjn61Izz2YBk3G7hLQ5J4Ibkkj2BrRSAzpOks0f2b
- 7pVRg8DK8/h mseblTb6mkW2nKfXrbUpLsZo7Jka2BwJC55YL6Gp7lk8mFIZjEs4MsgZsnOMCr
- o09pNl07G5GcAx 8csOn14qxHYSpdwiJ4bciMszzpu2kgjb9fSsXWhe9yY1YSd3K/L5/wDDlBY
- Z5EMsz/Z4QAPMHCo/ YfWopJFYsH2tkEmU8qB0K4+vFb9pbpILVJp4ii2jq/oDnIz7+9ZDrbWt
- v9r86J0kZXRPRQMAfXPN RGqpSsTUnRrVLNO/TRpf16/cZgR28xoGGwrmONlzgjgjHrUFsY/7N
- kuJNqyLdRgR4wyZ6j8e1XpV sp51tjMyTPIrK4PGf7v49qfEl8k7JPBGQWbJ8sAZ7fjXVz+7qZ
- 1qXuNSirrdbO3r1BHaW4vGTckS TKrlj91ieAamFmWBhRnm2EqZIzjeq8kj2rOkR7cXEW9UIkB
- mLDhmA4H1NadtfTJAgkKJbylWJK8s xB4B7D2rKpGSV4nJisNVjS54Ky83+Xcc9pmMsPPAbLJ8
- 5zt7nPpVtLCOK4ciOQxhWIQn7uB/jWDN qL+WI/MHmJMgGD/B/EKtz36Q3HnQrM6P5jBi+R24/
- nUSpVdF3OCWHxEYqM+o3Z9jVZoE8xvLAfaP vs3BI9uaikSSDT9tk27a5WPzDuKqpA7+pJofU4
- /tEJnRijQvt2nAG3kVWW7spktWZpOYnaQq2Aec n9a2jCejaOinh5qpFqN/lv5b9CNobiz1CNV
- 8mK7lTKoU546j61dg1NEMSbo5C0LhyE4V88Hn0p01 yktqZhBJ5o5GW6ggZx9KC9v5VswtARty
- ygANuJ+UE05e8vfWpo6ejq1Y3vp9lbEskkUmlWyR3UT3 L7V3f7p+9+Nab291Lbunn273QmLsE
- TGOO4+nNc59lmmSRmhdZy4VEHHGecfStbF9BNEjTxSRxrIu FHzEHGGzWNSCVuVmVWF4Rakm09
- tPxtuiMM8QSQGKVUX5nVcLg5H5YrTtoLY2hQ3UZZXAZM84Xr+h qlC8Zhj80jAAXbjG4A4Y/QG
- iKAPBcNBcxSqCUBC8t3Y1nU1XY5q9GlK17p9+j8tjbu/LNpJFbziJ GcMN+Tz/AAA+/FZs91cC
- 8t0uvIjmdW3Hy8Yz1/Gie12QIzXAOeAvcn1psTyGYABGZEKjeufcn9Kx pwio9zNYei6Gmthqx
- piV2Yu7OigIeQDkVjX00sNsypKyyvIrlgxAGO49j6Vsz3Eql3sgiYmViWXI A25C/U1lSi5ur8
- XEkW1JIWE8W0ZDEcY9MZrqoXveWxjTjUvepJKPbS/3GM6tJPdLCiSThw0WxfzI 9s9qdYi6m1B
- xcRiJ3bcCUABBBBH41pW9w4s4EneDEEZjDogBfnKnPfNKLiKW3ISeBJHkWSMMOUDD kH6YrslU
- dmrfM66s+ePJGHz3t9yMq6hiHlGJlA2oIVJ5jXP3W9W96Y6TQ3U8s9p5UXnbAdvABBz+ FX3m0
- 8RP50Es4QbY2jbAZM/epXjmuYJjbRud02/LjI+7/wDWqoza328y+WomubVeen4mbDOFtTEo aS
- OOZDHGAQxABA/WqEhY3dvAdsPyoR5o3K2CTke1PFjPHpcUhWWJ3XchkP3gOv5npTyInu5XEcoU
- YMbHoExyPc11pRV2tTR4ai6TdPr89fLZfgWzY2cjG6muI3UxsrpH8pDdmHtisqxjmitlkZ4VZD
- 99 l+V/7px+OPxq3cLHLcwJHMGhZiY0UEHGMDJqrKhjmInSVUQgZDEA4HH50oJ2s3uedChUkn7
- XV9v+ BomRMn/E4jKTxiZgcRkE7F53A+9Ytu8ciQvHE7jzwgfqI25xnIrTiEswiAbzp9rLiNcE
- c8A+5qqE tnvIFVGUq3HO0BvU8V207K6ZXs27yjNbbNar+vQ69EfzZYLp1QkhpCeAp5yMe4xVi
- ya5tdMtQiCN VG4TTKGDDJHH51npzJN5jGSYjDKOCzHPTg9Bip1lkiSKOOTcN6J5bjJ39Mfj1x
- Xl1INqx7+Nw0ZR Ttp2dkvlrf70X4tPeaGUzMke1xHwuO+cH396jha5SCWMrEVEm1Y9nIBHJHt
- irV0oIZZ1mSASssrq 2Ar4ABqnpSRrrM0qrJP5J8vy92S+c/N9MVgpNwcmccZynCUpu6jsv+CR
- LbRxX8TefldpDndyCDgZ PrWhDJG2oIrRmSMMYwndSeASe9JdPY2/2aZVdlH7tOcgqR9/3wahQ
- tbWyRQ5kKnLS4+8V5yPqDTl JzVy515VlzRur6a2K5Rft4hc8xxYG8ZGz+L8ff2rVMFt5KpEzT
- 7QykhuCeAD+PaqZilvEku4k8mP 7sRPO5cevvVtZLRtJFrLOsCYZgp+8nQqCe5qakr2t8zsrVF
- OUeWWmzSW36lM3VzDFJGzRl4pkUoF xuGeSPqcVaRAbO4cQSRS5KoHOQxzxgfnUbXVtDBGZDHL
- NMpaQIOS/wDCR6D2qMzyRQXNuJ1GJV3b 1yV2j/E03FtaKxU6Kk06Ufwf6Fa1NoJVd4J4xjBw+
- MZ+7+INWrXylupH3M4EmxtxyN204qrc+fKF lDxOFdRIij7wAzxWhYvZrqs2Z0jjdsszAkZZTj
- FVU+FsiULU5Sjfz3YBGmsxFFGRKoy4bklh3/Kq lrhmPlvOTsBY7+CTnn8K0YI0tIZU+1RyylQ
- 5IJGQvJx9QarxSSSXAmcIttgKsaKASDkdfY4qFLR2 2No3hBqMdPuf/BFFxIsHlqrXYOG9c4HX
- 655zSwm5+02pWEP5sJldccsQfvfQUR+ZNcSolpMm3AVw funHQ8VbAgaye6j8yO4OWUbs4QcEf
- iOaiTUdLbmNblppQS1l6P8A4Yql2aaOVraRYgmM5A8wHv8A X0NZ6ht8ckU32dZWEhMhJ2gkgZ
- qxDZJFELh55JogNqKrY+Ugn9M0yRVNtI6sBHHLGiuBkFdvWtYt XsjalSja0Ov3fjdDoLJo3Ju
- cs+4GIZxuVev9Kp/ZpzPsiLNJtKBGOSB1wfrmrsptiLUO0rJEjRFg /wB71Ye3OKkaWM26xsCY
- 9oLqp+ckcDn3pqc9+4QjiXUb5r/LT/L8iKa1aO4txBG880oO1E7KMbif 6fSnbNt6ELpEspacF
- uSdowpB9D6VPE98LZpoyEk8xioIyVzxj6E0qRX0EOJo1doMxu5XhQecfmah yfVr+v6Rk+aMru
- af9ea/IhguS9r5k1zC2844XiNccofc9qpxCB3iaRmuNvEcMZIJHPJ9/wDCtCIx PbbruB2DMPM
- WIhdj7cAH3pbfy4rRHjtZCIEMZbgnceMf1p8yV7IrDyTbtdv5L/L9B8eoqsTuY1XK kuHAYhxw
- p+mO1SBVlaO4BaOQx4HOCwK/MfwPNNksLh4YmBhdUQJt2ffIGfzNOlsmlljPlTRYjJkB fgEnh
- R+FZN0+jsdsoUo2Stfv/Vv1Ik02OBY5ILpZYo/unJwSejfT0pLxmaWMyQ4LfIjEffPQn8Kl W1
- thPKheSSKOVUCq5BGf8K0oEjkmEZUYRmMbE/d9vqTSlVcXzPU6lUnQleyb726fLr6mCtrMyqsQ
- eU8MNp4BHX9KtpaRSWTGJlbegJA7Ed/xFbHmmSENGyW8jlclxnkA46etVZZCqzg3EH3FZgq4Mb
- Lx il7ectNjSOIqVpXpzSt5P8d/zMtkkRI4/IZnijCFl43buao2sSTamsbQecPLO9F6lgD0rqk
- W2mtA 0kolU4VVj4OV6knuBWBMr7A0A2mSTcSBhk7Ek+hrSlV5rrY4qVf2kpQuvx3/AD/L1Egh
- zo0weRYw XXMZHzFiM7voKkja1SBPKLzuRhirdTmrtvb2cd5cySyM3ly7FGeNnf8AHnrUdxDap
- GVs9ysMxg7s 8kZFT7ROVtTGEoqpy6272svw1IkPm3O7y28hXEUS55TOf1pz3rxXS+aUAGWOR9
- 5upH5VFbxyR2qP JMjE7QmB1ycZpJLCaO/YFhiIFCjDJJxwPrVe45WZvHkm3GaWv9biPKsN3I0
- USSR5OI3AZjkZ3fQC po2m+TMSyW7EPHsXBZQPvA+grQ0+1EdxELi1fyXXMsh7seQo9Owqw1tG
- JriQxyRuJgjjOAoK/dA7 VjOtFO1jCq4RnyqOvfT9HuVoLW7ju8bTPskUbh3Pb+dPniie9Jt5Y
- 3O10eRzuBHUfj1qS1g1VILw wzRySLPGxbZwxx1HtWc6Bb9JJLmORRG0aCMYABPGfc+tZq8pt3
- +70MvZSlVclJJ22V7/AD6DLd2n v2mbKR4WIdssehHsMVWubX7E3mN5cx3FmOMhW6FTn6062Js
- J4mgcMUXZKrDIVycg1PYRwPqEg1CK dwiFDhsDPr9ea6m+VuXT8zun7WLdSWqS7a/iVZl86UeV
- EwjbcoYDAUgYqxbRytZ28iyLIDEVbIyQ xOB+NbEVrZ2sNvHCZFghDIzu+4OzHO76Crwi8nS4V
- YxgSHzCVXAADYK/WuWeJVkkjzp43nSjb/P7 ys8LQso3llBEaheoJYYH1zTJGnk1G8hRUbdMpY
- be685HtxViaS0MkqxOZo/MX5VbB3bsq2fSmXZE TXEiONwlIRh2HXJ9ec1zxb67/wDDHGkoSs3
- q9romhMEs8clx5LSujOflGMMcAY9T2qL7MhWGAjaV gKMo4IZgc596lgaRdUW5uBHIkgDKqJjH
- GPy5rQjTzLhUMkYDDCjHzEDOTmspTcXoJy9nJyh0676/ cUxMILRIYInQLtyXwQWA4P4c1TSdn
- uZlvsw/MH3dAxAPT2reW9AXHkp5DBWUlR05GTWRc284ijkK faFjxG5jGPmB6fiDSpyV2mrXNa
- Lgm+aOr/r+tTLgk8/U3CuIVnj3srf3iDwPapFht0RYmQzyZTcB 068ED0HeqlwCJhOrxxkxsEX
- HKDODmtSE+QUJuIEcQsWZlz05/rXbU0Sa/qx7mKpS5VKOqatbX8dD NvYkF+QRGJWJkV1GAxXj
- j0z2q5bWtzC0zTl/KaYNGp+9gDJ59s1dxc3tspXyc5CBwn8JGWA9+ODU +IntAsjOojhIOW+5u
- HAPvWUqz5eU86opOHJHfr1+4zZk09beViVaMMI3JOS5bgOPpmufuIpEkFrb q89tAyJKQejD3r
- pryWb+yUeIQyyFMyAJwxGOnpxXNrLfy226J02GRdwVeTzjmujC3te/3sinBql7 z083/wAAr3q
- LNJA0YVNx3NIPUnCj9Kr3UTK6W6brkmRzII+MHjcPalNptsRERJ56SFZELZJJbqPQ YotYHi1f
- E2YmCPh26Adya71ZLR7XM1ShUp817OOvX/h/wEaCGKFraOGZ388s6k7iowMYx2NU4rgQ X6FoR
- sUsNu3p8v8ALmrhCSQtEiyxyFQDK0hOG6lTxRDp10IZ5FgZQrAAyDPXqPrirjKKi+ZmkaEY 0+
- Wa19df8v1FsxbmCBmdxJIFI+bgbc7l+pqxIheyjiEgLIMiQdAo5Off0qC0tJFnxBPC7biEGM8H
- gHp9abIJbYykMrEOEx6gL1+lQ0nPRm1PDSlP3anvdE7/AOQsZ/cxXKzu2Y933iCoJwc1ofaobP
- TG kjkaR7h96MxyNq8HHsc1TMN6lmit5UhZy8qqmCAOgHoO9KhLwnzot6rGAAF4+Y/LgenHPrU
- SSlq9 UZKjOrK+um+yv5FqbUvKMvm27FpLhREQQMKACTTrO+ZLy4le1kjJfDLkYGec/wAqz53k
- dJFmgYkY AYcckZqeAieNPOlVQz49MBeGB9zxSlSgoaowxFKnTu59e2/4HUrMJLWUSeXI8kgMJ
- UdVC4JHtWRd XHkRRRxlmJj25zzu6Dn8zWBc7AxigklZ41P2dt/ATGSD6k1fS2mGirctMAc7lR
- uSuR3/ADrCOHjC zb3POU4UeV8y1e1v8i5hcRRAkSpGomdj8rYOCR6VkXEzwaxJJ9oBO5kEXOe
- V4oK3f9ns81xHOIX2 yGJcbjj19OlVwjw2dtJPIszOm4qB83Odx/DiumnTSb1uY02nPnWt9LXu
- EyW6WFgziaGRbfy5AzE4 cZPb61bszbyaX5kxEUrsj72+78vBwPyoSa3TS2McJuYVkTzi3zb27
- EZ6D2qCKySa7bbLHKMsXSMY J56j2q27xak2v6/rc7aNKUqbjUbVu35f8PoX7+8SeKZIkU2yvh
- 3QDqcHA9KLGV5IZZ5C4tyQwI7F eMceo5q1cxzGxePfBsZg2FTB7Y/E4qaGT7FZ3LXSKsTT5ZA
- uNu7jH4Cubmj7O0V/WhrUajh+SENP W/3eZi3TtLaLNENivbFo0bnCg4/rmqE1shgQXE6IgdeB
- wSMYrduWG2BbOBpPJ3QM55Vh1Ax71gTw PNPapPN5anBG7sD/AIdK6qErrt+YkvaUbN2t56/dr
- +YnmpDHIyPDlWZYjt+4meR9c1VJkltfnEig HcxxnlakuIooNyAhgh27gcjcecVSkhlSOWRUkC
- LN5ZQnkMR3rrhGO5FfDUaLjOc4y9dikHmEbCNA 0gl3NlR3GT+tRiUDaTbuzseCGwE9AfXip1J
- YyseGBAPbIHBH1qooMbqgyoeYAE89sYrsVjilNU0k r8t32t8kdUlsVuoTFIrsVMgwSSCp4z7V
- rLJd7fNktlU+YJTlRyQfmI9MVNGIYZVkluLfzIFZBIq4 VgecY9TniqjLci3h8y5iTbuEAI+8u
- fmz647140p871Or2sK1W84Kz6vS/wB3+RrjbdXL3AIOSef4 BuPcepHQ1mTR/ZEEkUiiV14I6A
- c8mnGd2sWZZY0CzqsZUYyjHv8A0pzzXCNeQmBrmB5cK6DoD2H4 1jCLT8jehDVtPTte35kqzQX
- GlqLgxyRYUKFXBHOP1pkd1DaxxIwM7S7nB7Haxz+B6VVd9Si0+YS2 pikZ/nwoAU/T8qkdYY5P
- MmXIVGTav8DZBCn3NPkXXbyMuSHKnJ6Xen/DbEs89y/lSIvlJIG3KBxu x0Hpjj86gjtjbT2kV
- yBI6RO0idw/UqffpVNbuGW5SZkkdBjzQrdJD938u9aE8SmCKaVi/l7gxHVn zgNn0yelaOLhaL
- 0Npy5XGHL6f1djkXT8xyurhpommyTxnH8gaqbrIyQId8L7P3nmtnJ681cgtylk rzj513QyA9m
- J4x6Ais+aeaeaWHy0iRiWO5Bk9Mc/hRDWTs9vM2oSUnzRbSXX+ty3Pp7SqsxVzIso jmVCOWPO
- R6ACoBZzRSLcW+4mKfayOufM54K+2KlijvWe6ltyZMlgWB43Y5wPX0psEdzdaLApdonY EgN1I
- U9aalJLWSsYutPmSdVcr6FgzzedGtzayMLdtrquAy5Ock+gFWpbotcyLA0LpEJN4Vfv4+bc PQ
- Cqcs7z2NwVkUtJMrbMfNntzWmjKkJk3RGV2Ln5OhPG0/XmsJxSSbRdWlTjFT3v0u1+g2W5uJ7M
- +eq2cciBmlIAVm4wRjt2/GsqQSR3LIgkdWbEqqeYx6VekkkUvHLPAkQO2NWTqfUe2eKpWN3Mmb
- kj lNsUgYZwxyOfenTi1FtImkmqbjHTy/4O5enjJZ45InMaodgQ42AkZU+pqaC2jjm2CN2gydk
- TEEhT 0z+NZzieHUVXLeWu5CzfxIR1+ue9ajR3I05fLXdK7AiMctEq8EN6+tRO6SV9zplSbik9
- O3b79RiI 091ZRmHASB1AI68/MTSzwwErLaRSxXEKlnWQ5BLDjj9auLcTW86yybIpnZkJdchi3
- TA7ZrR+1JHY sWWMTMMsuAMkYCiueVWSaaRlKq1OL5fknb7+5k2EktqC0xTy3ljxuHscD+tRTX
- d08hjSNsl0EvHD FuSfyrQkiZ70htsbZLbSPvFeQw9h0rHnE0kUEsQa2kYiYmQ/eG7g/QVUOWc
- rtDTjOUpWV+/b9S1e 2rnT5vtUUkSrJ+67ZQYzn1PvVV3IsZBHG7xKwOFPOAQeT61YlZ/LnLah
- FLILjLA52gDoMe5p6wRz 2Db5lilcgynsrD+H6mqjKyXN+peFa5V7V6X81+DLVvHFDdTXCOVDh
- xtds8tjBH0pzXsLWht5JoVu o8KzEYDH+8PbFTzuscSQMgcxp1AwAvGQfU+9VGa3m1a6FsiCNp
- N29hnA24Fc8fe1ZpTpSm+aSbt9 35CNawy6a5jcKZpRJI+eFI4K/U0lrFFDujTeu5SGLN/Gen6
- UsM/kwKqWsv2lwHZnOUGD8xxVu7jR 1JVJJELFS0ZwNxI5/pVuT+F7Hou7ioN2+f6f8H5EFvCp
- W2XzQsW0Nk99ucVaWN5pJDJCiGRlO8qM L8pyD6msiWXy2ZYZAkkZA2NyV7MPrVyOWZfPLMSIp
- ECj0OO9KcJPUzqwrOL2+4pzz2tvIVKSIykK hDYByOf8aRgrDG4FdodGUcNtHH5mtNprI7S0Ik
- kdPMjPXCg4OfU1H5cCXieQVEKIwjY8hkB4P5mq VTTZnEppL4NvPcrxNehIpZXt4WX92WaPhj1
- z/SoGs70xx3KyxS4AyEX7rBsAH3NWJnuY7xZCnmoI 3RgBwG4yfwFSRqbgHdDLBChKMd+A/fIp
- 8zjroOFWUW2rfd+BTdCxuJLf5JxcFIg/TB4PHsc1OGuI IQ1yBNtc5KjmQgfKwPpTzaW7Lvkkd
- pXBkYK2CGX0/PFNvINwaVp/KcltwOSFJHTAo5lJpHVGPtOW Mne/lr6dyT7QPsCXE1vPGZXRkZ
- m+UBe+KmeSO43TAMZCxdQD9854bHpnis37ZdR2Sh4lmIdCrhfk K9CMe5qSVriWNo2spIY/NVV
- wRkDOSKHSs/8AghUw8IRTqaa9/wBN/wATShvN9rcK6skr3AcDONoz jafcnpVRUabUlSeAr958
- jgAZ+Zfr3HpUEccaXL/aIp4l3n5WfkAg4P4VU+yqpRFuDCxUMA75Jx94 frmiNOKvZi+r05J8r
- aXn+mv5izI04k27VgKAmQD7zZ+U/j0odbopbSy3lujPCxddmDt3YYdOvvUr xRpZKFl8obtrK5
- z8x+6PwrTginuojJdQlvIfYuBjjow+prR1FFX6HVGr+6TWy6dfxLtoQ8c8X2eT 7OkyeSx9O4P
- qfeprmNpLxljjkjWM7cOchQT90+5I602SZ4QzbGti6kFW/hHcmrlu9r5EihyAWDBm OThTwf61
- 5km0+ZI82cb+9GNr+v8AwSsjW7xKrPGyhz5aIMMFByQfU+h7VBHpc8eru0dxHLZyOZEB GTtUe
- vv/AErQC3GLmQJEYmkykioMHI5xUNtdS24hi8yJkRSACvKg+tSpySfKcSlVgpcuvf8A4HZ/ Iz
- 3sofNtpGlYxmDLqGOck4XHpWqQ8MUv2dhIS5IBGSoA6VmNcSPJKlvbmWEbTkAdc9KlN1eTSNLH
- H5cu04GODu4PFaTjN2uzqrUas3FSlp6qxpwSvI4Wd4drRMyoqYPHU/TNF2sc9rCXmEbOvmSkHA
- 3q OMCoYpDECJ4jE8b+XHuPJTbV+Kc3S26SCPCxkAhcY9j7nFcsouL5lscqoTTvHZHNCKMooME
- ql2LO XIO1gM49quNaQTWSTw/61lO5Sc85A/KpL+5jgjiT7LNcJMPMZozjcemR7Y4pbeGO2mmd
- ILhTFMUj DyZGGGTn8a6nKXKpHe4SVPnWj/P8b/gEsSy21xAW8opdGPepwDwCuKx3EFpcPDNI6
- CYhiznofT8q 1kHnRndIibdo6cZHc1HJN9uecS+SUaQFDtGFGOR9cinSk1p0NMLGFNuC1XXXVG
- UjTzvPBaspjiKq X25Bb2qi9p5UYt0mCxKrHaoOTg7uv6VrhJoZWeFQ2VLsij72Bzj6ViPE8ln
- 5aiTMLADD8vH1Jrsp yu9NEOrDnqe7FL03+exERCZVu1czcFGjB5y3IJ+hqfylluXdgyxRYTzT
- yGJ5J+lQ2lpH5zXMtzGk RdQuc/Nuzg/hSSSxQ3JktbpWicFgCchmXjj866Hq7RZHs4qbhGV2+
- rv/AF+ZJILf7SjxSLKm1o3d eiEDof8Aa96iS4nuZbdSxiTYAE/vAn5vxz3q3BePFgxRRu543b
- fllJOCwHpiohcyGVEW2zFuCJtA znnHNSr9UDpTjFxdr921+H/DlGCxVbi6E0rwx27vlsnnnGK
- c88aSRTiHzI1j2qpGdxI7+pAqw0N0 IpYxbSZZjLIzHIBTsfrUbpujhuVjK+Y6FUx93J+Wtea7
- u2b/AFeDSd/x0/r5kSNBLbRx/Z7rzEQq H8zO7v3FSxzTK9qIpYtz25Z0KZbIGdv5cita68spc
- KIDGiSq0kx4USDgL7fSpY7ONIZLiUokxLsr Y+Xb04H6Vi68bar9Tjq4hNezcbLp1/H/ACOUMk
- LF0McksKjEJDc7cZyT3piQGGO2mjk/cmPzJmxk K5zgfiK2IraBr6KQvHHaNAAueoPZT7ms9ra
- K1tJo23AvMHkjLcgg/d/CuqNWL0RTdNz9lDVf1tcv wS21vZtLIY8TyLhyPuqO361QnfL2Yhcl
- fIkJzyCVOB/Om3TRXcSBW3K8YCp6MTx/I1SVolZISxQC N1aUscHBzwO1KnS+11PIjl6hLmb6/
- wBdmagF3KgiSMQxMS0e5c547/Ssk3l4t5AMQSBLchMx5BBH NaNtcJ9lZ3nWSE4MajIOAefzrO
- iWWK4SR4XKgMoyc/N/hz0q6Ss2mkY4WlBVXF2t06a/PU1bCVfs QlaSJi0iK7BPlj/ugj1P9K1
- ZYIbi4SNJovNicmIxjb5qLyT7jPrWJBNFFZkeZG7W7hCQuAQPUdzz 1rpDOv2bzYZYJFiKoSq8
- juR+NcuITUrouvRlF7t32v8A52KsmosWeQwqULBlPGNp6j61UWSG5SdJ pUw5fqc8Yzj68VOLy
- T7S7xpDGrqzr5iZDJ3I9MHg0wWc19bQrcyQRzCMsDGu3GDkqfU4xUxUYrXQ 6aapQa5vdfe//A
- /UZAtumnyyW7MqvIsgBbOzj7p9zUEtrbzxu2HkllK7kDcqWPT24zUsL28jQlIJ STGfM+bhSTk
- DHrxVi7zbTlkhIjfDD5uQFP8A9c1fM1PTccm1L3b3/MxbpbizkzDHHIj3BJUqDtbG Bn+dUpIJ
- YLaWQyorMweTcM/MOnY8nrWkZJbWZZfNRhjzkjcZJAJH8jVR57O709AxaAq2JWz94kHa foK6o
- OStpocEsP7KSm43T3dr/gc9crbgL8yKwQDk9fr7mqgjWBZlIyBNk4PtzjjiraqstpLK6bTH KA
- QcEk9M9OlV1imS7BWFikZy4c/qa9WLsmrl4qUXeUUrLa17/wDAOhS3idrkROWiVxtYEkYx9PWr
- X76ZImeVJV2NIVHDe/bip5kWSKFrGSKS3ZGxtTOecdxTzb21rpjSeaY7xNqKrHOVyNxxXmOpdL
- v/ AEj2VOM6NN21u/Pv5XXqT/Z4r3S4HRXiR2yuT1wMAfnTIEjWRPtNpd206Sqkm+T5ckHAx65
- rVilS a5dFjK2aTO7ODgZHQD0HtWJqCi5htsSMJJdshwenYA+/vXJTk5Plei/I8K85N05PS+//
- AAb/AJml HLMk91ujIlR9knmjI3EZHHtVFoVluoVkP7lwfPfP3ZAMqPxqovlSPKDK9xlT5SKfm
- LN6+uKvwyQW liqG1uHR49wkL4Dkdx+NaOHJtuegsBOLUqT956WVv1ZVaW28i38qLY+3bMM9WP
- U/hxTGtpDFKEn2 Iku0yMcqVxk8fWpzfxNaR+TAsOSnzNzkscGnPLK32y3NnO7gtsVT/D0JP0q
- 1zJ7EUlUhO0rr5jJm uLZpJAj3KSyqUYH5Qdveqtuw3Ol+6KisAQPvMcEjB7dKfarC58ss8LjD
- RSO+VwMgnHua0T5uIoo7 ZZJYsDO0cHHOfU9vxoclHTr9xq26TcW/efy/HYLWZrlZhBiLePMVs
- cLkfMD6nikiGYo7gMDH5REc SnDEHjP5k1HFfvLeQSyRi0gVDHtK/wB8kfpTbuxmmkhCo7xJhE
- MZxwO9ZctpWehjFxpNptL+u/8A wRIbG6MqxG1mhEaFXY/xMORV7SxdW0EqSRF98yy+WR8xJBy
- QfQelJDPMFkR4pogTtDs+QST1/lVK ZRbXFtLcSSblRl++QGcHkfkaG5TvF/1+JT/2jSdr9P6u
- aEnzTQ7UCyJC8RmZco2DuLY/SqSQKsfm WQaSVnJXccjGe47kc06K5dZhcX0TvG5DeWnGMcMPy
- waksJY5dQwgEEQlLgscjOCFH09qVpRi/L7j aFOvCLVk4rd9PTq2W4kuPtL74SY3wVfHAIOAP8
- afLcXsQe4VPNLviUovRs4A9qtQzvDb2gkIklEB 2qB1HJb8fenpqdrJFbspCRSr5nPYqcCuVyl
- e/Lf+v+AU61Tm5uXTyJD54M6ytC3lybfucqQQSfwz RPLK80uQgWONozIVGCx5zj6Uyb7MZLlo
- JDKJL1fM+bOHHQfjVq5vHgtC/wBncu7FipUcnpWPVWWv 3DpzfMmrO/TRW+8pvJE+nRvKHcfZ8
- 4DbWfoAQewqhNNcxDy4YTIF+UbhnIzyw9h3qaS6uy7pGIuQ BsMeTgc8ela0cSzzM5jc5YlwP7
- uM8egz1rS6p6tXHUgqfxK9+n9aBaWVv5k7GLzlUlUcdJVB+Vx9 ecVcT7Mt0ITblRJHuweoxnm
- pEgRYbe4wzDyP3m1sB+eCPQe1WDPFJamKKFt/mNnJyRnGPwrgnUcn 3PLnV5p3SuVv9FaCAvE5
- +T52zkEnnH9Kom3RXXZbyKBGfMjBG5DnOD9KsraSnWopI5VEPll2BHCl e31q39pkkmW5ZFMTK
- wIC9WPb8TTUuV+67nVGXsp2i2/M5mC2aW+mwWI3bkTPO3vWza27PNOyzKFM 65UjO4kdR7AVY8
- 6B7u0aTbDstnjePock8k/Sr08scWnOPNgMaDYsqDAYsBz+PQVVWvOVlbc66let Oy2v3X5abmN
- NCdrPMiXAlPm5iXBCrwxPt3pqxzM8pBjWJ5FBbb98spwR+lIryqsizXturyEMkeOY x0Kmo4Hu
- ZvLjjZRIqcKRnvyPqAK0s7FTjUS7tddkhs2ltBpA+0TIo8xVQnIPbP4UwRLE26FWuIw2 GZTxk
- HgfjUbSPNeQFXaZvL3pk5UYORkGnia7hiWVpYHdgIwipw248vWiU7avUuMJxjac1d/L9P1L hJ
- DpI0J80Psk5+UsevH0qA3EM4ZEtppoQwSNo3xjjv6mrsYSUtCt1DJIkiqzgcO2cAj8MiojLEru
- LNDglh6hecY+uM1imr2tqYuUIVLPf56GRskMikSBlBRDgc4P3m+lW5IofPWNLa4uY5d8keHyCA
- cL +dF2YUQhd8RiiKKCfugjjPqafGzC2i8wNskw0YU4yE9Pqa6HJtJ/1/XU76kuSKneze3T8v8
- AMy4I HjkmsGhneUKTjd90jt/WiyEkK7r5njVxvJYnHcZ/lTb67vWuDdeQ8UUoYqcgHAAB5+tW
- bcSJbtdS zQsPLKiNhk5BFdEnLku+v5ndOpJ0730fTe79dbfcLFfLIzSxhXjT5UDc/eHU+vNVj
- cSmfa0lvMI1 2nan8RHBHHTtSKY7uSeGYpA7EkMi7VA69B6GrL2MCWMsjh0MjAvLu+VSeSMe+O
- Km1OL1WpjH2EZJ TXvPyX/A/AzUWzaxjlh3mdm3bGbOzPAU8dfetlPMjs1DXCgRnbK2ThzwQR/
- KjTbRYZWmdR5ckqSJ nogAPyn3FRy2NwLxXhmAKYJLDKvzzx9DSqVIyly3HXrqS9k3dLW7tf0v
- 5fMvvtvrq6kmnVVSTZGT 0Klen51Zt43kntRNGwLxlpQBjaQpB/AHH51lzSWa/wDLUSyqGEZQ4
- G0EZB9W9DWnAtvA8k0XnhGy V3Pu9MD/ABrkqK0fyPOxMWqVrvXb19ehBsd47Yq0hRYQow2Bub
- jNPQl7x4nXyGVRtZ+c461LcP5d vKDHJGuVeJ88bVOSPeny6fmae9SUSjcVDA8epI9qjnVtTOD
- hHSW729fkTzf6FaFlCFuhIHTnJX64 70W4jw0pwskThGTPIJ6H8jSP5aWocXMUgB29M7icfNz2
- p7wbtQQgNcbFIdYuCGHTPrxzWKempnCE akL7W666+WqJhMSssZ2OQ6lGPPANSPKRbuPMiHmfO
- WC42lT938azYS0gQnbEDAWkdhw2T1FXH+w/ YZD5hKyN50eG+8FHy4+pqZQSZ1SioOLStf8Ar0
- K017KJPO8geVEAhXH3R1xVhVluWX7RGwjdgz4O CpXufYcZrGaRRbiZriL5tplXHQt1P4VoQXs
- sUyxxEeVENg387s9K1nSaj7qKrUajheO/4f5El3Dt VHBG9Q0bleFZ2xgj+VZPkJbm4nud+2OQ
- RuqsRt68fU1dlmil3iSKaF3YSmTf8oAGNuPU+tVry9jF k0qrneA0isc4f0+taUudJR7jpzlC0
- Hd839b3IDdwR2iraXC+YWG4P8xGRyPyqvEbeJ4zI4RPn2lu SVIwaagt7qEGK3kSSRW3ZP8Aqy
- TkZ/n9KpRXzs8CPEt4FQjfEMAnJOefpXVGlo7fMyq4eN2438+9 vUW53w2UfmhZYQyD5FwVOOh
- 9+lVbtPPs9oEcLRO3mLjld3JrWVpZURoYtxjDFiVyuT2x6nt6U5Ve ZSssAgUx/vC4+9Ifu1pC
- py2b6f1sFCm5csnDZ91+Wv4FO0lkP2VXgLb38zCjBYA4wPTHepLu2kij gMsn7ndIHK8YPUH8e
- gqMTC1dI5UfzfNUO3/PItzj8qY0t3Hqn2lIpJYd4yjcg844p8rcrrQ2UHOq 27Ra1t39GyeOcm
- 2lHzI4uN7BznjHSrEt9bHTohKoDbQItoxxnk/h0qa6aKG4LDYYEYxbx2PUZ9fe sc3MDCRZGjd
- 5JF8jAwGx1I9s9qiMVPWw4uEoqXLZL8zQdI557p0JCGZQqs3bb/MUskz2LGUt50Yj 2FevIxx+
- tRRPFdGVlina53kgxnClj0IHpxUM7XE0GZrWR5SNshU4G8cjj3AoUbuz2FKLqe7KVl2/ pjEu7
- c3XmpbTJlsIHbKsDxkD2NYz2TmeYtdR/upk80nPLHIJ+ladzZRTzI0YkjGd+C33EH8P1PrV SS
- BHtZ3Ln95IHIzjO3Oa66TivhZxR5HUcU3d6bf53IpbB1uYYUBlUgHzE4A2nj8+aoKlu08quJEh
- MxdVJ5GOhz6Gr/2KYQLMPPMTAMpD9CORz6YrMuMpcyTtufcu4lRwpbtXTSfNpzGcaaqzbdS6Xb
- Rk DLEsyqN0rvtKqnGWJ/kOla1zqEaSToskasrOrgjq2BjH+e1YQh3XrEllKOwVc4YbR8p/E1a
- eKSaB ZWCsWctK5GAGx0rapCLauY1pxqVouT08/wCtSxZvBJd2ql0KgFWXH+szncTV1PNKyeQ2
- 2MREyoeS GHGfyrIitXe8hd5UiD5ZXxhffFXY2FrGro/zAb2JORJ82Dj2rOpFX0dzedOTk2nfT
- b+tP1OgWT/i UZbaJFJjDkcRAkZQ+57U37JCHRI7o70jZGXeST3H6Zo+8t/JJGTbNciJE78/xf
- nirEXltHKsIC3O 5TtbrgAg/n1rzrtbHBR0jyp63e2y9W0Urf7SQscEDKe4YZyOoNF7Yl5PlMx
- hRVZZA/AHX9e1axVZ obHZPH81u0jEcEkfKD9MdqghQG4Ehv7cRFSAMZBwNuKSrO/MtDojimov
- m/BfqYd3HFFdrcRuHiki YZf5s5xjHtUVppsrW0bQTWzlSSVCfe54P4VpzW9tcae9ujDeg3I2e
- EC9Aapz+abBIVYBpR58hiO0 oRyV9s10xqPlST18zgxbVSmrXu9/T+vMyXu0uXmkcxQRhXQpsx
- 8x6E/iKyklZZnmEgEj4Vwy5Bz1 NaV6sKziRrZ1idcnkDg9jx1B71nypHJYRSLHIGUlQikbj7n
- 6V6NJRS20YKFKlBK10/JfodK8UElq 6wyOJlYsoUlRjAHSmmOK3voBJvS6nQvG8pBQZ4xjvnmp
- FW2jluZEY3OHOxIzjC7evv1o+0tJZWbM Ix5UJGxly5HUHNcSb2W3/APUbcY+yp3t0vv8/wDgj
- jBciM2rEq4bZtBwQBySfXHeljHmyC4Ks6j+ FTjfk8EegB7VKfKnh/1jtO5G1QeSp7/l1ojTTp
- kSKJblZTGH3+ZwNp6EeprNy01M4RjKDTbv2tt5 7leO5jZoJriKOO5j+Rwq7QGJINXL+xiCQfu
- p4lUlGLP8vr+HFNheT7agMlvGFVivmx5xu9fUj1q9 JmJGMVwkzW6FW3DIJx1qJzcZqxnKFWlU
- XItfV/5f5mNcwSWlmiwtDPBvyzBejLyBTS+5JJgksDOn yFnPzBjy30BqxMsdxYqROFgMy/N65
- HLfSmzSrbPFALZ4oWdkidzkBDjJ/A/zrWMrpLqdtFJRV1eV 9/8ALTVjEV5LgrZKtxNHIVXAyG
- Xb1H41YRJ2iImtrlZpVBQA/fUHnHvUaJNCzSQQyOJmDxlDjIGf 5HrVxms4LaLzXlW5KsQ5ckL
- xyuPfPFROTvpr+ZH72FW9PW/rf8HsVknji2IkLRQXLh4Wl5A4OB+d TNpkku1pPNG2LaUV+WOM
- 7h7DvTIHeS1SV7dzBkLGpxlM/dJPsev1qKWVxdXXnmSBvMEMkjMdqk9T jsP8aWvN7un4mNOb5
- 2ua3fS//BIY3ubuztrfyZHZoWmQr/EVPX6DHNSyxOGEs9tJciU+d8hwMjg4 9BTJT5AFta7yYV
- MW4HkEkHH86da29691GtuGh+R4nMnIUsen171b0V9EjWtRhThzSklH539X1RXc i41S282KWNm
- fIUtwTjnj04rahtrIW7SSwyxRJOGbEmMjHB+maptZXH2qO3hbNwsR/fNysijhmHpU TxXVtEoE
- c1zBIygOG+VQDgZ/Opm1OyUrEyp05zT52vJP8f61NcLHBYxRQM0beUwfzTuKv1xn6Vbh UCyt4
- /s3nTGAhGQAKQei49TT9gF0yNxcrIFkJHEjjqwHYAcVPPNGixi2t5S4Qyrls7AOgPvXnym3 ZI
- 1dRNxhCN7df8+5UNusskccLranIEofna2MDPv71PJbX9sSrJ9sXaQAg5C45bmkE/2lIPIZEO8j
- btyQgOSSe5689q0rkiJLnEFzK81wJoiH6xgcge1RKclJImNZKdlq/P8ARsWxs/tItjHGViWHy3
- kb uwPyjPvXQWunTRzKzRs+IWLIvWIgcq3qcHNWbQRCVLiCVIkGY0BGQQ38WPY55qaEiHEby+e
- w+UFC RkdyfWvIrYmUmzzMbj5Oo4rbtZmb8jWcSRQMsbIrKzHO7HIx7VOkAmWSVmiEm8A7RjKk
- 8/4UkxtV sA2HTceDu4UHinXSmYmJZo3hDEsYhgnAGOai76aEwckk07fL+v0GSR2ttdTPIrJbq
- pC5bqvr+vNV AlqYQouYWCgcgdxnJ/rUpWEW6y3SycKQkecce/8AOqRikayUzyQ7eMKi7TKAeW
- B9K1grrVnXCPNr KTv3/wAt9fmVWgjlulmKtMoXbD5Zx5ikE7v05qpLdwtHG8bKWmgEkkY7t0y
- PQD0rRt1s3uTvuDkb hEVYgHGaq3AspLm0El1axRtCdyqMEljkYPpkdK64SXNZp6HRCvFN8zbt
- 6/1+RmKYpWCHKMw2+Y/Q DNWJLSJLhHt5wylJPMKk8be31NQrLIbtX2KiKv8ApBYZCzHoPb6e9
- ayXcxiJuFhWODdHIQgGGYdD W1SU4tWLq4mtTa5Ze6+l3/X4mNbllTdEolmli85IQPmAX39B3r
- TTyzbNdSyQJCZBIGK99vAH41Ri vrWOGJmjaFzHtyT90McYNS3rF9IurePbI0UwChR0A5A/GnN
- SlKzVh+0nWnGLjp3f/DWNKSSBHtnt kVSYnMuRnbJkEA+9c7LdzvqLRbNiDIdlGMs2MVPdS3U1
- kB9imYyHO2PghiOW+g71KksNskSzTRNM snlv/wBNGYfeHpkdKdOCgr2uzeKjSjdLmf32K/8AZ
- m7U0Ed0paNvLlDZbkHvTZmSOW7V5N/lyHy9 v8ORlR+dXLW5Es/kraTlMA4Vvm3DI5PepBOss0
- YjeAyLCQYyuW57n3xzVOc+b3tbG9T2/teaetku 2n3f15nOzsGjh86YMnkgTqB9yQcge2atW5d
- pJPNVZGbcz4XhG6Yx71VdbaNykkcjR7Fw4PDHPytV a6kuBEQJBIZZczsgxsJ5H0yK7VDmSS/r
- +v8AI6qlLnSSW/loWIzNHqJ2wGOMxHzN3Pl+x9yeadvm VY4btWMUICMc8FuWH4ntS3n2iF5Xt
- juLzEqMZyu0ZP4UtpPGIkjvJY95ZDAccMncn1PvQ7tcyX+Y qk5L94o3W3mrFkjcYE+2Rwqw3r
- GwOSMZakvZE+wLIkrPsXaWV+BuIIH1xmrS3cRnjMUSFER9pwCT u7D8OazJZbCKyYW0oeR1IjD
- cgLjv754zWNNNyV0zPD0n7SN4vXorfj1RPbJFNqiyFCsETPGiHqAw 4z6mty2sc2v2eAtNswy4
- J5VRkj6msA3jtpaRRQP9pRhuUdcFeSfpVmKaeyty32pIs/KuQSX28hhj tg4NTWp1GtHYmthcR
- J6aNuyTV/yNCJctcrIkquVLDzGyAD7VMZWOivbPcwxokiguQcOeuR7VSGo3 E1lFODEkyREOpX
- +8ePy71NLeyyQwwBrc+c5ZW8vggCsJU5XV1/SM5UZtxbVrP8fmhyJPsYRlZZnQ sPl+UnOOBWm
- rSxyTysrsqvh0XggheTWa0837mfckcaRNhQvLBiMVXlCxz3iolzJL5n7tRJ95QvJ/ M1Dg5bly
- oycXG6Sfb/M2J5IDY2j7WMZhdWRTgglhg/TFZsdzG7/Z0Xc0LbAfUPzx9MVFDNdyafIk 15av8
- 6gFUxwetVJLuK2uCgdBKpKSP/tbTkfUVdOha63ZFPDxUOTVy+f9Mv3cy29rEWaCaRxtjVVx lS
- Rk8+lQ3c5js5oyvy+ZmCQHG+IEEsPoaxnhiFnBPHeASBBhXJJx6j9aZDe3YuUjhQ3LDOwlcgIP
- Ue9dMcNomtbfI0WEcEpJN27ppFm4u9QDgxbPlOCSuQSev5dajeW+NySINyZVJSRkEZxkVHNKUn
- ka WGVYGLgNu4JIGPxzUFpBcRFC8u6dkYFM/c55B9z2rdQSjeyOmc3Gm5NJX7f5pluVVW9lSW9
- S3jjk ZIgVOcDkZx19KjkK+UrmaKIgESEJwHOCBj6VTa7K3EMXlIUCfMsq7jke/wBOKszzpc2i
- iXbbAOzy REfMxXoM+pFVySTVzmk5qUVKL+VvyaL9rLEhWZYpzFMTKxD8DA4B/DmonKzhV81ow
- ULIxbhj2/Md KwBeh3jyWWIRsAo7Z9cVde33WzSrcoYUKqNvcgcfzpvD8ru3Zm8KPLJOctXt1/
- ployKlt5nkTIZH BMjtlY8cBT6sfWozvkgaSAkSmQgJnOwD5Sp9z1zSERGSzHnfNtj3J1yDnKn
- /AGj2NTbYrbVCZCwb DeWOnGO/qQTnNLb1Oec5xk1zNW/ru2KLq7j1CO3kUJbxKxl3LnJXpz69
- Kr3ZkS7t5YIQ7EBmXbwp H8HtU0rl7WFVJDxoVaU8qWHr9etTKXWwgxLHGOfNaQZw/b86Sai07
- A6iSulZy0tr+lyzZpm+keSG W2mlckZbAVQOePY1SVIoLiKeQzGMx71ctlW4NWR9rNgkp+aSWQ
- TFwOEwPmX8ahlkQ2wQxyIrI24u 2QpzlR7VlG/M/PQ5YSnKTUldS00/4JRk2/YfMS6XmMKUYnO
- abb28jwWrNPHejawW3hPzEfxdqc8g WGJnUGT742jhR2BHqRmpJlu/sYLIsXmHMRjXaVx2/Gum
- 7tY0rKrLljJJLpzW/TVfeijbxzPcJaZk gRUfaHyfcCqSR+feO9427LLuVBjIPf8ADFW1t1Qkp
- cExfKZctzkjgA9uajuZJEt5YJAg2kCQFeck DC5roi9dOovYTm3GLS032t5ruZpk2agZIysqI5
- 8wj+LnsaS6jl+yvMsqxQs4XY5OX78fSpIYmkae IzRQqCMswIAI6DP51AiRlBGHMs/mFozngdu
- hrp0T9DmxKUPdWr6v+tCS4kYmyWORXYKWCqMbue38 6sCRTpxcTRyTEkyYHA5xgenrVaRvLuFO
- 1WAUJuC8Zx0+tOjjWR1jlkS1cgMA3Yj1pOKshQpRp8s+ ezWun+a/yNpLactG9pL5yR7lcjoWx
- xx6/wCFbKb5Iw8vyXQAPyjHTr+lchBJKJZwJFjZiX9MgdD/ AFrRg1OOGGAtKX2hg8hP3nPQj2
- HpXFWoze2py42VZNe957a/MkDpPBFvv7dEKFIwoPILdP61cgtJ bdblYrqCRIpFiHy5+90NZNq
- VnhdECl1faJAOCO5A9hWr5klhbpOgEqJ+7wf4snhj6nHSlVTXur7t DpqYia5Y3u+istya6S4h
- uJIIjGY0QgELyWQDjPuaw5lultvlcCbzQsh28JuXhTWq2pG8SVLUqzRz FI8jqrcA/hg1hNNIy
- l4b2CaKI4AUHLkHntRQhPqtTzpRlN++kpf18iCeRxbrHNbSkhRGxLdz0OPp WSscsUJaUvEQAO
- mMcnj6966S7ufOhVZbZ1kSQkkHGQCOPqKwl2/bY/OLtCGLvHnk7T/9evQoSfLs JSbV+W1ul7/
- dZnTNeNFbzKluqBZgWyvP+zz6ZzUzXl3EZHkaBi2cBYxxhgM9OlPuILq3PlWuyfzX V5AV3E7T
- x/OrOWnlMV1ZOJtkjuVwNhB4Brz24WTsmvxO6nUpQs4QvHu7N/dYoSTiRproJlWlYAqc BvlwC
- PQe1W7KSWOKNTPaW7RgQMkiZbkZzn3qNXtLWeASq0YaFlkLHKrJgkcf0pVFxENNkhCSzqg8 9C
- ufMGeo+lTKzjawSq3hpv6f5k5trmCCI3SiS3SIq4UYYMTlVJ9apmK8DqL0eSGQxsQuOpzg/wC1
- itIkuJiyyMokJJ3cSFRww9u2KYriW6HmyJmJVKKe27hgfU1EakrHbhq1SVJpfr+C7+ZS+y2jzz
- /Z pGa383aELdFI4P50XKQpaRBVcDzfLAdsjkjj61ZkjhuLF2bFuMt5YPGUUjB/GpLSzmVXKXd
- pHhw6 iRMjAHT61TqWV29iKtSUaampu/8AX9bGfHebrgxIDIomC8H/AGuMflV+5lFrqsTAQfZ9
- hDiVd2/J +8PQZqjdtb3Ea4i2Y3LG6KAJAT19znimxRQPdRLdPJBbDCLFK3zg+59iPyqnGL1a+
- RlUnT9m5TXy 6l26kIt5S6mTMhwsfy4KAY/Dn9Kz1xc2yKjCOUxESLLyGJIy34fnS3ZjEc6rIZ
- WlYPIQeFJI/wAK fHbbyfsuN2HQhuSpyDg04JRhcKMKdPDqUtLvT/gq2w24triSC0bG4lmMmwY
- MhHHFWLWKMX8qXCXE McUy/MZMbDgkbv5VVNtqNvbRtLudFfIC9cdx9TWtJcTXFlKgVI3aQCRS
- vzO4GQ4/2QO1TOT5bJ3X f5hTs6Tho0+q6fn+Rbwz7rqBiwdhtQHDE4wCPRR3Hesclor17e6lY
- pEpRsNgNkZU/hUTRPa2L7pD LEZsIQxAIPP61baGCS9SRrqG1x/BMCSejDP1NRGKj1ujehTVKi
- 6knfpp5en+RHAL2c2+JhPkFW2A 7sHhjn2rZjuXs794IrW4DJKGVnbIwOpPtjtSabBDLJLMZ0d
- pJgz+WdoBHOAPT2rZM3kKZLsxxq6E oxXJA6nPrziuWvWTly2uc7xidRwUbp7b3v6LcoXzSyvu
- tzEkqnZGqr/A2PzPfNbCQusUJuIZv3YZ XkLfKp4AOPSs2RJrmzhbg74S5KDBBBBI/lWpDIjrZ
- ySCXzZo3Yqz5BI6ce1cdV+4l2v/AF+ZtWqR 9lBJdXtv8/8AhjchV7e0iRNqiMlWZhnacZwaUs
- 32i3DK5LwuGkXgZ7H6VB5T3Jji83ZuIwSevHU1 qOYIwY9rI4BIDH7hIHJ9h/WvLbs/M8rSMny
- u79CnbQw3CwbW2ybD9/kA+mPWpt9qrojq4DIZDhsH J4FV4TCb4EMZMHcGXjtwKfI+6RGCjzNm
- WyM4A7USu3qQ8M3O8l+NipaBJJYi+XdI2JBOeQDz9Kbc CCaCJF3uJ4TchlOAAABge2avwm2ba
- pV03LyxPXNUL0R201vFFHJEOdjPyAgH8s1alep5ik4Kr1uZ MMNs7uQdkisNvPG053Gsm+2iJl
- FjcSSkB7eRWGAg6j61o3UN1FDHPbzwgxPHHuZMg7vX6Zrnr2S5 knMRt7hzbuYo5I3wFHUqfUm
- vVw8OaV0z1aVPmqJ328/+G/MezQeZcl5hHvl2xqxwWyAA34VKY4Ss gSSaeRkIWNZOZFAxn3Oe
- c1VeaRlt5tsUMMhxh1yVBGAM+oq/9hhiiWKKC5M7hN0m8fIwPK/1P1rp k+VK7N6mJcY+9Udvl
- /wH912YTxQp5dsjyRfu8yLI2SzAcEe3t71bt7jVPskKRRrMWPmSAL82BzV5 NPsY7tzFvlAZzE
- 5bOAex9cnpTo7M3F48ipM67P3hjbblm9PbtVyrQa1/E7KTjVpvW6Wuv9Ir3N5e nT55HcRbrg7
- WxjAwOntUomhvrJ3MLN8xMu3jL8BcelTNbuLefbNCJBMPvrkNxyAPyq6ZIksWa4tZ Cjy4Pl4X
- aeOPqOKxcopKyHQdKnytxW/R2/D/AIJSVL0pPH50UUyFTKoTlcct+XeiXYtpcNBPDeP9 oBzCu
- 0le34A1ckslkW8uFuCG83ZKcn5sjjHpVBfszaYvlH7ZuTa6wHaykEYpKSlqvyN6kaU7TSe+ 1v
- zf/BMp7l2ZCQqqI5I5BjqScZHsDT7UyyaUImjMilt7FcAkL8tWWaKW/eNDDEFQhXddyk5BNaIk
- Z55YkeJEO5lkC8bSMhfrXROVopWOuvg5WjHlt1/4bb82ZVnbWkNva25lmKMkjuWk+6SeM1cY2t
- vJ bhZYXuVRvlIzjjj+tWo44IUiZ9sJWHGJOcfWnC4tLnTZVKpcZUHMQwxAOKwlNt31aOL6s9I
- 3fL/X l+phxahJmOQRoFVc4C/eOeSPapkurZp43lFs5ZWCgRj92PQ/U9KW9dxO0Nv5SBZEEa4z
- letTXTXD q5MMIlEu8ARhcjv+VbtRdtLXOmWEp3Vla/oVbeKK3vspI12PIcuqNyNo4qHLJZW2Z
- o5IjGu4Hkrk jI6dauXCEkTQARylW3EdtoHB+tWYy7i2nmmtWUl96rFgYI6/hQ6ltdxTxnsVzL
- 3umu+nbQozFRP5 CwSiMtu3bzztH3enc+9RLcyQPFdXGxp5l3sgGAOcDA7Ag4+tTCVmW9uIyAk
- ZEa7udw/vCoDBMzQz KyS7omH1PqB6CrSj9o1UKUvebsn66mo95b3GnNFsks13lSJD9zJ4U+9U
- Li2eOxMyieLy28pizZJG ck1Tfz3t4g0DhRuG89HJHWlW3mYuIZfN+UERHJyD+nFEKShsycLQk
- knGdopu63X3/wCZZ85/s58p WKvIuQOoyeB+VV5rKb7YZXhkjZZywRuScEZzV1rfzLGdVtp7aS
- OT+Ju4HyilslmUu80wxIoMu/kr Ic7R/wDWpKpZNomcpcjkpK35kN4Ynu3+zxqdoKy8fKWzkAD
- sKqo0RDeZvtJScE5OBxnGBWtcRySF EaAzziEtOIgV3MtZEu0DYCIzNiY7xnHHNVSacbGkJNwU
- NYrq9391/wA0WYXuHhiijMcjIoA3LkZJ ycjvVhLWV7QR+Yq3CFwXPTg96o/ZX3maDe8Xlneqn
- 5iD0IqOWOSG2th5joDAXkVmJJyfvfShxTfu s8+vh0q0ZU/h9F+P/BQyWKKDy4/OWSeYbSpwQT
- jqOOlUXiuIxEzxtDuGQHGenWtT7Cjo0rFmiDKE cH7q5xj61WvYTFfqJGJXy3CIWySO9dFOor2
- vc2dSU1yOrdLo/wBGU5FMke7yEy2Wyi4BPXjj0q2b HzLT7XGXSFkZgm77uOn41LBaXEcW1yLh
- iFK+WMcdhVwQNLa2xt0ljlEJhCO2Qx3HPH0zUzrWtZnN KrGEocsklfW7v99zDmgkRIudkkij5
- Ccscen4VLHHJjO8SERs8Z6/JnB/OrVxZRqrRs5meINgjPX/ APVTrG2tklijIfa0JIJbqcHd+G
- KuVVclzrrynGl7WesfSxBCyy2DkoyoB+5Qnk5P6mtQWUV1b2TW rneEIcFsjcTwf51Xdke1DR7
- PLVAFA6/L1Oakyy20zRo1u0rBuT0GPvD0ArCbb1WhhONSdOEovlV9 P+CiFI5YrxYyzyuoDqgY
- /dU85+tNMspke8iZZirmJQBlSCOuParUVs0aoySiNVh2O7jOT1BB7Crm l2s728s1zA0ZNwGQb
- cZC5zUTqxinJkYiqlK8mn36X/Az0s5fPgYSJJCIXjOOrEdDTjAJtJjledky N7ZY/K68Bf8AGr
- Lia/ZER08txuEqDCqc/d+vGKz47e7lmkS4heIOCyjphicD9KFJvVuzRwztOfvS St+RkYJExI+
- 8CWAOPmHQ/Spr0rLpY8weY7Yw68BwMfN9B0q08KxQxpG6MxXLA88/wj8cGql9HOsc DiLYjuzs
- uOIyB8qn9eK7FNSkjoxOIpzUPXt/wCO2kaC4fy/LZgjbWIyGz/Fisvyyk8RdigjHQjBb vitq3
- eb7RGSkNvmNkMjr8vzdO3aqlzHi6toHcOIoz5jr0c47foK1hK0noYctJtpU7d3e7/y+5me2 Gd
- S2A75YA+vYVMd0YlKxsZAWBJ5wMDI+oqJ4gm0eW8cQbcN/LD2zjnipXWUopiZtpGVXGSFzk545
- rdu9ipSc2o29F3II7hpNNWRVUMoCEEc4Jxk1ahGx3DvCPKk2hSuc+/4VTZ7hRbCEoEU5YLH97B
- z6 e9PRJGkcDLbe/qDzQ46M58PSqzvSl8tv8jQNw22NBH5MRjZ2XHJyMf8A16FuLeK1iAlaVlK
- sNxO1 yO/sMcYrMaHyzCil2HIXqTgj+vNLBFLhEdssdwVyhC9Pu8jrUOlCxDp007S0t0v/AMM/
- uRpxXUUc MhiURL9xR1Lbjy34VD+5O6CCPyZTIIyWOQxPcflVR40WLdMjxSGQbEJxlRye3rUQz
- O0g2NGjOZWy cnIHBz6UKkt1/X+ZhUoJtSp38u33atl/UY2F3taC4G1Ww27jkDBNNhndrUySRR
- 25B2EPHk4POfxq oEu2AnLGVGiJEnOMHqPrmrMMUd0lsJJGjRQURievPQ+9DSUEn0OWrKmoJz+
- 9f8E6UxXE3l7bedoo Ytow2GLlsg/QVNCJtt4tzOvmtdRsXAwAuPm/Cp/PaZYVEMqp5Z3OrYG4
- N7e1XJbaJrFgQxdSowDz IMckewrx5VLaNf1c2q16Sdp6Lytp+hmtBFc3F7AoMkTzMysOvAGBQ
- v2iK1Vtiyl90nAwV2/0NF3b CJZJRcK1uJsEJkEKRj9etMtEktLW4lcPOgk8uJgeCh/qau946P
- 8Ar+rFqKqJO910TWj+b2JPtZgW 3lLqsc0bSlWXO8njK+gHpStcQtbyOQFdJApI4LKAAW+maGK
- TRM4TyYgrIyyc43dh6VWNjcCKOYwT LE6AYJ+/2BH9aIqHXRnRTjh1G7XK3+I9s3F55iSI0RBw
- gHQHj9DSQx3dtYBQ2wAgNuGTG2eAT7jm pWiVdNVYg29JgiAHkqTwPc5zzWlFKsOoSQzxO/mMz
- uc/xDp+VTKpZaL+kEsReKainbo7GMtpIyTm R0PlyKvTGQOQR7VFciS4lYW0BSNW3NK/ONw4Br
- d+zQX7xzoZCZox8itjBIwB+VVFO3S9szxrBEBG iDhmBOOT3Poaca93fqbOvGpO6d5Lps16W3+
- ZV+yeZJEv2CdyYyqMrAAjPBPr606KL7NfSzStl4Hz IRwCcenv2rWtdPEVn/rzAys+0OxO0AYG
- fpnmsxnKpJi6t7oIEEjKuQzE4FKNXnuk9PmTShHERlfV duv46Ed1IL21jMKzRSlfkQv91Sec+
- uAKvxQwCNwjklgWiycliThWB9B3FV5UmRprdpIpFJLkIuGX aMbc++aUxI3kq04Bh3KQDjaxxg
- H8KUrcqSehM6UYQjGKsr/1qPkVlsV88xvIG+RwvyFc4JA/l6Vn K0kdxLJHcW0zF8RnZn5AOv5
- fyqNvtMbIRHI8Rk+6ecqOv6mtC7aJrSKWKMwSJLznkMuMHj9K0UeX Te50wpSppQjeSl9xHLbk
- WqZmMhAOHh+UMBgk/jV6CeXUbeaO2uIQN4ctKmQSOoGc4Bx0qOwDzqkM 3yQkboF7gLk7Se5NJ
- FaOz27xK8McsGWI6Bycjp9Kyk1qpbo1pzpRTjUa5lt/ka0dzcW+l3Ml2wLz TiTKrgRZxiP6mt
- VbyZYonkEKeXmNFKcjJ4B+tYqW7pbK0sivMXHynoDnOCPUVqPb5uLpnmVZPMCz AjgucEMPQAV
- w1YwbIlhFOC1vr2NNHZoFa5ilTZlTtON7Zzx+FTvcNJ5hjOzfIo3OM/Ke/wCFMAkT Jdh94EKe
- oPTH1q1LHILci6XY7ZI2jHAIHFec2rnAk1Llf4f5Nk8sFjPaMWYjDARbGI3ds/nUcUip qEIR1
- /1LLIWGdrZ6H8KUbftJhgjM0ZZCSO2OlRSyY1DzY0CiQFkz/wAtE6Eisopv3SIPSVOW3mST Qp
- NIB5ylSMDbkHJ+6ao6pc2lvpUYnmywATGeQ+4YFMlu7qJbiDMWxv3ocJjaF4xXOajOGtZTc28z
- FZVWPBweO/vg11YfDuU1d6BDCSUlzPT+vQdPcSyambWK4hDh2ZlK+nP6npWXqbCNEkAkjuJYsl
- 85 UKfvZH973plyrTtJFbRskrMm9mPPXrn071HsL3jQteW7qp8v5kJ3qf4h7V69Kmo2fY9CjRV
- NuprZ bq2vrdfqadnHC1pbOZBcBg2xR3Cjj8upqKNorS682ZLqOB5QzSOxI3c4/A1lCO8SaOKF
- gsRxJESP 4AeT9McGtC7R73KJBLNC77wUOMeg/Kh0/e1ejMYYHnquMp3Uvlp+RaE4lkjlALIpO
- FXghugz9OtM mj8qAyMJhsmVXkSTCuCOmPfPFMjt5Q0TXFrPFFP2VsF2z1B7DpWhb2dsk0xlt7
- gMAymKR+hH9fSo coQPXpLD0tnt26/NP9Bkd3FcXC/ZoHGwFULHIx7+p4PNVo3luY5gbG5a0mv
- FcuDjbt64NXVhSyWW 4t4mcB1Aj79OT+RNOmeWOCOa3jeG1JLxs3IKkYH61Cav7q+/7zsjGLl7
- iTvs3vf8wNvEJsySPty2 3BwHXru/A1nTW1ta2VvLHvjZ0LNh+M9h9TWxG9vskSW4jJU5KnqpC
- 5C/iahYWz2StOrRuQSu88Ed ScexpQnJNbipynGcd7X/AK8yOGFpIbQ23kRz+UUdXQEq56Z49K
- rPZyQSSz3cbsiKqkRnG5/4fwqe SK6WxthCyuoIMkyDhH9D9at/vCtyslrPKfNPmANx04FPnce
- v9XNqdWcNJWcX9/36Ff7d/aMqI6xx Ha4nDJnLHjj0HFNWzhs7a3fyZplERiIjfGSx6/hmpbZL
- f7GrGNuSiZXgknIzVhp0s9UcypIY1RlT PIYAcmocrPlgtOxhKUVJxpJ+hlXFjaNZNEs2y4jkA
- XLHI+v1pk1sJGH2i1u3kTduCuBuyOo9s1YX zJ0haaCUwsyGIjALjPJz6dKgiW4bVnZSY1eQzO
- JDncBwAPTJNbRlJLfYHOUYtb8uu/4aFk6cIo/k LQsQQpc5BzjNZ0rRkqr4uWCEEQnbgnp+gpz
- zyxXxDzeQ0S+WEk5yxOc/hVW6uSZZJfL2TedtMqj5 VOOBj3rSnCd9XcIKUXy1Hfqv+C9x0890
- LIOFjhSVWcIUGSOBTIZQIWnSVZOARCv3s+o9sVOJbkWp a4jWCWP+KRMhezKR61ltcWjxOkI3l
- 2+RhxtQdsetbQjeNrfcOKiocj0T6rVP+vvNW0uGOXcZw+UX HDKf4h7VSniura/iMMqyuh2r5Q
- 655GRiq0RlvrjMeLeJAUXPYYxj61K5uI7a3jvE3iL/AFbr8u4A 8/0pqHLLp6GdKmqc7Rad+mn
- 6lqS5eCIqm9Z/OIfec5J5B/KofNEeoQXdxKHBRjtUYBJ4H4gc1CJR 9rldv9YV83cem4f05qae
- 9lURxrai5TecPtBCjjFPktolua4mCp+4o6Pf/h+xctLi485jHukVv9WB 1I6daoPGkljcmV1W4
- dlZFIOVA/xqSGWyaISeY0MiuqMNx46+nvUM4uCqS7oJnRwsgROq/wD1+lKC tPsZ0ptVG7NL5K
- /zehfhaWOAXqlXckiRVHHIwuB29asQWt6tslwjQzPFGsWGTPzMT+lQTLd3d0Y1 hMMsbkwxqMf
- u9vcdyPWtQXUwt7dIEUJIhlcgc5BAX8u9c1SUkla2p59R1U+aMVq9U7PT/IhW2P2O WCS2mLx3
- CK2DjJHJ/DNNkFzJDPdYt4WSRY0MkYPyH7xq/JBcurJ5ojmy2+T+FiMEn8R09KrKUl32 7N5kY
- K4wedzdF+tYxm3qRGnOMG1bzVtv0Bbq5jKIYEmiThVjQBip5Xn1qvIkLWUN1tktVwNqueRz jd
- +BPNaW4HV5tP8ALbfEjh8dcheCKzLiS7fQ1iUwlJY1kjhKfvCF4PPoDzRBaq2hn9Wipxjbl792
- n96+9L1Mq5EVru8yZLptjJhDjuM/jzVKRo2ztjltyIyAWbPTA7CtWYJNLa+SVknjhbY2PlZj7d
- yR mqK2zz3rIwChFIOD3xx+dejSmrXZ6mBr0uWUZy+Hpf8ANbFhPLngi+0yI7mItiJduARjB9y
- cVHm0 t9Mtfs63DTq3lyo752ueRx6Y7U6GMtEv2sqilMxKowSAc/0qKXbPMbyaQQeajOIuc5PG
- cAfjSS1t fQ5Hg4ydoS07a6/gX7v99by3ShirLkBTgY9R+VRxXBh8u6inJEh2APyEyfun3NZst
- 5NFDH5U6yDf ukbb8o4+7j3qAyXLmIACNkBYqVwFx1z7jrTjQbjZ7GijWlScJfj1Ld5DLNOjQw
- zR4J3Kp/j9MD0H Wm7oRAkrPNFPGpHzvkE9j9M1JYSlQ1yrgySEsqHnPBGfw71EsKC6j8qaNZl
- gK7XUkEEdfQ1Sdvdf Q56dSq4OLWi8nr5PfQqWgh8xEnma4d4z5jJwEY5wOR1prsoji8xAHKAY
- 6FCTwD71opFFZwRtJLEW GSQqcleq/wD66wiEvLyWSeQRsvzI3bHpj16Yrem+aTfQywmIxKlKS
- b5f72qfyJyRb24QRPK8bkTf NkcjPSq2+MW0iKj+VIysjM2T8vUZ/GhomEy7ZlmcHJRM5bA69O
- g71JCY2aS3kPEkm9SOMe341tZJ XM1FRnOa+etvwK89074aJkFukwBjYbmORyM055fMkSOVUIi
- JBKttBU9v0qwqqmpxFoluJGjO9FAG DzyffHNRm1EUfnTwyiAnCKGwX4659BxmnzQ0Oe2G5k38
- kna/4lKJRNI5iljK5wWJ4XP8PTrTBFho ZGE7gMcBWwpI7HjmtRYZora4jUW48pwrIke0u7dDU
- OEtmiWVHklti0b7ThXB6kD15xVe17GUqtSS tKN/JP8AXT9SsYrhhHEyPnaNmFx8uc5+nvUQjZ
- ZJHaKRDuyqsfu5NaE92ViEETboADs7sAeSCfao Y4ITbs84mCK642MAWHeqU3a7R0bU+aUfkv6
- RQnVLvUINs3kRQlgTIxPHpx3rQEMDoLhJVMCtkr3y ein+dVldBcylFDJvLAMoOSO/0pI5YpXi
- neKRlX/WbDtDHJ5pyUraHD7OpG9uutl/Wn3sluJIhZok Uu/BBTbwozyRj606NHmvpTHZy7PMC
- yIpGUB7fWnWEjW2n3Ea+RLiUOPMjDYAppdUu47i78xi+XUx nYGJ7cdajVXS/r8jKNOpGjJJW+
- e/l/wzOus7MJp0ixeZIxw2N2c7W5I+g6094SmomeKO5Z5cyQLv yNgGCMfjUtlfMdyOm0ySAEg
- YwRk4q5JeSyzK3lqfMR3aNBgxkAjb7fhXiTnUU3dBPFTVZwcLrvff 8TMgVQkRW3uJo3UEgtkf
- 72D2FLDFcG6SVpo48IymNh95ugOOmMc1FC1wLHTGETRNFblgG5yc4H4Y qUGVbUywXEE+2WMBA
- uTjNaSvqd8ZVG37y107/wDDF17aRNPYD5jHLHGXPRuOWqjZ4F6Uk86YEAjD /Ko54+uOaciP/w
- AJOss8jtAyOxKkhVODwaWFbC1tLAQ+chlx5ckj5HOQT74zU7Raet/6/Q5atWUF yuV79uhKkd3
- DY4DoEaPzPNZcjjgfzojSN5Lee6V43+6zbsAuQRinmS2geO3aVihVgYyclcj/AB5q m9kiWFqs
- cdwU8wOCzk5H8X45NJa76XNIu6UZOz7liwSYR+T8wuQVBUdQQcEfXFab2rpGXkEca4yy uuSpz
- gfzzVd5UmuoYoYpPtETFpCD1Yj730xmoJrqYwRea4QIyhdw6oTzn1PpWUlOUr7FShN1NF+r Jp
- p3j84SsjvGGWRlGAC2B09+tZkkVrEGRZ42iwpi29SByM+vNaBhvjqaNGiz2sm4ZC5x2GfpUaW8
- 1u4t5/JjWKMIJ2ThiOeK0g4xWjO2lRp0k0nr6r8inOQs0ki3CCTC+aSONwPI/GrD31o09y80DO
- BI AXj4AJGQD78Yqd7WO5lXzHQs0ZZ9gxtIOdp96fmHy41Elqm8MQGT7xxwf8KfNFpXX9f0jpl
- UozUV JO/zX4lL7SbnSwY1VZjggY+6CckH3GM07yppEl/d+aZW8yNsfKB3OPSpbO4MUojiaDbI
- A6ErnAHH 8+Knm1HyVuEmAkmjuFAEY24GOR+FD5lK0UQo1qErU43e+u5Xgto7j7T5k4VlZY9y8
- cdTj69KsJdP taG2lWN3kEr71z5RXov4+1QEW7SuIHKTbTkseGwMjj3qSWayfSYEMiJclFM+OC
- xJz+lTK8nrr+hb nKrK87y+W3quxNb+fPPubduddxQ9UI/qauJJILyN5FcxFOWz39D74qOGIqZ
- 5/MQr5pVUxyMjufQC rd1ZgWg3b9kUgSMqeCMZJNc85x5rEPGUvhlK99Lf5GiRJFEHO8RxsIyz
- HPLEc/hT5biYWz+a+6Np C59QAfX0rBgRZFZVeUxqwBLOcepH196vXepNDYZihKo8wLhgCYicf
- Kf51zOi+ZLc5IR1Vtdf61LI uWEe8kyRvESfLOCrA8DPvQ1zcvaxSJJHFLsZY2dcqysRkgemeK
- r3NwEucMyy797L5Y2jAwMfnyKy p75lSBDGxZVXcB/snk+w9qqnR5rNI6KaVVXiti3dXl9BaTe
- W8IAk2zl484YjhR6Vg3OqXQCRsUUR 7UbcuSSfm/pinRxFtWcPDdSRyKzyrv754qrezRwxsHgb
- Nwu/LH7pz0r0aNGCaVrs74UKDajKF330 X3EjMY2vLmOZJy7bmRDyoPb8BVZd8k1zHGYSyzHyW
- VeXXHHNPvmlNkPKeApLlnVV5LDAxn61Xtp5 I7+2id41jcHeuMEBff1rphH3L9f8hKpTnTc73c
- f62tr+BMHYWsAuG3ThTDGqccnn9D1roEuZEt5g GR8yIr4XuOCR6CuVjn23IaaJg2QSM8hs8H8
- jWw7izM0VtOl0FOAByQf4s5/Gs69O7SK9hUbUJNWe 2/X7zciYT3TW85aOBJcqSeSo6Ee2aVWt
- 767ZEaWUTK8uxW+Yds5/DNY0d3NJbiLz4IkMTOSV5+U8 Vr2N5i1imgt1ZlQwhlAGd3f8O9cVS
- lKGqNlhPYvnvbsr9e+xJa3VqscEUayxlIvLZ5GyMnpn9aYt y0dqq3iCZVUoNgwGz90j6d6zry
- 3lS8cqymAMhyv8ZQYGPqTVqG+hEszzrtd3aSNW/hA6j8T0pulG 3NHU9OGHp814av1Kaox0mRn
- iaB0b5pHOQCOin3NaOnqbiPZclYFQSZDDJOB29BSkrd2cdoqlWkTe M9VwclT6sexq7a2lrKsm
- Le5Sdd3yGT+A8mlUqrld9wdSMYrm3T+75sq2e7YjGOS7gG1EihOC5x97 PtSzRCKSVkjui7Ocn
- zPvYxVcLcWgVYklKgBh/vZzU8duLi7tnuZJYTKhlPzkdKUtG5X0DESnTi56 Wfa/5Ji2V3BPMI
- Zyo3nzC6cKOwH48UOIJLZxIXgUPuxM2SATg/rTJI445MleH2EsvHQ/4moL82ro jSvJI6kRNEh
- wST3pKMXK6urnH7OCkpRbjf8Ar+rlt4cBjch4LaIeWjbsD5iMkfTp+NZv2WRvOXz/ ACssuS2f
- kAP3fqeKsPfLaXEttK4CKxYGQ7s4GSP5VHDBazQGdpn8zeCYtxyR3P4VcXKCuzGvOpQe qtf7v
- +H730KitZSah5kqTTu7F9obG054X3PFVpjE07QpulM8hdwvTPYj2qW5igi1jz/NCwgnao6q D6
- +vNZ7Yhu9rSiMb0eRWHzAjrz2rrgk9U3sdNS2s03ey8/yBp7pdyPE5hYNnfyckcc/WqSw7ba0d
- WQlCBIoGCAT09yTWhPHKYJ3U+egnJUqOpPU/SqmJkZQmwNghht6HHGfc9q3g01oRS5J0vaXs/L
- b5 ixTyQFY0yrSMRhgMBwOn5U15JXaLfDNIWwqEdMHqf0oFpN5ClsfvVEir1OORmpGuStpbRtE
- 2Y4yu fY1Wl7rVlTV7ONm32Vv1uS3VwpupYo4xGjlipI/h45z6cUttJI9kZ3eNwrrGCBwPUH3x
- UMZZYoXh ULGCysZOS/HQelSxvCmnMu4LKrKGP8Jbk5xUOK5bJGVanOpRUUrR+/8AHUZLCU1MO
- kTGMneExkbj 0rQF8ZtqzoE2D96AuPmBqratb7fnMpCsEDbuOeoq99h8lk8mRG2IRISMh/8Aa/
- Cs6jW0txLbklHX +uupYslukl8zaSjLvZmPKkZyP1rQkgMttcJgTbXC7Ivlbpk49qpOk76SjzT
- o1tlVAQYOWPr+ArQl jdLuY4csx+cqcAHHI+vSuGb9699TKVL3lNNL0/pMtfYJobBLhJ1DFirh
- 8nlsc/QiqzWrjUvK3Kxj VsKg5bA6/hTysUVpBteRnO6VAXJAjyAQfU56HtSCG/uppoXUpIJGU
- soxhSMkfXgVzxbV22cULwTb ehVMV5aTwlXCMIWR55BnzM85/XFUJIrtQC0TBYSpAPUKOo/Dqf
- WtcXCo8AkuEEJXEYfnIJ6/nUrg LctJI32gq58x1OFZivAA9/6VqqjT1RFWFnfkun/X9bnOIsc
- tvJE48uVbhRFKpwoUjJbHoT0rQnlk jlgVogMRMLiRVADknt6YqSWXyNLjhuERhtJfYuCrZG3P
- tVCETSaw0jW8qouYrhmOVVz0I9Oa2Xva vZXKo4eLhzVFbsuv4W/UzJIXFhiKGeMpIoRpG3bQP
- vmoY/PjVwqLcxSAyoSmc+4z/CO4roGit/IX zEuA7OFdA+MsOCP8aqXPlKoVIpo4gQjvu4zt4U
- enp9a6IV76WNKdWHa/9etzHlillZVkaGKNxmRg mBkjOPqccUSpcXNnKRKmPNC4C8gEev4Vdku
- VuJYoUs5mEcWwLu5B9/cUW81tbARTho9y7gCfvAHj 8a255Jba/IVSLuuZO/bRmR9knc/ZiyyS
- JOdskXyqNw+7+OOPxq3mYSgQtA1ycsqrH83pz7HrioZ5 Yo5YZYSWjdd2QTkEHjPvV2CVGigaQ
- iHzLd3DDrkdOfQf1rSpKXKm0YYmPLG3R9H363Kd40kl0luB 5rIrAbR/rF4JH4VSWCVr+OK2KK
- fKI+cZ9+aleORbRriWbaBKFGTyCw5/Co5TJDcsWzsiOzKnB3AZ 61rBWVkJq9JxhZeWv6/8Agk
- tLqIK5xbrkKSy53H0zUohQxmGIlZJGeTy2OSMdFz7c1CgmMqmOQ3B CFQoBbGeSPrzTrbZEVkl
- Ei8mSJMndx0z7Hn64rR3tuYqNorRc62t19VroSuVjsxM7h53OH2nAIYc EenSpnmigEEkEwbYC
- HVxuCk9OPfmkuUMaedG8UsLhWKMOwPaq5uHfUd8USH5SChQEE54P4ZqEuZX I9i3T5pLRd3+A2
- 6+zo8ioZZxLhwVboF9c/jU8zW0ljcS27FZzeY8p/mYll/lSxp5qzJPPBGkaurD HzBvwHY4rPF
- qJJYXWVERoy5kOcHtkfjVJJ7vY568Iyak29Oiu0/68iK42SSQQRskrSMDEE4YDGCD U8Zht7i4
- dYpCY5Qq7myNuDuOP1p0Fi/2xZJIzOIQXPl8EMDwM/rUEjxXOJkV4mE4WQZ4cv3Hp0rW 6fu9D
- n9qr8s/h8tH+hoW8yNZQIixAxzb5cr97j+XqKylkS2kRoVDJjnI3BOTwQR19Kmc2YnIBkiQ Ag
- nOdxzgEVItudnkNJD5O8sHxznPTNTFRje/UyhGjScoq+u9+3luQQSzTzRTsUZI8/ukGGc9PSod
- 86XCB1TeBu+5yKsvCrahKWmhgbdljzznkkADtVXyUklt40kMjSDaoxyK1TidTlTo8yUtGtdHb/
- I7 KZI547YWjkwqMSuD1Zj8vP6VqssEZkbf/pEM8aICezDBB96rGVjbt5NuU3uHRRjlFIJ/HOa
- v3M8F xePKAoUPuIA+9/tfTH614NSUnZdP+G/4YyVGTgktUZDy3lvEyBoikbrsi25dlB55rUik
- jCTT3Dww xGQqMLjBPOPqOKcl/G92y5g2RnahKjIVuv1qaeJCdiFCigRxjH3+cg/XrzWc530kr
- ET1fK4cluv9 aL7jlLm5S7uYYhFNbzSFjIxfgfQf561pWdur2ygyK3l7vLQ9Rg5/+vVtklmgdU
- SF/MPmsVj5DA4V fx9KuyRoLxpruJg4Zjsj+UISANp9zW1SuuVRWh11Z3UYr7r3f4laOeFGKFo
- /PJSR3dcgk5zj2Aqv K6vEXjkB/csS+flBBw2BRbxo8Kuq/vQ6DLc9G6Vca1Z5JVCMJHVpI48d
- FH3gazvGMgoctOreKt5M SGOCCB5oFmuX34wrcnjrVULMQoEDrGQrOz8jr0/GrouEWJhCY7cyg
- bUc5JOflYe3WoXt1huZpJbp ZgCA0anbkjuPbmlGTTd9zspTVHmc939/+RLHte4QhbgpbP5ZVZ
- MHJ+YD8qkd3muZmnkiWFrhDFkf e4zx9TxVa2nsYplZ5Tl0aSbLcb/uj9KfLfu11JHB5aiJ9ql
- lztGOfxFS4S5tEZulJ81S11br/wAE ZPO0VmXWe2WSb94yleVx/CffmpUngkgkhupreLbIGzsw
- QF6dOxFQu8UsqxMEMbByCR6Y2/map3Uq q15mMTM+C0gHy5A4wPQGrhTUtOp00cPTqxUZNprtb
- 82NvD9rmYwosaTFpom6AAYG3j35qaxsgbjy bpjJ8xbcDyMDkH1NVILtUjt51uIXlj6psyD13c
- e/9Kn8u8igF7HIGjbaJABz+H1zW8lJR5b2NJTn y8rdl0et2/X/AIIs80iG1Ft5bM0Ls5xnDA/
- 4VDEWkjjnZUe6Y4UqoAC5yVI7nHOatXRaVGgkUK0b 4XYuNw7j8arXFvKliHmuYgQp8tQuCyn7
- x/DilBqyT3MaTb5eeyu/X77FporWZlkM8kaTuHjw5G0e /v71Y811uZUguhNK0mDzkDHbB71QV
- 7a3sxAtzFlMpIWyfnOMMPRcdqLS2E2ni4hV5JM+XuRuGG77 w+gqXHS72N/rLUZKUrJ6K60/H/
- M6hZokuZ5GBjbzAjg9M9z7cVkzTTG4bfE00bzqGCfwtjgH3Ip1 iuHZ5SfJeQsrOcjJyqt7ioD
- BcQIJGuEH2dgrErkEjJJ+uDXNCEYyauYYeb55LnUn/W3/AA5o+RaR MbgmVEyFBd87M8BT7nrW
- bOBb6nKxbdDHEYyzfxluSRWtJJbixiIuIn8yBiFxn8frXP8AmWsixRMZ ZJJXIR1fhlH17mqoX
- d27mlGvDmcpN9n/AMC9i1a3cK3ELtdwmMpuYY5JGdoz7is+/cSyF7lHDK6t C2OMY5U+tS3Rs4
- 7SWRFEOZo2jBHVfT9KT7R5UjSXEBmiJdkQf8sznO1vXiuiEUpcyQoxXtPacvNf rez+RWtRFKy
- DbJ++yevTaOR9TSSSZ1CzxbEMSXKHG4Dvz+tEaXDndbRkRtC24d1YZ49s1C08Spa2 13L5bm3K
- tJtJ2nsDjn0re15aGtShGad3p+RJC1xNdD7MhdlyGOAck529R37UguDKVdnCSrDhjjGS D396l
- ikf7T/elUMrRRjaQ2MZpUS0hJN8rJlVAUHG3jofc9Qad0m9DuwmJ5INJbeWvz20FgvYjEkU hS
- N9gUOVz17VpSanvtTGhWAqVUKV/M/gapR3EEeoRLPaFoSCxAwGjYdMn24p0NvE8p3zwymZfMYI
- MMp7jPbNZSUL3aO2lXjGV5wemqe5pRW0kgtgk3mykM5wTjAHJ+n9akspLeJgJ7WWQzIph5GfLz
- zn 1571DCzxLFGiMo3GMHP+rycgH602YyQ3JuJmyrZxt4AxjIHpXO05XTM4znKMrt67Lr+hqwN
- ZHUyq szNIXkLhuBt6fQZq2Gu4YYrqcrmTKoFGPkP3vxrLfybkrboCrAFlKnoO4P4c04vdy5Mi
- P9nCnafQ AYx9SK55U0/+CH7ru0/Pf5O5dMd1O8ciypDGqeXECueD3P0x1oiZIJJTOTGXIKSOM
- hF6EY96qRSx iHbIk8ULKJYSX7AY5+hpbxvJtt3nJcs0m/avB4ABH454qeVt8pz1qslU5H17b/
- ft95JK0FvctdLI s9s4+RAeSBwKquwhQoDG8sC53EZ3Y5NV7ET/AG2KBIzHald8fm/NgLk5zU8
- TyOQ0jxtMclDt4ZM/ NxW3Jyuz1/r+rluEVU/eavvf9CvLdQ3YcztDHayMGj3J8z47g+gI5qGF
- trSMkiyF2ZnYDAVgPu49 6Y9xJdwLGYY0hd8CQKMIP4R7dPxouFke1eNXTz0ONsa4IBHIPqfet
- 1FJcux0xUJRUUku/axXnuVk ESKhMrruK4BPrSpCZ7NwGjSWOVVYyjlmJ4P0x2qaJg9xEqWssj
- jJj56L/Hn1xUweOGRMGOFGVvsz PyCv94+p96qUuXRI86vWqQl7NW022f8AmV5oJFkaSSaPD58
- xEJGJM8L0709IEYebHHI0k0pkMW/5 htPAqqxLQTfMzNDKkYPbB6k+p96jYXsWqF7csF8tgoIz
- jPT86qzatcVSM6q3Sf3J/cWb5MRSbZMS +bhuf4c549BVSR7a51aJVyokzI5J4G0fdqeNp3soy
- 0ZaeBypXH3R2VvUnmqPkSAqpw87/vQiA5AH UdKunGys2PD0Zyp8t7W8/wCtPUlS4ZbpVaEMjp
- yhAwM9D+FT3FrEqiFGEczOCzNyvTAH49qjsY4L a/cSrLHIM7GdshB3BHer32d5PJjUiVUdR5i
- nqxBK0VJ8stNDpxVepGom042+4SzUEm3l8s+YwcAD mPYOAfU1ftIzdXjLBHKWUKuc8EfeIx6n
- FNgWRLyJHjUGFCPNxw4P8X4Go7gSxW8zo5BadTGyErx0 I+tc0nzNruFJybdOT1e39I6BIlJub
- tQCDL8kJ5wCMdOmRUtzMgmUM6nETngYOeBz+FZCSx/aLcNP 5bksdjE9M4zVzcxtmKOk6LMFDq
- OoJ5FcLhZ3f9dDmp3Tbldtd9B0U88jLiNFcIyoWXIZepNMaG4Y ySxziNhGq7T/AA56598UrCK
- e9LDfDJggLu6gHJxU8ySeT58Q8yNiHVR/CvRgfU0XszWUnG/Ilr36 /mZdvZOuoxfb9vlvE7Dj
- AXb/AI8VoiH91lg7tsbeUOB5mMgflVSZ786nEsSh45EaRl25MfGAtNju Jfs0iwQT+e7jYpbqC
- OW/LNXNTlZtnJX9p7Vc7V/69BkEjfabXeyBWtyGjcZJJycn6VWNsIz+6Z4A sTKpkbIcdd9WFi
- e3817WGSdZHBU5zs2jhST65NUDai5kYsLi2JkJIZ84X0raNrt30/roazVOMuZS Vvl+RWllRZP
- OkjuLkH5nMb4CNtxj696ogyxtCyyrMY1I8vB+b+6/5n9KdsiiDAiQqSXjJfjaOCp9 +nNNWVUn
- 82AEgoIkHXO/rz7Gu+KSR0XpuDcv+A/68rFzy7i6W33lfMRTE5jG3eW/i/DGKjaOOMGT 5WkaN
- izMMgMo4WqlwsNo+ySdhKn7r5SfvDp+eagtUje1it0huXKE+YQ4O4nPIz09KFD3b30PPhQc bS
- T930/zewNfK0tsrpDbyCArHlOCfTGOtNAmNrHvVtm5ZFZRgKBwQfr1qOaAbkIUIxUuC2DjA4H6
- YqRbe9aw3q5YqygAfwg84Pvn9K6LQS0NsRyxaVl6/wBXGxXckKF2jMtqhyAQDuJOFP0qnPbM08
- qN unkSYybl6Px6fWti4lmuIJYvLWRjJgbFABI5LD2GKxpwGkLhZPNYA/KegPXNOi7u9rM4IYe
- lNup9 rtr/AMALW48m+juSpbKFnRRjk8U27lE01tIytEFQg543sT2pk8KJeNHFL+7VtqM2Tkev
- SogigQ5Y HzBlVzynOOa3UY3Uh1qVJT5ndPvqa1kpTVP9JglEP3QT90D3/GqbzzPPIpeKBTMok
- O3G1vwp8t3d W0vU8t8rEZUnufoav2yzyyS3EqwlPNAeMIMlwPlNYy918zOLEy9lJyk07rT+rW
- uYk0YnC7i6OOcZ 5PYk0rW9w9rbkZELuSVXqDnGB6ZqeRnNjFIIJo5GygyB68dB1NTIZ5bm23I
- YiJAr8cFsHpWrm0gx FWlGmnTd/wAH+enyJLCUPOsAhmGYpHUh+vv9OKrXLNAIEIiETkOp28se
- uP50LiOWKVRJDtTZknIA yd6n3NQm633yTImzy1ICvyApG38+9Qoe9dLT9TgeFhKo5Rimrde/r
- f8AzK8zN/aC7kxFtwF287Ty OfX3qL95b3Ufl7jGgwyvyTkfzq55LTJBGJEdURotwGd+Odw9qh
- AhFo6sxxkGPI5IPfPtW8ZLY7Ka hiXZ9PXQquqGSMsrFFX5RvwSPfikdlR5iwG4t0A6e1NYmYR
- naoDFV3dOe3SpQzBJ5JCFuCSWTHIP Q1tsbQnC/Lf5/wDDWPQtNDQlYLWKXylYB2kO4tk9R6Ad
- x3rb33H294hYFkXJOFGcA4J+mKxgbZb2 FJXeJMPvKvjLdVxT4b2W9iNul7EkjOhdyCPn54+hx
- 0r5irBzblb8zzoqUqsny2XXfT5rcubYRcma KEJAq4hUgFtpHf1NZ++ZgqxywSLCrLhRzkdT+G
- a1riNra1ZFUCVz5sYIyFC4JU/zqo88F156GLb5 7faCyHG3HBH49amnK6va6/r/AIJ6VGEpWqc
- l0UJZILazjiO8LLGB9/DbgflxS2cErXU7TSnypDgu zEgtggH+lTy6YssMlymfs/nI6knOBnDC
- l+yLBb4iEgj83awZskFWyPzBrb2kOWyerN6+LjCm1Cbu /wCvkKCLO0VjazRjMZUOwyoB6n15q
- S7iM32gxXaRusx8pv8AZ7j8802WK9bVhNdwvJFGxSIKMBcn Jz69qe2pKjzR2yAsvzK7AEZ4rN
- KV046s5uXmanC0pddV+pUWy1C6uLV0ubYqi7Rtj5UN/wDqpjWl u8CxPL5kxQtgNzIecYH4VDc
- G8eXz5IZgsk2+NYjtIAGMce9aDSRC38+9gkhEcq74hw4kxwM+ntWz lNW1+6w1iakJxnBL0Vr3
- 69v0CygP9iWgQRxOsLKzSpu5Lc1bW2CIskUOUVdsink7jwvP86iHnGAF 3RcI/nLjmN+oBx3NU
- bYXLQxtvMathgWzxt6KfU+9YNOV3cqLlUp80ZW1ZYuvOit7uFogjpMqxEgc 4A4/GoTA10s8Mc
- sOxTmIbeWXtz35zzTxeTx/vXw5kbeARnZ25qKO9LTRyHyxb28PlNIoxlyev0rW Kmloj0KM5uE
- Ukrp/1pZjzBJ5EFuq2wkWJfMJiB2ZOSD74qERySySwRyBY8jyvbI3EH344q1DOzNM YADdGaME
- HngfepbYfZ7mZLd0mkE+5Vxksq5/xx+FLmaucs5TTajFX/J9zNmayiaCRorrMkL7C0nQ DkE/j
- Tmkku4FgmAiIi2O56bsdvQdK0J4DNBZXGxGtxbuT8o4w2R+fSs+7lLXUd3FC4Z13bD/AA7h jn
- 1xWsJKVu+pNGfPCL5m2m+vUhtrCK3Yfa5I5WkbCL64U5P51qQzx/2WqxAKrIXmVTgq44X6A88V
- m232OWbfcSMAjhFfdgBcYx9T61qmN/7OjgjMewhg42/NkH5eaVd3l724YhRVZObb+Vv8/wBCs6
- Sr aq33wyAxBe69WP4ULeGd5EmxBHu8x2boTjgfjWcv22O+EKAxyLgx7+flHOKt3PmTBJg8cjr
- KMIgx 17Y/M1TppOzOmeEhF2l/l9//AA5PeQt5ANy6QKyqAcYHOOn0pzzwwIYIoCZokdY5cArg
- 9eKpedNI k0V66FTmVTt4baeMexoSWeVJEysU81wJUV/4F/uml7N2V+n3Cq0nHlUlon8v+CRl4
- Lixjtw6QgBU TzBkkA/e/OiK18rzkeQl/MVJPrg81B9kEUJuZgS8c6rJ6Nn09B0pJzLJffuxKt
- wsmwc9d3St0t1F 6Fx55XUJWj30sSym3+yjyLqPzWUZUoQc56A4qnA4aIJfeUjiYSRlkyVC9Qe
- O9aCwyrBcI5gLLIDK ioA4IOOtQzwILzDkukAxlemDzVRkmrXO2tCpiaVpzbt979NP67hmS4uJ
- Ag8t5nMjjGCGHIGfpVq5 hurpbePy/nkVn3kcA44H1qta7JIp3kjlFj5u4SbueOBzVRoZnvJBC
- 0rqn3WBPODU8t5drHDTpt1P ity7f1c1Y5lijRZ0KlEVVZhnC5+YN6k+taFq9hE91LGytD5yhS
- edqkcg1myxrMZWuI5IkL+YjE9F /iz684FJm3NkrQspkMgIXscA4zWMoqS9fuOudpRs01f7vvL
- E1wV1F/IlWOAY2Mwzu4xmnCRbaG3g m3TuylMH+L/a57DvVaWeKWztCWWIhAhDDgZ6/liqkhMU
- bbMkn7ztz5ZAOV+pqowTSWx0SlGq4x5e Xz/qxsxh2mhJuoZpY0kQtGMKfQ4/SlElwmnOZ3Bif
- MhbJG2QDCj6e1Z6uY7KVWR5LffGx28MB3Gf rUlw9nM6WtqJsFt+CxI47fiah09fL+mRKnOE1z
- q6XWy/P/gFiG4VGj89JI5Ik27mORjuMfka0yk1 ywNuoYzQl2kxxuzjgduK5q4tpLYw7pcyL0U
- 88dyc8d6Yst1FexRp5imNCEQHkoeT0qpUFL3os2lR U17alv8A10f+Zr3LzWiCORxvCMin0BA4
- +ppbGNomVJd6rErbHc8EY5H49Kjt5XMSNbp+7WE7DMN+ PQH1PXmq0ySXt+Fg37mVfKAbA2dST
- 9DUKN04vQ4788XGbStu+33afmTz20i+Q8cy20Dw73LgkDIw F+tPsfmt4y6NJLbxnyiD99e+fX
- iq0Ya5uLnzpFCPIpB7fL2FadonmzS+ZC6sgZTsOAxxzj9KVSXL Gz/r+tjKrVpxj7O7l+X6GXF
- dtbFJYSPPRCE3Lu/dk8n6+9SlxLqb23mL50T+XE+Pl2kcnH+eabbx SJbbXspkneHCBjn7vapo
- WZLuFvKEcrRfMxGQ5PQj8eKqXLq1v/X/AARfu+VuH+f3+RVhEMazr9si hMlwPlcE7doIzVu4S
- WKyuHkYTvK6lTGcbRioRDIrmL7N9rZVIbAH3yOKmaJJYYQ9yiTouyVDn5nb GMegxQ37ydyqUX
- CanLbraz0+SuvvIALdLJEkS4h2cFi/3t3J+tW7KIygxIphlEZ3O4ySRyPoMdqs XMFuJooIMTS
- 7GZxn0xgc96rRSlJSzK0MJYu5Y8pg8KazcueN0VUvODcE0/P/AC6ilY0uIYbgxyXM kTYwPv5H
- 9BSQSRyXSwqkkCkBgzHgqOAaXyJbh7meK0nBkcPHMSMDsQPaksILS2keVpQpBZJfMOcc cEego
- fLyvuKpD922n926ZJcBQyKS6hx2PRV7Z9ScUk63tzbJFJiEGJn2FeQV5ozaMIEVJpEIBLb+ GG
- ecfTvT4ikmpXSxXUJiklGwk5JYggY9AaSulfsdOGlKnG838Ouv/Duw2GLzkgnZQjSqzlj0PPb0
- FbVhKLiOZIU4SVURfXJyT+GKyXVo7aSCSVY5Y/nXjpt/hqGzvIZXd4IZ1Vn4G/7g5JB9T71FSD
- nB sdXnq0m5v0N4+fBfSm0MboQxAKZLDuR7daswXH2q1+cCCLpF/uHqDjqeM5rPuJUZ5tysu2N
- mhKNg NHgZ/P1qrC15HcI0Vu7RNjcuQeTxx9BXN7LmjfqZypJ0+a2q79S+IWUyXErFmRkRdvAO
- 7v8ATpVG 5+1W9uzLG4uEUur4+UADBFXJdR8qcrHPbkKwieMpksW6H8KxLu4dLkW8CzNOA5Ys2
- QMHpj6VpRhJ y1RhSd6n7yK+f9f5kVvcI2mvcKZDFhQED87wOufQHrVS6uk+3w3CSGSXb5rYYh
- Wzxj2oN3HGFgO2 IKu5Dt4Hcg+pPaormUyfNF5CIkbIfl6Z5Cn3r0IU/eu1v+R0RhJv3oavzaV
- umyIFluHmCTL5jfMw 2rjGeB+Ga0IZoY7QSEok6tzuGVwODgfXisqERxQC5lkMUZGI8knecgkU
- 24cyzTJG28FywAX7o6kH 8a3nTUnboLEUlVkoQXurdp/g+5LudmVmRWmADMXAPzA8/jzVkm3ht
- 9sTb5Em2ttODg84/ClimvZp raNbJmjJy+EwWY98+nSorSzmXUWh3KLjd5u9h8uB14+hqXbW+l
- jJrn5rWXKujNJYtMnjucM87glN qtyMfdI9qzpLZbS9S1j8yVmJWRN3L8ZBHsKmit4JrkySLII
- pHdwUbGVH3f1rOWVRfRpJcpBM3yq7 g+vHOKmmnrZszw1T2kJv2jtHv+WhZmQRyhHt52i4ICth
- gSM4zWbds7XBJ2xkk5xxu9CPatAGVLto rpzKZyJW2HG3BxVUN9sugYouJ/3sYPITaSCPxxW9N
- 2d2RCrBTd1bu9l/XyM+OMz3TDcI1MmAWbIG egpGheOY4jkyDgHHtk1NPbBbsxllhUL1IJ5xnt
- TEWKMDZIXVycY6jiupSvszeUZOqoJ6Punb7+pJ iOS2VmnRmlYs/GQp4qOB54xLItwjHzQWIzg
- 56mo0RQpViqvuUbj93vS20pkHlsm0s5bKjA+v0qXH Rnm16UYKSvr1029CT7XGZz5qkQBWSMd8
- Z4b+tRzJAjKgL7h0O7/WN/eX2q1Z3Hm2Hl+RbTMWd8CE E9evTp2qCZ1ubeGW4jcqke0NHgc5y
- Ov+eKS0la1jOlUmoqXLZdrp/hoVSgQBlYyMflAB6MTyPrUv 2S4juLeaKRUUbyxkGQ6j7uPqeK
- gR4Ycy25MTCQkrL82/PHHHFajRwz71cTu0TbFUP98+306mrnKS 9DlqxqT+J2X3fLcmtYilmFE
- 0SXCAqVZehYZAPuTWVe2zxpG8jAttw2BhYyB9wj1NWLkmBZZTFIpk kVgWPDAjqP6VHDPHc3ck
- jyxrub5S/Krnggj1x0NZwUk+dbGMaNWm3KPw7+b+4I1t28OrcJiN1+Xa ecnsfwNCiK4t41Y4l
- Q7ZHHb3NV/NNtGIoplaPaUT5AQyk8t07VLaC1tS0aSAKJAGJGS5PvVyTSb+ 4wdOpBXb130f/A
- O3SYS7kkMcjvKrKir83XB/SnXOnW0MwktXJLuWiYE4yOgPrnnFQJYPaRj7Mr3E xJbg52heo/E
- VoWotpWLBzGFb9zG5ySjjgn8e9eHKXL70XodSw7ilOL0fl+auNjtLxftF1Mzpifyi jc7c9R+P
- amy2cKSBFD2uI28wyN905xt/KtOewEOlpHHM8p2qJFLkkvkYNVkMWnT6isgZ4HuDjzDu JLL1y
- e2axhWctUz0cPiZS1Tu/JW/AgtcW2/96zRsVAJ+7kHnimSTvJq4ZUZY4YjG2ejMT6etPWKO 7g
- tFwZRCFiOw4yCcn8eOtWPswgaRgJFtpJCyhjksOm4H0zVOUVK73/r9CcRXvJ8+rfyGi982J2EU
- qsZcgscjjqPqcU6W2jmeQNC7JM5lRo8DZgfcPvWlbWyAKEiaSZtzKvqBwf8AGsgwy2kUKO7TAq
- 37 xTgEg8YHvWMZRcvd0/r/AIB5PtXKWyVvPX8xEmuI4osFI0kw48xc9PT0qnDPOV3Xmy6dgSx
- QYC/N 0Of4j61atf7QLgTRpPEJFV0VPm5zgD0pYhHDPIhUhmOTnu/PT29q3ulfRHZTcY3en9fi
- XJZ7p2jl toMSOrtsKA7SDgFvXvWB5sxkt4TIslzKn7wpwMDPIHarcz3kH2RIoncmMLuB++M8s
- PakNrN5zOWR cuSzY++o6lfQZwKqkowXQ1w0uTeS8rNX+fclhgmeN47iaGNdwEYZOSAPvfTpUV
- tGkhfDRS7pFEgA +UkfeYD0rRguopCyi3kJchuSDtGMMKqSRxLbWazRSoiJhirbcgHDZ98Y5qV
- N3aehcHUTcWt/QuGG 3tpnuhIhQOCBnqB3z+NRyR7buN5HRYyhDOo6Eds+9WltpJtQMVo0f2YA
- iJXXcdnfPrgipXsJFsmc zxOElRRJt+ViTnOPTtWHtEnqw9ooz5ebXz/4bT8ShIphW3lkjcRxw
- PG0Stg7mP8ATNZ627QrNErq JIpFgjD87lP3j+tauLhLy6NxIpE7eeCV4XZ979cUtncyXNi8iG
- F2adJXOwcH0/CtVOUVf+v63G+e m3ZJ/l+RWNnZ2zRpIyPBGHCjHTHUn1rOWST7JEu9ApXcDjn
- g5K/XpzW20UEVtc3UqSSsbgKGDfKF /u49e1VEht57ny1tJ4N5LqjPzHgcqfcmqhUurvUinOct
- WtPkVLS1uJrSRVYCZZifmGSo67frirRX zdNhBtZY7cqPm4zyeDn9KmtYEigE0UrK77N0ZJJjY
- kjafcip57rar2m9GiPIXHOV6HPpUzqNz0CL lOquSS06tbFeKI/ZrdIYQgjiZAZQGO3PP45rPu
- bV7qS33W8vkAEPIuBhyOp9geMVcZnmado4ZCBK jKqtymBkg1o+bbta3UARyrSFzGG+bpkEH0H
- enzyg7nb7GVFJ3u+/b8TjlW4ntxDdZzGcFQMc9GJ+ nFSk21tKSgaVgOG3d14P+NakLpe6fKGM
- cSK6oJSPvlgefpxUU2mQi1DCTyXRWWUsfauv2yvaWnkd tOvFQ5Z7Pok7Ge/mm8WOOYC6Zd0gI
- 4zj5s/hTn0ib7FC63cd0ybo4xHn5xnvnrxT5YrdWeZZ1WRN qkkk/e4x+Wa0Y9Pia2dcywBJ/L
- ALHK5IwT+FOVbkSa0+RlicQqNSMrWXmv8AgFFNMmbR5I3lSE7w QjA8flWlZ2FxHfPJDLE8aFE
- zt/1h67h7VoBoI3eJ23xGXKt3kI4BB9M8VIXAWMrhZEDeaF6GTPHH pjtXFPEzkmu559XF1Jxd
- J7P7kVX08yPi7ljERgYQsoxlAcs341TklhlQojwW+xgRleqgc/4VqQXH 2tbszJseWRWGT90EY
- AHoMjpVC5itRbGRbaZJiMEFumeP07/WlCTvaRp7SpdQm9trWsUJo4ktg4lg Ikw5i2/MCOKha4
- thdRNbRSSzONzKTkAZxtI9cZ5q8F32IhWAu8bCIMOwY5yfyqOZLmS+uVEaQt5g fbs5TjlT/nv
- XTGS6m0KkXB3k387fgihFcHbJ5kkUayT8K4zuUcAj2qmziI23lxlpMlXkToME/wA6 vxwQtN8s
- yrLje0TLyAvoen1pzyQzwxRIoTE6hnzwdwyf/rVvzJPYVScoNXu119P1KMwMskCz7wjQ 5MgOA
- v1/L9adbPbRym483cwUoV7kHqf6VpS2yy3KpyvlMYxnkEt3/Ksi5V45YrZRG6hg6MigblHU 59
- zVQmprlOjD1VVTipW7pdvuf5lmaBdkjxpMsDSL5ah+ijrTxcBIzNbxNGYN0asxyME8D69aikik
- uru5itJSys+8nPG7GcD0zUkK3QEayshcy+b5YXpjnB+vak7curKquVve963S/TzJYpI0itJ2kU
- Oh yy/7Hv7n1q/9uFq8cZSSeSZ/3LIcZTGT+Oe9U47yC4uJDHaySBpzsUEZQY5U+tSNZXKXkct
- ycxOp AK8YU46elYTjFv39Dj5Kbi5S0t0vv5K2pm3QMmp2WJJlUw7mG45B7CtiGKDb5sRZmjm+
- QM2SykYG Pxqo8MUGoJmOWaHbuOG7rnaM+9aWmT776WG4iWF5U+0ZPG0pwB7c060m6d10OrE03
- OlzQTaS110+ etySygvgXNwNgikCYx3YfMT9Ksi4g814Tbqp3rIkhAy4UdPrRb+VPbFi8jM0uL
- k7/wDloeOPSpXK pbyieJ5I1AijKnBO4evtXDKV5O6MVyTjzaLyWn66kDyWtxfJOkTMkqMylWw
- VHYfU1BG9vBam7lid 3YBNjHIIzg/j0FAgS3WCOEthEdMHncTzx9T09KUwia1topEe1jYFlMh6
- Jj5gfcHvV+78gclKacn7 r0a6kMt8qwYCuHQiN0B+65zio4EsjqDRXc6FmQDAyN4wckVKloN0Z
- 8xE+QI7MM/MTwT6kY/WpJzg zSLFHclB+9eNfuMP4Pqa1TjblidKqQUJUqd/vsx6CwMcsoBSKO
- RU5boWrPCxxXG2RQ/lkEmPjaec GrJhsxp4W6tbizjkZZA5l4fBx0qzdRWdxqJWOUNN8xCIT8w
- I+99BUxkk+tjNckLc3M777NffqZv2 cSadJcKWKOyqznnLg4H59aWO1Np50MbJuF4CwYcj0X86
- vfYrZUlDF3Rm3fK5AJAwMU4IXs9rFY5h taRzyFZeuffFU6t1a+hrVqup7t7r+uhnTRNDqD3s8
- oUqsgCEnrjhalVt9rb5hu5HaP5NkmMDHIPv 71baOS5SQuoQSEypkDjjgH61TFvLa+XeM/WDH0
- znimp8y1eparua95pNfiVHCw6PG5t5wQw2sWzv To34iqlxK6ySFGY2m47STluRgAn1PWnvJe/
- Z7VVDBPKwu4A4z1PSneRLN5qmN7gtKAsUWAzKBw49 hXTFKOsjLkpQXNUafdq70+eifzGQBhDI
- JHt9+5hIXTJDgZX6DHanokU0dskZSVmgKTlTxvPINOlI W6khSN4bcljI7c7iR1FY8cclsDKZk
- kYTCNdvZSOv1Aqox5tb2Zlypu0ZNPpZJ/kXDa7LuOKArLsT c0bLncc8MKjR/scbmCSF2LlW+U
- ZjB55z1qjKVM8rQtKNhARg3UDqac9xIzPnZsDEABeSP7x45xXQ qcnvqb+wqOd07p73Wvz0L8s
- WLRPJuT+7GNoY5ORlj9BVO3jurbTJJon3rkB2HVQTx+BpVtRFEJWk aTYwJ2HA4PT8afcuJGdo
- pAUmbPljgqegBpK/wrVCoyV7Lb009P8AhyENc3DLATvVWIVVAGcntx0q nKEjuSLkhz5mFI757
- j0q0izrceW37twu0joVz2PvVdEmuZI12KZNw25UFiSeK3jZPyNK0eWL5bKP 3W82ORi1tFJbEy
- SqjIwz0Gev5ZpskczS/u0cqJdse3+EY4P41f8ALMF3PCkEsUjPwD6E88e1RXNt JbLA0Eh2SSk
- ZIztUDnNTGor+pwqs4S5Oa76aXb+7oQbJZWmA3BiN4B5IA6D602GB54gAoXEvDEcK jfeB9we9
- SAwi6liEu/L/ACEDHy49x+NSXEix2lulnJ54lVlTbwSFOS386bk9kTjsRFqOr5vnp/Xm jKIxO
- Yyrud3AHU4p6t5yK7MyBVJfA689R7dqt2iwQXQkQTTMqsDznJYZ/wAaRzaw6cEjl8yVwAM9 Nv
- X065rR1NbWJWJ5pck20n1X6joDHDcK8ToxbLeUB8xA/p7U2Uo0MZWCQg5yQ3B79Paq6wkwqJWE
- LKArKwySSeufTFNXayNJFHLJHHMFByNrBep/GlZXuRKvClUU5J39N/yX4Mmfa1uqCEMyJy2OM7
- uv 5VWRJwkpkmjBEu3KJ/CRwat3F+0stwiW+3c+SVwMen4UrXDvE6LAC6j5iQDyB7inHnS2KhG
- rUd4p xT7P/gFaK7gjt4YtwEhO5POG4DngYx9agke4VVljhh8k5LmOIBeTg/lSn/WCRooZBGhQ
- nYMDcMj8 fQ0ExbEgYvEsURVhnv1A+uavlSeiPMqYd05W5eu/UjWYRXcUs4jni37cIMH5eAOnG
- anjluIdQBji VnVsyIY8/Mc4p8c0ZiaGUxqZ5lmZdv8Aq9vY/h6VEzCG5MqRzRu8oaIs2QAOoP
- qaN7po51rzqS1f f+l+B2llMxska5nRWZlxH0fhjkZx6VfkurSG8imgmjCKrjDDIznAH6mqVpm
- 2vtsiDgsmGHVtpO4e wGKlitpZtKhWNobjYUMhVerDkD8a8OoouV3t+B3ynT9o+bVfcvysi9+8
- mhlEaygs6uATypXt+Hes 2NZLieYNKDJIWYMQcL/skevFMf7VBPNcTHaIpPKVRxuD85rW2Tr5j
- w+XuRykI29j97Pqan4Fo1r/ AF/kehFzi/dktdv8imlm/wDZ00bM8W6aJmA788kelaQtZElZtz
- QmNXKeaxYHPXA/L86rSMRaxTWu RLt2lXO7cv3dw/DJrXtYRFfsTcLMY4WhhB5DL3P19/asKtV
- 2vcynWUkrS+XmVLe2uYzbKrtJhFV2 U42E87fqaTasGp+WY3gzEziSZtyLjtj15oW4aWztkMcu
- 0xlmCNhnx91gao287SSYkV0UDZK0h3Zb B/LtUqMndv8Ar+vQ8xRk5ObXy6/16otG6uTZAPsWR
- gHyF7rwahW0lnlwd6MCUyfzP+NTPbXLWP2d mWRSo3FRyR3x+HNSxypAojSZQJv30LvyGC8H86
- fNZe7uPmUIpwfvGch8m7JEU07qAD83BBUkEe1R /aHazhmmikP7t2jIONy9GI9qutcx6jBdNar
- 5cjSxu3sp4K/UU0soysZSaIkiIDn5QcHFaJ91qdSo QlNSqL3u1/1EhW2/szMcheQsNuDyF9a0
- yktxAqPD8uMFvXJ5P8qW1Jd3tkSLaWywwNxPqPRQOoqZ poUtItsy7fKcKxPUMeP8RXLObcjL2
- jUlFLW/rYhit0/tFCrFjHHJFhSRjdzk/kaGtvtVsRFMPKEi gqDyw7MPbNUo2voTsFtNcyhfma
- PgH+9+lXtPaOSWWO1Y2t0doiMx3L5XOePz5pzUormvsdk4VEnP m27W09ev4GZeRyNqgUyGXJw
- Qh4BI5/AjtStBO4jWBH+zCM/OnHLdKu3wzZxxuQiOpcy+oyMVVj06 5uDJLHOYDgZkfO1Mfw49
- e9axn7qbdjSlV09pOf36r8BdLtg0iRzOwY7S6s3TGc/jU5tyuqxXUayS wupZWU9z3+nAqZ2WI
- Nl1Z2cFdowVXHIPvjmq1tatbqDbSM8GCIAWJ4/r3qHNtuV9zKVS6lJytfQb FHkxvMrgSFXBHA
- JGefz4olt/tN1biWSNopEZpIkGHU9Ov4CrTwxfYZCbpFMkqzhsnCBSPl/GnSSw RXjebbytE0h
- wUIGTwQfoCeRSU3e6ClGVlNKz6dPzH3HlWlkA43NKnmNsGDkMFz9MVmvbW7I8tpIW aN9i/MT5
- g3Y3D2HSrSpbSXZkkSWQFGM534GfYdh04pGjUERrYXCFXKffH3CMsfwODTg+U6NIpaO/ olf/A
- DKEr20MVxFKu1TKuwLxgdh9ahkt5vMuoo8gKdspfnL7f6Zq4+mKkhKsbna2WP8AexySPr2r Su
- CkVmpjgknadzMVUjcAMdfw5rR1krcutx1cV7N/u1dv8PvOQitVDQxRwzTDaFnYNwzE4BH0rq4o
- 7qOzkjitpGJDEM2Du2HrWbCmL0T2YMX8IWT5twznd9B1rSjjvLvUjsLEhyFK8BxjqPaniKjlu9
- F3 FjpzqNLmvFau9/8Ahh0Ane5mknWOKBnVk3LypwT/ADqK8RhZWiSxMshJAYcYI9ffmpxAIrc
- tdSEg Iqtg8ZOQP6VflMk/lwSwPAqRu0hfHLgAcelcbnaSa/rQ82rGTnGS2+77u5gw290JIti4
- J/iI9ODj 8elRYmiMYlhlMYQqxb+HBxgn1NXltilmwS6BAwXOT8m05/8Ar0gu0lhuXlvLaYm5G
- 7YuByOP0rbn bba1GnUTbi7p76NW+ZFj7dI6wzRxgEeYoHIfOFH5CrjRTg3SECSZpt6uAMHPH5
- CorkQwz3TpGSxl VZthwGOARj0xUUEp+0XT28zFzJtQuchRg5H1zUvVXW3/AAxpXhBpOCv69zn
- r6KSPU5PL2xXIJXkc upXkj2qs8e2VjKGESPsUA43YH/6q17oXZEMsUL+aqiMhhuOSOP061SmU
- RWSApJJdRBUdc9zyDivT p1HZI9GnXqStZJ306XXz6IpPs88AecJyN8ke/lWFDW8lvvljI3tMr
- R7+T/u/rUTNJNtiCfMsLJLJ jktnI9/arEMkzyyJ5yRZUuodc7eOe3auhqSRs6U5Sdvne70+Qk
- 0kkL3CMPIM0sZ29GVe/Iq5LNGj 71uYpUVHWJFB3HJwDn8abZQGc4klVkEPBIyX5xkH6VNLLFH
- ZxCGITREEblAymD0J9qwk05JExmva KD/Db/MzbhI83CKrAxEDMZxlsYJrRFx58SPDL5CiUcTf
- NtPQA/WlaCGJZLYyjYzeZI7dWA6kH0NC GxaXzSJCgIRAp+6G4wfVs96JTUlsVXxMeXl5Wvy/E
- 2r20NxE0DuCylgTGMZxg1BbzW2VkMbvKq7d in5sMfvZ9AKhaZZI7iNJDJNFdxqI14YhRgjP+e
- lSwXJ3T+TbMsjuphDgHMecGuFRko2f+R50lKKa k9Pu/wCGH3gSW0IidQqEI23jzGJ4cewqmlv
- DJqMZkM4iVGH3zhCPX3PanG3eC/u13eXEsgEYY5xt HQ+5zWsjzCIAQCR41KthRwDzg+pI6Gm5
- 8itFmksQ6KUISvft/mytJPFJppUPGRuyQv3lBAzz9RxT 4b3zIzFAMRuR5bv8wxnDD9M1Uku5D
- AqxeRBHJC8gDplowP4T6n3qlNNG9jbnzEgjlXfuHG09hx68 0KjdWaLVOpGEbrr6/pqad9DcXU
- yW8GxsRM3mAYEhyMEVnb9Ta5ngRYy5U5AXGTjrUrXlyLx5HzEs cwjLEYAJ6UnmK19CFuVEkQX
- J67ufmP4VpBSirNI0hKdOD0TtqvX+uhVWGCSC3+0Q3abmQqHl4UDI 2/UmtZdRsbK/INvI7YOV
- UjKY42k+vNUbrL3f2hWE8MTeW5j6B2Py/lSLEftl24mha5QnfxkMRySB VNRmve2+ZdKFOpFqT
- vfZXdr/AKEkTxXV/NECYRDIApdsgqByappcmaQKsokTaG+TjqeQfUnpSCeS SWWeaL947oSUwo
- +bgjFUv7MuIZ99vIiIFZH3clGwflPvWsacFe7sXKGHjNqTs7adfuLSX88mqbxG YGLbiWORGAc
- YIovpHMzwbmLRLhXB4fHPA96itwohjC7pCoViOrMxHHPpntVq4V5I90jxwztb75lY fxeo9ABV
- +6prQ6aNSLrptWT07/8ABKzvI9raFkdZHVmUnoVPGfpUTzPbSTNbbnmLNGxH/LJemD9a Y8jwX
- NpK8q3MKpx5foB7+uaq3DSCFSm6Aw/IMjls8kn1+prWELtdjmjSndqOsW35X8thY1cXqecs vk
- yL8zk9fRh7VXcSRXJZ4+REwYkcEnoRUiTXb3MKwSRHCO7Lt7d1plyxjsEL5BGIwG/iOM4+tbpP
- msy5VZqraVlfTf8AIqGCSVbeKFHlmwQ6ge/FWo5HAlt5Io1Eis6uVwVHp+YpYGmbe0OItkgXOO
- Tn nNNFvDILePbI52Oxw3KlTnn61cpdH/XUyq15QldfLun+ViwJbn+ylQIJMDG8LwT3HvVSxjm
- uZopi 0YhjYCR8dPr71JbzTw2xuBbyO88jAgdFz0OKnt7gjTJlVQhZzn5eGx1NQ7pNJGdKDqLk
- TV5X8vzu RXweS5jKlIlDHeSMkjsayZJZGnQqjnOckfwntWlPHELCAo7lpOChYkjHf8qjy8cu6
- Hy4x1CuueM9 +OtbUnaJ00YzVJ8t9Hb7vkLEIyWSdZ43ZS3mGTPOOTUbK8mn28yszRBQshxkKe
- w9s0wvJHMjocbQ VAbkc05i0NzJGfly43IegK9M0Wd9BVqMo1PcWi1syFTJvWdiFKMVbK9Djp+
- NV4IxIAzYj8s4BJ6e 3HrV24jaWzhnWGRRhlkOcjORj8ag3MkBiZAJiWMuR159O2MVpGV1ocUJ
- e0nzWu+t7FhDD5kZIbck Rf5TjkZP49qekME1qtw0iwsVdSrDp6/zqKAv5EskEWGEyx4b5icg8
- fWo/LdIFikBRhGwIYn5uefy rNq70ZVaUqtoUrLvtf5+Qy2e1iy5jlkhUNH97JJz6/jSS2bRws
- 8BdEjk2ybz9854x+FCBUsQ0S8g 4k+XIGen4moAsko3jayxnbNhccnpWttb3OaVKCirv5f09Pu
- Hs0RikCh423ERqTy2DwatSQ3FxcW4 hjwrgKTnjpk5rNjiENpI87Kk8c4AXHOTz/Srq+UksCW5
- lmnCNvCE8EDNEtNjF1alkue1+r6evl5l MxsbyzXcsm7JcIMYHfPvgVanmtnnDph1bIUA4KjHG
- Tjk1FH5jzwLHJGzuhwqjnGOn1okiS3ZNp83 kF4/4l+Xk/mabtzHHNxjUum5SXTa/wDXctQShb
- HyktgWcqS5ALMp4bB9OBUoSK1ncEnEPzQyPyrg df51m2sP2G2SdnMi5C/7xJ6j2FXEdJtRBu9
- +1W/ebc4z34/Lis5x1dtjjjzTlJ04XW/d36qx0mJI YoGaeMSIrZkbnDZ5/TFNit5BNnzcKcbl
- jcghgcqD+HP0qOyjMEK26AZWNmmZ/mG/cMD8q15khfV7 ieQSTAsQqwttyAACfwrzZz5W0fQU8
- Q6dGUZW18l/X4jF2m8kmAdojkvuORkDIpDL9ujhMEc0oWIE Kr4MmeMj6HrTpJ4ltQsDpHsmSJ
- S/IJPr60sVisNxFFv86VS77YTg4B6j256e1Y3ja736GUa9JpTc dfnp93+ZTtIEl01oZZGt5I5
- VJZ2PzAnqPY9MV1SvNb37TxlPJjJjkyucO/QfhXIrczi5UPNBHHFG FLMmc+g+prZa7uotQKKU
- MLSYPy/eJ5P49qjE05SZpjaEX7tlZ9/00NGS2mLvsUiQcEjgAL149xUU peKWGCG7tFMgdolaP
- 5iB3J71I01vkZS5QvhnPmfdPv8A1qbT5Z45JIJLUXA3j5gozjknB7A+lcd5 JXfQ4Zzkknbbul
- /wbk1s9zBbQyzQO6xtGkjADGOf8ap6hdR3MDwQLEUjU7NqgEBSDTbm2vXKokhg GFJDfxc4/wA
- KiL2kF/JIx+0SwnYY4uCd3b60RhHm5t35Bh8Iub2srN72X+b2EVoy93JeWlwqSTll WIhCwwMk
- Y7Cqypbx2gjimFsInC4lOTwcn861LkC6WdJswtET8/Reg4/HinpKwtI2je0VppRK5eIH Cgcj8
- 6r2mn9W/UiUpRkua+r26fqZ1pZzzzCeItLNIoIKHAXJwRWvHCY3kiNxbNF5mCSnfsR7e1Zo eW
- aUEK1uxfEQU4yv8XTvnvV1pIBb5lvIWtw/mMVGCe2c+lKq5SZ1TxMpSUNL+Sv/AF+A27zD9pkc
- Sk7gpCNjJPYe5HIqGNrGIRbFltTs3J5j5IHVQT+dTXd/E5ZVhkdGmCtIDwW45/HoPeojfSS6i0
- bw oI43KOpXlc9j9AKIKXJqvxOxVJey5eVu2rs/01Qtx5H2GCSIPdAx5ljVudwOQR6AelWbG7Y
- RTBri GMySrIXZflyB0x7iq12zI0k/lEIW3bBgFhjGR6AUs8ccKRoJog23OTzuyRzSsnFJ9TJw
- j7O8ou3f 9LDbi1nlGzzFMoZWdFGCgU5Ofwq28Vo8M4M+N7CcMHICr0P4YqTzJ1WScqLhwHRTG
- MZ3YFZNpbTR 6qEeVUQBFk3gnaQOh9jzSi+aOrtYmlUc6LlN25dv6szYntYI3gWNHnt1gcsqNy
- W/h/DHNUYmQ2lt JHDMp8rJ3tuDHPJH1FajTwRASuQ0ZQ/KOoPTH5VHBb2qTrBBLulSNiIy2dg
- A4B9+9ZRm1HW5WGT9 nZxbXfW3z7fcUbqZSGGxiVyqRqcErjPPqalWSS5szugnSOQI4Jbrg5b8
- DV+OG2hsoGkIuBIAQVOC +eMj2qNrMCKWBVm+R1ULu5UYyc/zp+0haxvGtSlBJ7973KzRl9ZaE
- EtGwfeFODG2MgH8KznkWKS3 mnt7pSVww38bm7flVxLEFFnbz5N2HiZXxuHOfrntRLcQtZwyRT
- RQtNwiT/MUJb5Qffg1rFq6S1Gn T5+Vaxfrv8v8hph+x3TKkMvnuxdSzZVQq5xj3q3a6jK9lDc
- +TteWMtIqqBh2GOPQY7VWYmdh5kou 5gvzeUcc5pFWCNJ3iLRSk4G85X2IH51MoqStLcirTp1H
- yPfv/WpNFJILF0wIgkqoWlG7cU5/TvVv CXdzdebeI7ySsSyEgEEdvTpiqsBkSNFZfPUHcSPXt
- mnRXCrNI0mn3FiPOBeSQggFeQPy/nWbi7tr f5f8OeevaRcrLXuv+DqUTDGdoil2wkAFCST055
- p7fZjYtE6CJ1kCJ8uCwbGT+HrV3c91fyXG2ImU NMiIuMheCR7UyVtLk0+RlcpsdFikZ8iQP24
- 75q+dtq9yo1Jykk38+3yM+6nl+1xW8LpIgjbLAenQ n3ogeSW1hiht2nkj4LR4G75t276VA8qQ
- ube3ZEniOHdvmBwDnikt7SRrqO4dnjcxERhGICrg5B9T 71vypR1DEQi6bctLbX6v5WMe6kv7b
- UWulWWK1Mv71pOQx6jHpUMaNc3MckrEbrZnXDYIKk4B9TUk 8M/2K0CzLPHcASbSCcEHA6+9Ub
- m3kju4wXYXCMXlT+8wIyB6DHavSp2asnqdtGrCcLxfvbdfyZLI bIWoYmRZnVMpv+7jgj8+agc
- R+fcxKkkhD7Bg8qR/jS+UJr2Qh1hhdycuc/KD1q0dPu7NIZNpkGHy VH38dG/WtOaMdL6s9GnU
- jBqCqczl3enpctIxSO0uACoERzg8HdxkewNVY5723sMFAVhbycFOpJya tjbPbZ3iKRYyqs33S
- oAyMfWs1pgiXWYJA00wcAvwFxg/jWUI811b+v6uVQp80mlDm7+X4luW8vhN bS+XGrCBsZQEbT
- 1J+varcNxZLpyGJtsyKqENznLZz+FZ8TQ2jPm1m81W/d+ZJwVxxnikzeyabZGY 28ysrZZIgu3
- nIBpzpp20sv6fmXWoLminDlT7WX6M07qWEIsYKNsBLSR8ZyeCT3JFVTf3iyW0cGwk rnbt5AXk
- /pSBUgsbeVLyFZJfmfcpbkHjseOoqNrW9vLgxrEWQzDa6gDaD94VEYQS128zhdOmoOUt I/3v0
- 0X4M0lvLoC3lleKWFk4VV5yTwSe9W11SFY5JZbuG4KKVPloV/n1J7elcxIJFvAVimhkUgkO ch
- QeDx7VPMkLRK+1ZkDsIfLG3egHBPrnsfaplhoNq/4GU8JHmSavfayS/H/gpmrJqFg93HEEYxcc
- nsAemf51RubssLNjbhjP++Cj/lmd2AuMY7Z+hpLVHisprgRqqEkB5ACAeMD8aDqFwyuCkUUp2n
- DR j5SowR/WtacYQulG/q3o++htLDQi709fV7fcJefa5LqNxMirFceTJleDu5yfpWjBiGZDJAx
- fa4Zi OAOmPqTUCR2H2eKNoLy7ZNrkxy4z1IP40xHQf6Q9rdLCylipk5JHf6CsZaq1v6+85Kan
- zcq0Xpa/ 43+8049OS18kwM7jbiWLdkh/U1jLdTwyiXaI5HObgMMliTzj0xxTp5LloI5ITJbxy
- WxaZ3OQW6Ae 2f61nQzgWke2VRN5ibo35J54H+NXTpSablqdcqUoxu9blppoZLeaS33yFSqZXo
- CD/PFWEAjvwrPK SY2bhvvAkY/E9KWWGVbmfzWigRpG34XAVl5B/HOKq/ZUEkbXF4kb+WVZDnO
- 7GcVScWtyZVoVIct9 ej1+75E3leY0k8EcsbIdpg3fNGT0U+p71YkW+RkN1F5khhcbcDPUbs/h
- xWcst4IFdUJdUDTcdW6c ++OafZpK1/se+QSpkRFskMvf9KJRaV3bQVRVFC7Sdvm/vW33Er/Zz
- pXnW0DxxRNgCU7gw6j9aGKT rELn95ggt5fy4b0NRyTSS23lKCLO5w8TY+6B/CT68VVMrrM5Y4
- YOgZfU+v41UYO3mP2EfY3bSktd 29yyLdDcR27q8IWErJJnqd2f/rVTt/7PmmRJFljdpCWkd8o
- GB4bHp2qWS6nn1JLeN49rI+Pl5OOp qqyQ8Qput/kG6RznnqoH1rSMXbV/cYVI88rzbvbp089d
- /wARC9xbPMDGGiMh8zHHPYZ7U+3e6eXa 0RVW5PGOp6/Sq5t7qdGRElJR13J1Jc//AFqha/ZXJ
- jY5V/lP93HUH1rfk5lZWubzpp0nHRy9UW3l ZJiY3WZQpB2jjHTNQx2iuin7TGuwFyCTwMfSq2
- WmkaV1G8tuOOMjPI4p896jo6wwtsJCIVP+rHXk 96vkktEdPNZJ2t935Dw+5U3OoTYwXI6dqml
- Ahht3ZPNi5A2Hkntk47URCc20E0LwtKWWNY9mTjqT zVOWRXupd8wH7wmPB4Zs54/lSWsrI46u
- NVSTjTaS6/8ADdCeGa1ICybS0bsOv39vIP41LFsuJdvl MjyZbnk5J6VQ3SpMSIQBv3FSBkZ7U
- z/VuFG/zFYjBP5VfswhGpBOUrrs+hYkeJd8Rm8gFgAjdSQO vFSAI+9ndGM6E8DlW3CqBVVkia
- RopOucKTwO9StIZAI/MjjCHKMBgNj/ABzTcexyzrVU2lJvza/4 BbaVob9oYATKZc9M846Y9aq
- AfJPJJJvZCEAI6ls5qwsTlP38gWRm3JHn5yR05qRLqZmkiEcaNLvI VkGRkfz9PSoTtsZUK7pu
- 1k+7/wAylGpEsSNA3mJEwBBxu5yMjv7ULDJMziCIorANOc8f7P65q2YI 18mZdSi85YwCNhOB3
- zxyapyzvDelrcq28MuMdBnIzVpuT939TJwTm3Tjb1T3+6wWot7mGETHyiqM rk/xMSdpqCJFVI
- w6SRtn52Z+p6enAqKC6la5QxxbHXJORkNjkGopZQIozE2fm+bqehz6VsoO7MqL hGc5Nt2Wiff
- 0t+ZIXcruaN4m8xdgHGFH3qr+arTqQ29GkzuJ4C/wk+3rT0EbDDSn5Ry27gHtmqwR zDHuPlfI
- VII+9zWqSFifaTqXg/Xb/hjaEsks0UciF0CucoMKfVh7VmebF9njdJ0bCMMEcsM/zo8y eW0jj
- dGLxJsiKHGVPUn+VVYYTJuU7EZpQFVhjrnjpUwpqN7nnSlOF3K0Uv6uejyfZJJIIpmYo6MI 9p
- xj2PqxPT6Vct2NrBI13wVcqzHjLY7e2Kz3vLR0hlMRmdmyfLOPmzgY9OKuvHLcRXQMUodrrPmM
- cpgYycV4Ek7JPRHqRvzLndovpdf0vuIWLtfqY/LC9GDAHDgAj8cZpbGS5urzzvvTmUbWUYCxHO
- f0 5qNbe3ublVknOxVkZo1YhiR0OalhnU6arAqqnl5FOArEcL+lOVuWyWo685JNxuv6+9mpeWh
- eVWgj VAU5JUENjG2mrJHJMWuJI2Zn80qoxsI4I/EViNeTyXJVplCyDzMAYGewH1xUkdwlwjef
- PFGzAvKQ uNjY5U/Ws/YTUUpDnQm0lUlv1SuzpZfJuZZGRhErO3LHhRgZP4YpAZZZNyxSxpFk+
- YDw2ev5Vn2U tsyQSwndMIBEATkbWPJx6irEAuAY4mcnzM4IPBABFcrhy6du5moxpt8zbXS/T7
- v1Lkkm284aQJuG 5mbhHHRfxpGjiN0UYpkShy6jGVHX8amSz/0LyDIr+UViyf4wxzu+vvReS2i
- zsJJE2cRIVOMqTz+P vWXMm7I2nONRNpP5Fie5iivBIkR8vyZNxcZCE/dB9T3rCRXtUjkQNPtI
- jkPbI+Y9auhp7e4JjQsg uVDbxnB6DNTT3jXH2mMrHHH5nyybflzgDH1p0046LYqhTjLRbdWVG
- d3gd0tZo2mcyFyeEzj8uOfx qzbXMLRRySxKVVWjQYHzB+Fb6Usst0kSxBoCVBibCdAT83446e
- lUIZY4bySIgyWUWBAw6vj7pz6V XLzRehHs+aGnutee5oT2aiO2i2v50RLMFPDOnH5Hpj1piR/
- 6MwkkWQlgwVRhiCO57kU1r61aWVjI UdF27y2AQwyRj1J71nvcXUjpECkm6RG2IMPtAznPt3pw
- pza1O2nhpxhzS6fL9bGsZDJA2AGjlILs P+Wew/dP1pl1bImoteSyKY4o3RohwcnHP4UyP7XJq
- WWQeW7MyqgHy49fXNP+z3kzT2qrucsCHIyD x8xqV7r3sCUef36mnXW2n3Dx9o/eNIwhtkQ5J/
- iOBluvTpUsECDSRtulupTIrsqfeJ9c+maxWiug yqzMVDAGM85Rj839Oa6ATIkMqJE4hwS7LgE
- MD8vPp7VNVWSsa1qMHFKK/L/hx7SQz6bKDEYi7M7u 33S38IHoOMYq1bQyujSSSwb/ACyCiphu
- Rk80NYh4ZzAHebz1CEn5SMA4x+dWI2uFkczTW7qzZASP ByOn5/0rklNcvus5nVdOl7jt5df1t
- 95DD5pie2ZVYeYhYKPunBIwew46VG0VuodpLoxebKjZYn5g RyadKzreTbSFeSQNHkdPlxg++a
- p3dxM9kC8kMEhIVldc7d3X8qcItvtc0pX57LRPcjnMAEkSzPLb hiUWNsNgjIAP1qO1vLZyslx
- beTuQuY2xu3Abcj+dEkMu1TbFXlWEnAH+sA44/CqcM13EIyEiEZcq xdAcfL8tdKinHc0lGKpu
- z/HX/gfcXfOki2Ss0T26NtaONMOx7HPoD1p6X8TuivGiqEGxyBhic5xV ARsJomu33yxwYkiQY
- PT5j+HBpkTTC4LxXVo4hXy0zFkHd3/HtTdKLRz1IRcHbfvdr8ja+0/ar25g QCQF1wiDDYC/zp
- ILazR0ncziQpuxJJlW54JH0rNF3II332kiSmRfLZQBgfdJP0PWnOziZ4JJFkRW dJAByz4DAj0
- A9Kz9m1onZGcnLrK0Xvr/AEie7+1LYvMRzuwxQbQrA5C/T1Hesze0cskjqsTmfId1 +QfL0xRH
- crJefv5SYXAWQZ4EhBwfxNWnuEhuIES/sFWMMjRyJubdjOD75rZRcfdsFWPKuSKTa1vZ 7fcyB
- N8bxreSQTQuE2+WmGcnoc+gPX1qlc3kS3Hlos8FyIyCzt8uCeTikuJkuYdOdEf7YlvuZQeu Dx
- xULDzhNcmIq0xBZDydwHOPQD0rohBXvI3jB815Oza22/DsZFpd3yW8EsBSUq33GXJzu9PQjNNl
- 2SanMJ2dS83mIQ3Rf7v1PWtKVnjtIooQkryQliETkn+8PbGarQQOiG72NbrHlA0w3AA4wenvXc
- pL WVrXPQoUNHVStJ6aeXe3T5CQ3dtHa22YJSEXEjE/x5JA/KrjI0UJnn87fOm+Nd2Ain74x69
- KnEyr oqoIlnJk4KqPmGcZ/CoIoLiW9uYoyymLgCQ7jz1rK6d3sXhaPNBt2i+uv9WGTGAWjrBO
- qKzhmDZJ Vm7Z9B3qF7WeK7/fxuiKjIHPQnHFTBElgMinziHVJEQYyRzSvb28dvBvmkkldwSik
- AqCcgdOpqoy UdLnZTqUqV4Qk7el/wBVb1ZVUMJYZTIjJEgZo3GS3GPyzTnEcYkllRlGAihWx1
- H+FRgC1uRI1vMG 7BjkDrgH8ancSmFoDBK7yuG3Y4OBjj9a06m9Kynq7J9bojR1IiWJEMSSCEL
- IudueQD79asFb+NXj gDxwvNuHru7c1atrhkKR3scccKjcvygFmHfPrV1J7VbdJPOVYiu4gnOx
- uoU+5rnqVGn8N/xOGs3z 6U1JfejGkl1CDzDdIsuJv3pMYHzN/D/9aiVZ4B5FzEoiR8DauMkHo
- MfhVy93/ZEklkRJWbZGhXqP 4mPrgd6rTRmTThCkcrB5so7vkMBjafxqoSTs7IwjUp80JOCSbt
- 1t/n9wk96r2cYEZSI5eM9Q3/6j Tra+ubhts9ks6glm8tQp3Hrz/SpHkuEsLj7Q0Fs4dk2vHnG
- cE44OBVIQSjzZ7hwsAuCWdDjJIx2o UYOL0F7KjKN0km27av8AD/hxZJvOiUbHhTap354BHf6e
- 1SOZ3MsVuDdKhWLCjqxGc/jjpTJbqKGV TMDHIynygRxt245H1qCGFVtgrTtFPsBZuSAQDkYH8
- XatEtL2NqNWT+z6aX1/P8SaYhLQSMsyyEDY rNkMDwTj0FVZLUz3AhjngkIUvHtXBbaPpV6M/Z
- NLDAw3O9AyLIu7aAcAc+5pqWVzcC2ZbWXKxYwg A4YkE/rRGfLrcFJOF5JaPdr9P+CVUihnNxJ
- NKyszKck9C3Y+tXIoB+4M13DNFhi0ajDnBx1NOazM EIjkKQSo6LGX/i56n6Vbit3k1N5LmIqS
- cB14XeOQMe47VFSqrXTIxUlyOXNb+vT8jPjna1ljGxrd cBQ0vzAqTzn3qcajbokiwRo8oz5Tj
- HT+79TVi4hsZpLRrmRk35V/mwFc5OfYCqJBjiiaG3J8tWDY 58wEffHoBUrknq1qcMHSqtcid/
- w+8j81JmkeZWtkVCqA9z2xTNQCsVaO6hcunmEKvJAAx/WrErwT 6Zbopw4X5M/xAkAmsqVTHqg
- j80SqqvGu0fwj/wCucVvSSbvtboddKpCO6tJdOhDM0nmxu8LnzBhN q/dB45q4xS1gYO8Z8wAb
- tudu04IP50x0eDTkSJHaRic7jkocfMDTG82cMiSeZlecJnfjnI49ua33 S7FU/rFWjfmio6/P5
- sgn+3pO8UZcsjkEhCN2cD86rvE1tHKjJyjgMGj+4wPANWhc3KXhmyWcxEBi vr1P1psb3N3HHb
- yyROpBKOQBkepOOee9aJyS1tYxqUqlKVpRjyv1uvO1jOEk6kyeWcs+FG3r6ipT GjJcNvFuVO1
- FZfmIGDn8qkMM8Cxx3Mi7hJvV1U4wOuPeniWVYzI1utwZCx2bc8E8GrlLseXVxddx Sadulv8A
- MzoXdQBAX3MzNweQP4R9etKSq24Blhdh8oOPf+dX3tJtjIqx28LPuRto5x3GO1Z32Znj leWaB
- nEuGKAgA5z6VpGcWysPWgpOKSbt3bHLc7lMaKJY93oNwI7E4zTkSKe9DeaLcRkMu9uWwelL Fa
- qkErySpLI0xKiPgYPQ9O/NMaMecEMErMCACvc9qfu62NYxc6EmtL99P0ZYQmS8kbIU/MqLt/TF
- RMhEm7EixgfddefpnHWmGF0uoVWQu046AYPLdfzqyWuHtpEOGCykDj7wPBb6cUr2ehtCpV51Ra
- vb tfT5oa1wUu4JW2rNj5MJ3qz5swkWZvLWZZN2Cv3OxUiqXkNOwfZJjGFbrViK2H2+JXuEhyp
- ZzICQ p9xUTUbGePgqsZTdml836P8A4YrynMxKxkAADA/hBPf6VE63O2N2gbBVuR147/StKaSG
- SwV4ZUl7 ZVcHOcc1lF3N2+9cBWwwI61dNtoxo1Jzpqzsuy/pNEkc0sEUiKUULIAm5eSp/iz6D
- +tVpJY47h9x S4TfxtHbp3FTQqbi/tzIFmTkSRIMMwA4GfWmQBPMMUdvJkgklz9zGc5q0kmzj5
- KMpyXNZd3/AJLX 7ypgeXIY1cLuYEkVJ5RSFBt3ZUsDn17U8m1eCJ5QUkYf6scZwfXHtUCwtLI
- zGTIV87cfeHXFarUh Jyn7sb2+V/k3+Y+a2kKKYX3KuN+0HpkcVUkjeS4llfchMvmDb2rYciWG
- WW2Cshk2hUHKk84PrSyJ EYTEbaRZliIdw/3mz1xjgVMarXQmcY15OSp38lrb+vI6d7SHyo7TD
- QqVaSNyfvYPanXl1bGysBIl 0rlSxRZMYy3zZ9cAVopduW2vash3ncWwfn7Y9FA5I70RWwv3TM
- 0MrxwtyF7d68L2lrOfTzE1FWdR WUddHcjtkt9zeS26N3bAzyRnHX6VWY28jSWcbbYxs8pgfvo
- OrfhVmHzYjDHFBiKMYDEZ+g+tZslz AZLi4gUJOspiQMPuqR0/PNEE3JmeHrTvJNN9nfb1NdGg
- +0XEolgl2zbIzjhoyByPpT4rNbS4uT5k RhEnIZeR3APuaxwn2e2kmmjdmFwoVFOC0Y+8RW3DL
- bSa0pfdNESwi+bgqRkZ9TkcGs6kXFaO6LjK avOL5k/QtLNHLFE1pHGxRdgRQMrnk5+lR2y6gi
- Ncy42wMEK7e7cZqaxh8+SaKONoWldZd2e2Ocex xirssZiuo2ihuFhaPcdzZCk8BD6t3zXJKai
- 3E6sPZXi0n1d7DnEcdmCgfYsTbBnkg9CT3PWs37Je SSW8fk7EZA+9xnIHBxWxN5l1axWyBo5t
- jqxIyGwB849FGOlQyJLHa25WUPujyoH8Kk5P+NZU6jS8 y6fM6ajDdt79CF5LhbKVUMagziTLD
- O3HrTEuT9uBQKfl3jK5DA9xVRzdF3McqyqZhIgUcSKBjcPa taIuJDLKq2bI+N0q/LgdBx/nmr
- klFG8YU4wtJJt+n/Dlbzrh5SEmgV3lU7imQcDrRdyyBF226TbI 9kbIAA0fBLfnxUjG8luXeGS
- 3kVZVBjWPBTuwPvirYSaO6iuI3iuI1R1RFXkD057nPFRzJNPQy5Ve z+S1/PoZ9wy+TdGPy2Pm
- lom28AYHBHc9cVRSe5Nx9sS0O853HAxgjnj6dKddW3nwxlnMEsiK75PA Zc0kRwIo4nN5NgyMs
- fGG7Kf89q6YqKj3PWiqXs1bV9d0RxBjGqrM0EZYFVkyWAB9fep7+SWO8naO 43RrcRuETIbkdM
- +3f1qO3ZppVkmjeIhB8x6M2SRgelXrmQhbi7URyvK6gbE4TgcH3obtPVf1oaqN p2klrte1vvE
- txOtzG026XClJSO8h6Y9OO1X7OO43zJ5f2iNAE2qOR8p5PrVK2iaZWil3gSSB4W6f KvJ/H3rS
- WZIQJwzrK0W488NzjP4A1zVm22kcs6kpScUlf7whup4ESJIpZcRjZtPMvHLD6VYiurh7 ou0kK
- Z3YLJw3HUfyrPE48kq2YtihWJ6ls/Jg9gfSpAgnnneeVYTJJ5hHQRYGNmPWspQWraJVJJNy SX
- 9eo6dg8MMkrfvYkZJVXghm5/QVDDO7hI4Wh+V1L713EHsPxq/bXNpsaCORGaSMyOG5YN1UZ/Cq
- DS77dLown7VIELqvAAPXj19PSmuzRM4pq0ovy0/rQtSC9innaW5tZDHcCIrHHgjcM4rNuLq4Mi
- QR RDlWwSucAHk/X3q3vgneQhJIiZlU7myWPY/gKyLuC98yVFdFjw259v3QDkj8qujFN2f9fcc
- lOEHP ldr/AHfkZ83kyS284mk8t4H80lzwM8fjV03P2W0E7IArH5iR17ZH0FVXgeWGHLI0Usys
- MLjKDqR7 dqnuDK13PIAttDubJmXcoZugx/nrXa0nZMIwhPeV7f1p1BbxFO/7QssaZT5f+Wn+2
- PYfzpEnSVD9 pmQ4kR1C8MFxzk989Kql7h2uFhgUOR8yhRw/YD0759arGS5+zRo9sdyxKJDgA5
- BJ/SrVFM66WGi1 yvWT81dG5MqRXF00RyGlVkXHRgOB+IrLB3zubsJamaQ4Mg+6wGMH371VbVS
- u1FjL75hJKxHH1HtT bq7u5lZCglgMzksiemMNn0qqdCa0ZcI4ty5ZRS83v/kTxQzLE8oxLJlS
- rJ3H8RHtWi87NaSGOWFX UhQdvAz0P41nrMv2WNJw0blHLyA4UsCOAOwPHFSfZvNt7mUSAxyzI
- Qo4IbGMfn2pSV3eRo5VKslU qO3S7XYcun3P2ppPOjJiby2Kg8secj2xTjp5JnzcKkDIXwe+0c
- f41FcpPZ26qUuHXd+8YtwGGOtX p4FZzbHcjSTtLhj02gHFDnJWfNv+h6ClXp8slPR6aJFWyhS
- VTunV5y6yjaCFCD73Hv1okjf+0DLH e25jD4UKCCw6g+9MgvLgXkjwiCKSXlFMY6dx9Kcy3G2K
- eV4gpXcE24LHHOP51UoyUtX/AF9xpUw7 jUak7X9L/kW7ZFF+u5M+ejSOq8c9AR7d6ou0tjaQr
- 9nlVhhQ7gNjnPNS20aNPbubgAG13uRnjnit BkV7me3Eocq29QfUjpUN8stdTWdBxnd2aS6r+k
- Z0qbVmBkimnnl3bsfLhe4HpUqs0FnFJFiW4jY+ Vx1TrnHvzVoQRNbwyXPzSJExJU478n6Ypsc
- 0csEiCaIlySkm04CH735Yqea62MlSlV5U1pfXtb03 M2RElTfMHDMf9Hx3TGc/0ppeNLW1RYjb
- mVC8zy/MpOcAgdsCn+Xd/unaI3VurAKsXoBwc+lV7iDI Vmy+CJCg67F7/SuhWbs2bOz91v4dr
- flb/NE9xbNEI5rrfLAqMigNjpjHPvVOK0WVVE100crksgLH CAdc1Ok4urtsxTyxGTzUi3fe68
- /SqQMshlgwI3I3lWGScDt+FXBTtZuzMFGrUnaUrNden3bFqA87 1PnKSN+85ByCB170yUKWidW
- WcxwiAoP+WjHkMP8APakniWHTrCSK4jEgiI293JORU9nEJraBhhZZ jvZz93cCen4DH40m0lzH
- FOUYx55yur2/r/gEQQW908+Y51yy/OMkn1GegqOK3jfzhcNLFNGpDHP3 8n5WH4GtWWS3lngSB
- 02yRtKQRnYc4Gfp3qvFbPbLJvjeV9g2yBuGAyTUqr7uujNac3zc2ql9xYEU cUEsVvKkciuBKs
- vzcjnirqXDSbHS3n8wNmNlOFVemCO55rMhmRmU8FWIKKPvKP8AaPc571ektWUz SrLtnWRfMPR
- cA88Vzzir+8TUoc8rS/Hq/kZ9w8wjQ7HVo4yCZOcjP9DVy1uZri+dpLSaOUk/Ox+Q DHPHr6H3
- qyrh7gwzzQMGYuWC+/8AKq1xLN9te6kjZkMZA2fKpB4/Q0X5ly21Eo+3vSsk1/XkZKzx rcx+a
- CRIQVJbiMkEAn6d/Wr0L3L6itvbR+YsTbJ5Qvyl8cAegPTFWIms5dFUtA/nQMICM8kn+tZs tl
- Khe3t/OEzSDZ8331HJP9M1reMrpq3qclSEailzaPbXXb5oi+0+TfszW7RLtbcrYO3jCgVbZfPh
- iVkEzn93P5YwUOAP/wBdVZEtCZJCXWTzdpSRs4BGcH3ppmjmvnyJbYNzIhPIfHB47DFauKeqRN
- aE J01Kle662/r8yMmC3leIxuJ4wUbe2RJn+ID6VVSFzeRRorSxyQlgyenc81YjgMtteebNEhW
- aIq56 learyLY29w2VnkiUOqsr8fMOnTtmt4vdLc6FOdOLjF697f5/oTx/ub2ONo9qqDlnAyyn
- qQfaq32p RdAxWxlzGEAU9Ewcj6571CkFxFbo00UuyMlWOcD/ACeKvwpbSWpkKSCJ0OQDypXqM
- 02orV6nLyp+ 85N303/4O5WMM1vEbgKscRjZAJl3enHSoXnuXcweV5sMbFMImCARnGce1aEUEF
- zdopnJjERY5PCq BnJqhNBcIlvcvFKF2byduATng59OnFVCScrPc0aVWTUtZR20t92t/wAwtIF
- lZIvNcrjKtu4UkUGF fteLhTKYh+7CDbvGPmPvg1Iu1pPMLfZpwW3uT8jEjjA7Clhed3sY5XUR
- SMCxZeWK5yoP5UNvV3PL rx5FKUtu3Vej6/eRFriC3EoWN1kQMxVeAM9qYIo7maJpX8hUT98c9
- WPTHp2q1ckwtFIssXnoDCyb crzyePYGhrUPZRyJFLL5kRyUYcso4/nQpq19r9TSlWgqS520pa
- XMv7NcxiNpYmTjHJ5OSeR7VItv LA6wNNHKsrMEAXczlecg4zj2pryO/l7kWInr8vOPX6U62ga
- e0ZmUiQSkDH3l2jJP8hW8m0rtnTXi qU1epf5NX+5ldIo45l88svTcQMHnn0qz9oimjieSCSaR
- VfZGBgqBwNx71F88l1byMARMNxyOG5wM VYW0NybgF7cRQO2XRApfAxwfTPalNreTOPHKlKa12
- 6X0/C5meT5LbAGIVG6fxd8/1/Co5pxHcK5M W9n43LkOcD+ea0o5LZDI00ckiEDABwckc8+nbF
- VSscrAgKqEqEU8sB659q1U9dUViW1C1rfiQxPI kgSC1kkuTI6h1bhOOmMdcVVdQiWYMxilkyZ
- Cc4Htx61Zni8iVUeJ0Bz8z9x6023k813liSPdG4ZR ImduOo5H8qtbXRwywzmrKV29kv1fYI7V
- IUKxxzuWJwCclW7Dp25oeR0kVUaFlEyIzbcjHr06Vaju HkeNZ0JN0jyKy8BcAjn8qht7tksYE
- PkyKExuVB8vPfI5PvSvJ7oVKc6kHTi7fp+f4Iu3NusF28iQ uxjuTsWI7Qigc7vWiEIL1wkQJz
- x5w3eYOuRntViFx9kObK7kvuszE/LgHOMfStG0vtOvpnikKwRS MW9CvbAP4VxzqSUdr2PPrS5
- IStd269fuTJ20/wA+KeSMTxgToIwz5x/ez6nvVjyEiuS1is0oQneF f7x7Eegpbszx6a8TSLl5
- MqyDG0jBP6YqvatcHUg7zxso+VgoxlWPP41x3k43b0Pbmp1KKm5K3bo/ n1LF2LuWFGYMgVWEs
- YGDvP3RWAkVxJdslwoiRgN4xg7sYX9a2ob6KHU5o2indMjcS2clR8h/Kqtz qNzJayGPyTHs25
- MfJJ6mro88fdSKoOvCXso2Sev3/eTyxKmmOyrLKN6RFlbo38Q/E1pWV1KqxrFH CrR7UcMuSpJ
- 4NVpJAmlnzEJjWVUYDgqxAxn1NWW8kW8CHcZEUiQo2N+T/hWE3zRs11JiqkqSjP3r N7frc1mu
- DatI0sLFA21mQ7RyvBHtzSSWxa3ha2nfMcJCB2LYIIwD6k9qzxJpiy3UX2iQAMBGsshO 9O/41
- pRpZyBZ7aR5FmYlsPwmDjafc9q45Lks7P7i4KOHs1e/mrp+VtSC2trh7xGuRJbupwct2I+b 9c
- Va81WmjUQTTytEVKo2MkenpxT2uf3UipaXEqyzeai7ucLjP4U2W7hMSyR7djjfIy9ju4APbIrN
- uUndo6qdec58yX9fmStBsRirJbRmRfKEi5/dnjj8arCZ01Z4J5AYY45Izle55GfpV+WQTrLPIP
- LP 2gPEn9xc8qfXPUUySdFnmuPKA3K23f3OcbvoOlRGTe6IpVXKUkkn29fxZBbRrDaefI+CWCy
- N2zjl vwqq0l9bm0W3YyKvAkA4ZTyf06VoKIHtImZxLtUCQKcAMAePxNRyT3H2EC3lhEhG/wCZ
- M+WccqRV Rk29vvFTlUk0uX79v8xJC7adIyTRNG1xvUleSB3B9B/SqbanapKiBo3kZQymIbd3P
- P5mponme0Y3 AUxxNsUouAAwxt+vepDbW9vbwwuI47oKCrOuf3ajDfzqlyrSWvoOnXcFyyXXpt
- 8+pBI0bA3KrJvZ wxGeDngkewp8MtxgRPbuYSjMCV9DhR9apmWFbuI/62NcIqqcbtx6/lVp7+P
- dMlmrsvnjgtnJHce3 bFXKLta1zonUk0oxV/PoMLXSywuY2TeuenAwcfgKngjuGinN9BIio+2N
- AfuLnlT7niq13dXjXrtI 0UNurHcCnVgM4HpVjcGW3lmvI4oJ4iy7s/N83B/MUmmorbXsNwcIr
- mas+qv+HmTBJTEVQAhpFYBu u3JLH8KssYPmMUm6AYIyclgenNYJUhGmaZ3y2x0UkY+npVmG6g
- VwFxIuCSA3TBwv50pU29imnKK5 btfcarG1P2mQwyF5HDbUOChx0pjWcq2R+1SqpAzlRjcR1I9
- uaha5LzRAhQ7IUkwPvNnJx6cVfuby ONViWGV4JJyXLNnacDCfU1zvnTSRy1qzjaMdb767GRbx
- Q/apgJt7u+8c/eGPvD09KazfbZlTeYl8 plCk5LHr+dXfsokmmkVTbyqdqo3XCnP9arXVyrzyI
- 1nIyySeYRFhWBPHB+tbqXNLQ7It1GvZ6+b6 f16lcwgWmWtLnfHLkAPjI9BSPpzyXdribZCS7O
- kmSQR0zUUwmgWKSRZYyrKz7zwSM5P0Gal23J0u RzFOGFwoikJ4HcA+ua1vJK6ZzVoWio3Su9/
- 6uir58pZWR4y+zMhC9Sc4I/AGo4p5g0QHlvHMu9iV BMYA6H3q41tbTN5k0m2RH5CkjJzx+B5G
- KpfYIbczCOK4TZK6sXkznI6fritE6bViJqjFcqWvf+rD obeGOS28wxGQRHYCox17+vNVGsrae
- wEkry28u05+bjjq2PSlkKNPaW7tgFi0ZU4MaJ2Y9yTUn2uy 3RII5C85M0gZujA/dH4dRWq507
- q9/wDhzOnKaqLlb/r1GRRQK0kC3UTIVwHbkMSOMfU1ckV10YQS SxyCPB2qMMSMFufaqjTBY4p
- ljRAsLZ44LMeo9hxWjA159iX5I2lDqwJXIfAwWHtipqN6Nnqc81yu VrX0vpr62K9xcw3ssCsc
- Kys2A3RCe/qRjrVQEW+o/LJJPIjkZJJHsamXEksUVttMqxuoO3ONxzTE ZooTaPEJLoRlVOBk9
- 8/lWkUkrL7v1PSwtFtS5tF5vRee2w2V7YeW8cEgmBKyuD8qZ9vpSxNbtqEY 3v5aj52ZuN3t6C
- nW4lcxSgIZtwwNowQOScfSr9y0BaCVoC8fzNGUwMKByp9T70pSSfLv8x1HTpz9 lBuV+qav+N/
- zBNQiexfEItnk+b5wMcdMe3tV0XRigt3keCKWORY3kZeGBGc/hWdMI3t13jzYyF8k LxkD+H6m
- nXMSy6WgKEzISzx5+ZeR/KsHTg7K24pUKdVQppWTe99vX+rDrpJplGyaOUFvnSNcEZ42 fUjms
- hkMaqsSM4zt3A8Dnp+tW5wy+Z54YwiTflDgseOR7UyNZRfNLGVcIpZhjAYnoR7V00/diejF ul
- Tave3b/P8A4BYlldknR1aErMFUrwFQevv71nRXNs+2KUmMIpVZCffgH61HFPLcRFpSoKnYWHR8
- HP8A9anRR2huna5lAD5dgvBVjwBVqCimn+BwVOV+7q+unQjheOS9cM4ijLEvjjHsKgkEKNEY98
- i4 JQ7sHHTrVtbp44CIYoisbKFdkHfg59SaaGLCeG3VGjLHy3ZeNo+91HarTad7aHPGo/ae+rR
- Xn/wN y7PPZtYRKE2RFsBm5wx7f1pPsot0jlKSi3gBjdC2CHPfNZawBLRJBlkYI5IPC84596uX
- OotMfJiR zGjYVs9s/eY96z9m1ZRenUiOFlFpQlo3rpsvJ9yylkY7aKR3E2+MqNgxuGcZHtU3n
- pD5FqkckZfc FeVtwVD6/wAqoPOHW6PmsrGQIJM/KFPJIHbOK04UErTSxIQ8bmJd43bdw4z9Ky
- mnvIxqQl9uV332 X9fIdN8gaVFSMwusakjgA8Ekd8VNZ2TpMZDexM8YKcqSoAPRh3JzTYWsprF
- rf5p5mYeZsfHzAe9V x5hurmNw6AuquF4wxNZauLV7HZRalSdp2a30Wq+ZbmZWJaSF4XWRYpG7
- euP8+tVZ3sriGeTz2kTz QibDjaD2PqeKs3Jiud0ckv2cRNtLseGYnr+H9agnGLSTyZLeXzyp2
- Rrj5s4/PFFPS3QmnJRa55Ne m33hKtpHZXDW0qguWkTJJ3Dpmq8Nve/YUWM75HK7GxnpycfhVp
- ZJhmCX7OwhfayGPBA6Af1rOLzL CD5oL+WV3A4XceBj6irinZo5KkajpuCVne+qvdf16i3VosS
- tcorRF2O3zGyCuRzU4uIobwOLOS6W eRpMpjlV44qCG1dpopi5ePyismTxyMZHpUgh0+wsVi8q
- 6uFDEFll5QYyT9DVSatyvV/15nnYh6Kn K7b9V+v6lDZDPKfJPkyGMsfMORkN1+mKrTLCDapeI
- 8gt0JOzjfk8H3xUksTS+SLWTyT28w8kHrz6 CqUsc4d2kJkWBvJLjo2eRXXBJvf/ADOmDjWSg3
- a33r5/8OaD2zGOSSecLbhhtbJwc/1PWnGOFtPV IGEMbFgN7Z3E44/HFVAssis94ssUEkg3OCB
- 8w46VNeQEKypGxRgfNXuHUcfSptqk3/kQqicopVE2 nppp/wAOQxW6+ckUj7ZGQExr97jqPwFI
- sFzOVti8nl28ZAIb7wOfm+g4pZEjZU8y4ETOpd5Sp+Zi PlIxyAa0obbyoGhKyXC5O90ONvy52
- /U1U6lle4sRibtSk9Vt0t5roZptI0trEvLmJod7PnIJU9Px 6VDHJpqPIxt7pdzEqjScrkcVPH
- qTq9mPISTJUKm3O0E9D6k1LMbf+2Z5r0qY4nMaxxjBORyfw4p3 ne0r/JnC5TlL95e/ZP8AQyU
- KTwJFJgzhVjjPqeevvRBbpcrAju8PykrubhT3yPoM1cWOedA6w/ec EsP4V6Z+uf51VYFIgY1O
- RJuifPCqOx9Tmt+a90jSpJykowWvS61v+IxoHLPtljkJnTy2A6oQaka0 uo9NmkmkSWdZlVtg4
- 2sOB9asRpLLaSTDEfmOWjGMZUEcj2qpLBsvJIGuGiXcGdmBIJHOcUKbbtfY pTqS9+Mr8u+5ei
- aOK1jXymd1DpGncDOR/jVaRrdUidQ9wTGfM8tyASep6fh9a1khOx7hpoYopZPM Vip6D0rJ4tg
- xkuIZFOGARcZGc+lY02m20edRjOabir791fy/pox3tDFYRMjOA5PDnJHqfpTgYvLl NsWlIlGG
- A6Keg6dfWrtoxt4rm3WFw8rEuZedozk052aAXci2f2aPIHIHGR/n867HUd7Gnve7Buz/ AA9DF
- IfGzeHYEgMRwQPT9at28ai1t/OubeBZFyDs5AyRzjqanhMFvcpFIiyMFOeenrxioGZHuS3l +V
- FvK+W3JUVbk3oVWpyu43aa1utfu1f4kZuIJ9UtApKQRK0bNz8p5NXYRb3c9k0UkNyI90bQwxlP
- MYgnIGO1NWJonUqkQYHKRPGDuUckn1NXw2kO7y5KSGUBvKGxcHqwGOlZTmraJ/1/Xc8tvm6t/e
- 9f zX3lqxMSXFuSGtnW0KM0rZEhwcN/SnW1ktzoUMUM1usYGclPmPrz7VZSOyWSTdcpAJGKIXO
- d+e6+ wFNjkBeKOWNsRv5YCHGM8lfc9K4XNt3X9feYymlUVSnuv66mtFEP7NdlImj8xPqBu659
- +n4Vm39t eSSEJGJACSEjGGGTyCafEGTQBEsjHzJVZZB0XB4H51PcKyyN5sc5lkyzhHxt2nkfU
- iueN4zuenT5 FG8tf0/JL5DYI1huhb+WSHYlZTyMDJAPfmqqRI6CWaKQ/LtkC4ADNkgVHPbm2u
- Z1ijuCA4VAWyT6 H8M1AJY7h5LRWeBWJkG5snco4z7DBreMW9Uzr9i3PmjL+vk7k8MN1E+5raa
- QFgGjByZGXnePatSz uY7uVhLEYysnylu67Tmo7QSg4huo3llkVjkZ2gj5j9KZme0vYt0LOQpV
- FUD68+uKyqPnbXUmVR1p uKav0eq/MtTMnl2xtI43X7MWQsuT14Ge/erfmtHAySMiuyiRlUY4X
- uP60sc7+cTIq7mGI/lwpB6s PQU+SS2lEiSssksUioqrw3TOPzrlb6NHdFctoy1a3t+pXnXy7d
- ZYZZDJIhZzvJA56D05q5BbI1zc yvBMTk5Ct8uR1OPYVJFLJCt280tsQs6CVdnfGRj0Ge1W1uZ
- Jb4s7xDefMZFXGeOR+P8ASsp1JJWR NfGTpwdr8vfuQMi7j9n3tFMyyYB544yPY0k1pIb6eRm8
- qMy+ZEr84UDG0/jVea5k3oxeNV3CNFC4 JUjJx9e3pVeTU3kDxxwTBWyYtzZwgIHP40owqPY5v
- bOony9tf6/4Bf8As7BlAilZF5kCtjaw5AP8 6RlRJDO0MyQMgLOW4YscAj2FZllcLI8qxJcsRO
- cAyZJHp+eav+YRFbpOSIiCzZP3eeAaJQlF2Zk5 Si/eV/K5Dexf6KEVZGdZU37WxuK8n9KgnEi
- yXZF7AirMwjdxnCsNxXNX7m4Riw8+J2dGdNq4/wA5 7VmTeW2kh3+duQFB54xyfp0NaUr2VzWj
- OUeW63f9biqYmhmWJC8wKsIgeVU8n8utXBLaN9q8xGdW uFVWi4zkZH51jLOwvDIhUF4+FHVxj
- HFXvtVstqEit5TKkmWG7+7jH861qQd9jpxNSTmko3Xr+ZLc aj5spgSzmkYksy5B2k+tMedo1i
- geEhdg8hXGflHf6ZqQPpxvXOJSwASRlbG0k5Gfesl55odTEk2W i6ZI+8RnAX05xxRTgmrJWNI
- 4mM4unGNrd3+Wp0Fiv/LScxqRGcIf4euc+561nPArmNtwaOBEibZw ck5IPue1MkdjF/xMNyKy
- AyFDtKP3B/SoN1gyXCxXW1XuVl5Y8gUQhJNv8loRh6tVNrX5amwypp8i ThXZfmBRjktzgMD+N
- ObUo7C3ijSB3ZGbZv5ySMgnPXFWDIJtUuwAGi83ace68Eeg71Nb2CyRKuEm WKMRqCMk5GS3Pp
- XJzQ3qFUVRc/36bt+KKEeqXJsre5up0YL8pUJg4J+YH3IqWI2ss00q3MSpx5eT zuB+VfqalWw
- tG+zrIS4dQyYPHTmpZZbVIJUhiVptyeVtwAV6v+PvTcoXtFHVGpS5uSMdX91uhXFw El/0oK4M
- RMiY5BPBP0Hemxq9vp3lR3KTRcSEHJJYdSD9KsyXCPfzeWqMjLI6naOoxj8O1ZxglujI OI5tu
- fK6EY6n6dqcVffQqph41FdaFYPM175gMcls08ZVlXqOauNEsl0yAPJEZW3MpPyDqFb1YnvT BJ
- bCSRUuYflJZVA7AcEVkOb6K4MqeaFypUn7uMcg+/PWt4wc3poP6pUqScnaL/P5D5LKOXUmhhkE
- Nw0e5/M52Duv1NPMv2ey88rFPAZVYqqjcMcdfQmtHNw/mPbIiMqMAzjJxjr+HNZHl3Bs9skkdw
- fl 3+WuNoPQfX/CtIy5/iZz1E6sl7X4eu3520+8ry3P2l5GlUhBnMSjBBP9KtrIqJaNb3IjwCw
- 3nOV4 yv1p52R3bCe1ChG8tJccMpB5PrzSTWMccixGZXk8psKowRgZ5rTmg7LY7F9VkuSTcbbb
- Nfht8yRd qX9zcrnyFzhwehPO0+9RNdxvBJMtu0U42tuf64qujy3Uc7rcwNvwWjCkE8ckfQVfi
- ljs4YpYl/d5 KAuN4AznB96TVt1dm7lF2Vm326MrXAnlkeaOMZXcJ0Xgp6D8qglWSVLaNZBhY2
- 3nJ5yM8fhUzwwr M0yNLLFcO2Nr8AHGAfU1TiufsrlbYbnUmMhuev3h9T0FaQu1obQxcpxUVun
- 8/Q05lgXTwWWQHyyY WzkEcc/jTIr+CyuXuEPm5UKQ/O7I6j8arm4laSziWIYjhZTG/OFzzn1x
- 1qGCJFspo5IWaNJE78vj OSD6dKSguW0v61D21qLhNd9L7/16jzPKb1ASPO8hlbPQHnIx9KbFJ
- Hc3QkkZhEYdpCnHIXgfiaqO rQ6hHNKd0bkPkHG49OPxNRRlrd5zEDGsUiqyvzlhmt+RNaDliK
- krwpu2nf8AUfLHJHbwowaJEALM e7dcVYN1HHHDJbRL5w8xZFkXOAemf1qBWeRvOdWl2xjcM9e
- aqossT9dyiUZcjjHb88/pV8l9zlq0 Vo6i08tn67kqvcMkk7cmOSPbtX5cDPBHrV2N3RXkI/co
- 5jRRwSXOc0C5EcojZQgRiJkK8sazraQS 2rT3mIyS2yLOC2P/ANdTbmTujKTppPmVr7bu/kX3d
- 50SN7aUhJQFKHAPv9KYDJYXjlkATftcMOrD nH41H5qoYnimJ2ybio/iO3j8B3p0SyXhWSd+do
- BA6+5oSstdjKlC1+ePuv1/zGRvs0+feNs7SAMj fw47/rWhDewwh5JEkNw7Kw2tgHjb09ealDW
- cLw7rmDAUq4aMkliMAmqBjS6vYVKmGQJjeDxxxkis 7qd7rQ6aNSNSnJSg+VdX289vwNK0t7dc
- TwS/vYZChJOQoGQSfU81YbUbq2kh2wLIuQGbb99v/rUk hjjsWihhfIYq7g8Mxxk0kNw7xeVPt
- RmJbzCML8pwcfhXM/e1auZ0o05q8ld/p0sTeUhuGcoFjKEs pOTnr+dVmCMYQ0qJC2HZFyD7kH
- tg1PE3laxczFgI1Yj5uQuR0I96alz81rmDcEjO7HYvxj8KWp01 IuS5Vt3Wn5kk+oRQ3IItjIj
- 4dmGMt2Bz7U2OG1ighFy6hZk3Akkb8N2qEQfZvs8T289wAGXKng84 P86tF4L23isjZ3MXlkx7
- nYE5Az/SpfKkrbdWcFSrSi1ON0utmR3L+XbpNDl7eQCVwvXcpwAD2B4o cmbTkYbWYxu0qY53b
- h/L0qhPJcWI2grycmNhk5bByPbtWebkkho5gJG3BlAI2ZOOfwzW0KDaTX3k zwqlZv79QZ8osi
- RyBpCDEM9s81auppUgYxRYRZf3pdcjfVqeSzW4iiimjZEcbWB6gcgj8KzZ7mIt GQHSKeB5CXO
- QWz8v8q1j7zXumFPkqVUqkNPO4Ti7lKO21YmRn2sOqnrj8amsbwR20Fv5TSyqNif7 QOfm/A1F
- DJa29pDvZpLlvlJLZCBhzkVJO6GKJbcxstsuNyjBbn19x/KqkrrltoTVXtkqcaeib6NW /wA/v
- LEEVlsnQt9qKg+YU6A4+XHoBUCyz20dvcWkgLtBmVWGd56BvoO9NLz+VKwCW5UfL8oyytzu /D
- +tN2Ty6MS0L+bGxRCOAA3LA+tTy93oZum4uKqStF6NN/oE0rCaK4mkhOSGQImAyqOv4npTLgCe
- yile5gVZmLFtuMMM4/Oo4rO5PlxFPMETbAmMkgckj2GasLbWEsmFkwypnlsq4PO4e1W3GLVnt2
- MH JKakparsk9CqqqVhRfMk2xlHkjbAZ2OePYdKt2/2lbG5S7twUkuIxKdg/dE8fh24qVI7fa5
- QgtuU RgHG9eu8ewrOEsyWsmZBmdt2WzjeDz+lK/Pp/XcxqVvaOyaXrv8AInS8SC9uFcgxQ5hZ
- Qv3u2R6c 1Sku1ujH9oXbtXACjBPH9akltbZ7V5beR2LSMwQnJYAjLfQc5q/PPbwaXNPC1vPKz
- EgBfugkYH5Z NXeKaaWr0CdSnzx5U3J6dV+X5mUlxPLLJC6OsciHykPO09BWfJt+1qJIZNsJCH
- J6EdjWje3UjmRB GkAJOH24Z+nzA+gH86zfKcxOqiRzJJ9oXJyQRwQfWuqnte1jatKVrRjZev6
- 6E1zPFJLKodnmLk7l PHQZGPoMVG8TSwW1wkihPLYgE54B4J+lQuGZFkmVVMkZIAXBJ6A0WUcs
- V9bTsDFD5bsA4yGHTp71 ajaOj2OWpdQXJq107+X9IuxJNcaTl2ijkO5i5TllB4YH0xWbM8TR2
- qqNrhHyeMsO2ferZmJt43Id Sp/dgHA2n7yn1qi0QjhEKrnOGB7r360U1rqcmGoVXa8LO/f/AC
- /Uv2awCVVd3jgCtvZm7kcfhnHF XEjtn+z+VtOUK3Hy8I56D8ax7ZjHOrPIi7RjDISMHnmnzTM
- 1wHjDRMwO9ccEf4980TpuUtGddaDn UbgmrLorK/mdPcxRRwx/aoXJiUoCDjA43fiOOadKtt5M
- rBJd7tkYk7ADmksr6O5tEieeKQRRFSGG S5PJbntxVmLUIrS0vH/dyTb0c5UEcDoPrmvMfOna2
- v8AwTx41qqlyRT5+urtb7i7ZRBoTBHJH5DH JJ5JIyVI9OaqzXM8Ytp2/eTFFYvj5Wf0/KpZYo
- 2DkrKBHKRuR9oHy5FQ3E1obaPaxSTH7lGbJCsO p+hrGGsr73PTw9V2va8PMoz3krS2txcP5i7
- d4CDBIU4B/XP4VCbi5M8SW8SlpJciXaMD2/GmTwSS zW6Rt5swty0hTowB4IHYVEzs0aedbywj
- a/mnOAGP3f0x+dd0YRstP69DrnQpcqlbX06ely8zSxpN HvSFtwk9Dz2/PFPmWRoEnd3V0VhKC
- f48gAD061HbXGm3NpFLPHKobAIL9wMY/SrP2GUWgkhJDuQz Rtyc5/p1NZuSi0nodeCr0YOKWi
- v1W50Nus1vbODA11MisSq87Rgcc+9QwlNyugCXMh6MM7Tjv71U Nw9oywz3CSCSNpHdSeo+7+B
- pYHk/s4eXcwkD5ZAyZZXboufXFcDpuzfcxSUbycd3uXdQMk1l5bTQ KLn98zBfvbR1Ht2rOhju
- 0iuD5ckvluoGDgrkdD+dThmCeUVwyp5aI3PH94ew709lkguYYTKoGxvL bJwyqO/qT604Nxjyl
- 08RKlT5W+byaLVjcLbzSSXTRSKVGeOrEEDHpUXl2Uott0U4ZIwku2TGw5OQ femRW8XlBp2zGw
- LKAcFSPu5/Gqf2bfq7QC+jhUBRITnlz0/PrUqMXJtOxPNGo5VFLla9TV27Z1+z BYAswXzDznj
- Az+NQzQ3rTJJIoe3izE8YGGDnkZNVbotBLJgmZNjeYUJ4fjj2PerdhPDPFdNNcFC8 qyKpPTav
- Q+9S4uMedanPWo1Ir2kHzX8l/SMi1t5Li+DyLLbyL0VjwVHXH41uzWhN/CyRPcRFZJ5V jOD8o
- wBTZILe9szNb3KxyOyquSf4u341Pb2VzFZyW7u4aGb7OTz8yscmlVrX1vbyMcZjYS5W5a9t Uz
- F+0FDE4WMkR7sFB064/HrVdobZ7mI7pZ5XV5MRPgEAdfp/hWpcGOKeWNF3M43lcdQBjj2FZuLh
- oYdpiESIYy4XlApyQT710QldXWhvGrSUk3dP1HxG6hhRp7WRoTGA7AAfMRgVnSDybmeKeQpsmS
- Mu /IC4yfx96lee4k+0JCWfcN+M8YBB3fTFPghSXzXWCVw0scoLNnCdwfUit0uW7Z0z5qadS92
- +3/BG pDmwkEjSkDI5fJyM5H16Vo2dhb/Y4Y1w0uV3g8k4PI/rVaWd7Z32Wsgh+bypG5DqeM/n
- SW4vJJjL cKyJFJg7fl6L/U4rOfO43vYiNOs6PNL5K+/3G/dxNv2BXSGN9pdTjdznr9Knhv7Y3
- ds1sszYR9yh s9fX6VjW85HkQsxtmmPztM24MR1I9B2qxaEfbZUMfmKswMZj4MYUHIb17Vxype
- 60+h2U4XptSVmv UvyO6x24jjfYFU7s5zt649OtOnkd4dtmI2lLMSNuSg6AH8M00XLl5ljCvk+
- Zux8qEL938etTw3Jt rSBvI/1sS4bAwM9B9cms2mrO2p3fVpwjGX9fiPjjKw4SSKQiRQ6qPmOO
- hHoPX1prQqNQmWRiQjFW wTl2Azke3tSsbpkeKRo41jOHbZjcVHGPqTTRJcLA53xRzrMmSyZ2M
- F6H61HvdyalOo4tp7/12KVy Y/It7lYGjMcfzPxtUnnBH0pFW8aaWVGj/e7QkTx5wRUF/DP9g3
- PFI0CyjcFONoAztPqeetTXFoJk S4S88mMxOQCTwcDFdC5VFa7/AD8ypezhRj3lp3T8inLczoS
- Xy7OGbbHx8o+Xj+dQLGsttFBPcoeM fJ8pJzkE+2KIC8Ee62kSeLaMoRlgSM4z+pqjDLGl1bTP
- l41/czFegZs4b6V1Rho7f1/kS6cVCXdf P81oLdFo98Ak3bcgHPXBB/OrgmhuJmW4SUOZWdSGx
- yB0qwkMSynZLG8aHyWB5Oeuc+prDuEhWJwj mYqxbYMkgY5qoWnoZU3DESjTbtbW+paiZreTyx
- NDArK6KrjLEsM4qKWa5jtYw4XYgQoGQENx15rL WUhxKsifu5AFB53ZGMitOK5vIxMTGWljyrn
- aGVScYGDxmt5U3F30O2M6lKfI1Fp/L+vuZHIlxHcR hI2KtExRh0IHUioZpQsqExsVVAuBgdR1
- zjrTXuFGnwkCQSOcyEtxjPCj0zzVqXc4kLRbEjZs5H3W IGAcVabT1R2UMZKcveVr6X/ytr+Qv
- n2xsgYmKygqMk9hwAf1qQRCOwhmklJaKMhUyeRn/wCvUEdr KY4o0VCZAJWGMlWUdKRz5qONks
- Yd1yWOcORwKhpXsmcHPSc+Xm0W76/jqWYJjLHFK0BMUb71J5AP Py/U1Uu45VltXT5nZP3qehP
- c/hUiRTiyaFj8kL4nPq+Dge2OhqB0dfJ8+YSgfM0aEhvzxRFJSujl p0UqjcVa/wCP6fiiwFVL
- YxRxy5kLBeeWAHQVBGbSWxzJMd0qAgAk7McYPvTIm8qSKRSTsOWU87jn HH509ImjupgqIfL+U
- 7kyoq+Vrr/X9WOmGFxCfLKVktVrb8NSoGJWXzJkaQtuYhetWoWEljLCQrYu IykgHA4xj6Vcnj
- ki0yA+XBL5n3HRABg8HNN8m2huHQLIhjkMe4njHbPvSdVSj/XQqU/b0kkrtP1M uS1ZVlXEhli
- nCDHQAg8H3zUsQnUpsDCKC42tnrz1ye+KvzyXeUjJjXOOduN3v9aW2ZVeYQRuXMgY h+e/X86b
- qScdQnQbp2aTt/X9aEb2Rd7jkMBIN6jqrdqV4DLIdriWRWCtsGMk9x7U6BGgJjYtI0r+ YVB5K
- pkn9asbA5edQxEqgqqcEZ7/AIHis+dp7nXQq1Ixeune36Ec7XEQBlw5DYyo4JPIFaTWsAgM Zm
- VbqM7NhPJ3DLfnUsziw8gGHz2l5U9QmOMtnrzU9tZ3Mm9J4wud0nmbeQ46ZNck6uiexyzrQnOL
- ulbrt+n6mQk0sMokiZVwArRSDcSADg/hWskk8mlQIhglEqpIzKnRh0/D2qF7RxpxWcq028AMox
- we tTsst5A0EEiQlTuChMEAf0pVJRlZm1b2dSmmrNp6t9hsc9zcJcpHG0VxLIZg7j5QF6gemao
- TzTbk dHQqUICAYZQ3TJ7nrzT2jleAmebKOSyyJkBVLYIqN2CagySMq+TmJAOwcZyfXFOMUnoj
- jknLWCSS +f4/oZ8sCzBI4bgONrGPk5wOAT+tE6LFpM0C/ZxiY5fbyTxjn3qIK0M5kZTNFEApM
- Zxu545qxIft dzcNJG0UDNuZ+yt2Fdeqa10RpiaUozTTuo6303IY4YhcCIqXVSGDA/cPYH15p0
- oiVDayQFpI/lIX ggk5yPYGojIz237wEuU3yBePnzwP61CxhliLrP5cvmiNg5O47uapRbd2ee8
- N7/tJpq+ztpf1LbGB pZXtmiaU4GSPlbcOw/OqN0JY4m8sAGErGGA4YEdfc1bispk1EpEyqY3G
- 4Hkgj/6xNW5raGSVFjDA tvkV2bI2g4GfpS9pGMlrcdavGlU5ea/9f13IY5kMc8e8IyS4DtyBH
- jkfhTbeGK2u5J5DJLGG2RkE AAY7+p5qKFZVlGCnyk4BTO5e5P0qzJdTRG3gkltvsnmEZ2cgD5
- tpPqexqZJp2XUwxkZUpcri0nvq 9V9w0MtsZZDHL5ynDNu+VvcD6cU+a3Atrie1QyASBFA6gEA
- Y/OqcV3cG/ZgY2EiM7AjPQYJ+lV7p ZFslkklI3IBEF43L3B9SPWmqcuZXOapRnCqoOyf33X9e
- ZetpmZreSCSJpLfKTfLw574/lVAGS81K NEkigLtnLLwuQcr9apCO5tUSNZY8PG/Qcrzk596fH
- O8srJcRmXch2tEdpBxgGt1Stdo54YdTUpJa 9+33ktyluJvs9rDceaR8vz/6vjODUcM0DReakq
- ZdG8zcMjJOBgfhU0V5J5CM0Qh2KB5hTODjGD9R StEdsUUJguABtxGmDnr+lUnZWZU4y05nZd7
- /AK7fIrQ37TWbys8S7XAQOmcAdPzqQXsf2gPMjElN qhRjHOTT0tjDpyy+Syt5Q37gMZLHBxUw
- WVkiWWS1kYjcieWOGJ4B4+tKThrZE1I04Qumtd3s/wAm ZMitcz+bK+yN32owGFB6gVfXT34mb
- f5DgKxz39vQCqIguXuJoIY5M+fu2f3MZqxcs5smQrITGoG7 OF59q0k27KLF7Kc1pP7t16bGZc
- 281oIVZ1Eb5PzdSAeo9Kz5vP8APiCzDYxz908itWSWJkJjjYbc qpc7hg9OKryPbkp8mOBjcet
- ddOTtqjVUZypPmla3qm/u0Ic/daaWMfLg4+Xn8qsecUubeRURI49y OzDPJ4wT61Uz8yywlJDv
- 3AFM7fz4p2x5JGW2UyLyxBG7k03FdTln78bP4e7b/LRFyygViiksksW5 ZADjIPf6VrWkcUdsv
- 2dXnDyLzuz8uDk1hRrlkEciuTEc7euAauG9JtraBQUMULKhHG7JyDXPWhKT 0Zy18NVTUYz372
- aS9f0Okvb8ShZYwwt3jdSgONpJAGT+FZqSXFvpUYyhdZCnlFMuvfrW2tvBYxXJ tHiuplIG3G7
- Izzwf0+lZ17Z6ozFYIWigEnzlwCdgHAz65zz3yK4aUoaRWi8z0VUUFBW0T30/9JbT X6g89wuP
- N8os7mPcgxlAM7h7ZxVUS3dwpWWaEM7jgrgMvdxx0GKuMto2lI9zMFeSVEgXPOGGD+oq hIjW9
- 0UM8c/2VigZFxu7H8O1a07PZam+GqXbVN3f5GjbwQ+TJCrLKi3AKhRzjqD9KszmG2niks7w M8
- sLt5ZYnbz/APrrMi2NdNG+6ORX2pg4Geuw+56VcnuFuDbR21vuuWDArs5iGeB+hzWcovnXY7Kt
- KFOpfmunvsvvNCLZPcSyOvneXMAhX+IEcfgTmlubaGSLq/zuWkVf4cevvjNUivlRvcNdRK6N+7
- AB AKk8Ej37VIZbq1DzR3Nu00b4ZGTqOvT8aw5XzJxZzyo1PaqdKXpuv0tbzLcE8EEPmxhwJji
- ISHJ2 g4BHtzzTL1Un1m2tzcCEWiMskhPDnqCPY9Kkljm2LKkW+6dS8S8YjUEZXHqc0/zCby5k
- AjaWMhXU r1BX730xWStfmX9dBPlk25tv87lszl4LhIFBkllVgP8AnkOmD/Oo7uSEpKkmyNXBL
- SgdCuMD1ye1 UzvTyt1tMlqUKRgH5gGH3ie/NUY7Bmw0kwaKBfL9iSpP86UKUU7tmlKlCKvKVu
- 3/AA//AADW85vs 8z+bGYpSJRhfu4OOar3d08ojMBjhaUkPlOFJ4x9e9Ogjhh06CCRZVmFoVkD
- NnDZBNOWMtesw2g4y UYZ2E+v4c0LlUrmMqkVPmta3f/gEVtGE06NUc/u2Csp6gjP8qum9SaOS
- C5uhE0k24tkjhOv59qyZ UjsGWWBjcIxyCG/1ijgEfUmpNqvKJIruC2nj3mVZoycgdcVcqak+Y
- HRdT3ld+aW35/eWrlJpXijL iPzVYrn7ygclCfXHNJFbQQu3mSsodWK7m4LHgVfhmknuojFGJM
- li6gcxkjhT796o3W9o2Vp0hWMh k3DlwB1X2BrKMpP3djkoOpKbgtGv62K0YCtEl4PIieMhpOg
- yBjFK0VmNkTLcNGcIHSTAJPAHNILZ rmSCOdxJvy8pBxtccZ+mO1RvBPbX8sjgLFH94kZCuFwP
- 0Nbqze+p1UKbvLln7/ZaL/hyxL5ccYtX WWIJGTukbIJGCcfyp9rPO939olkimjILEovyjPXj2
- plrEshJguorm7IyAVJGPTHvVwZS3EjQG1yX K7+ijgbTjuetZyaSsdUq1OLtvff+tySC3tFjCM
- j3LybfKZW6gnoPerQtHmlkgkjdZoW8tFRsF16u frWQikPNAqvE6vj5jnGBnj8OafczTrPb3cj
- O7NvwEYgnoP5Vm4Sb0ZvBSfvRevTV/wCZs29l5MpP zLAgKqrHkhucn6CkmilN1bQ2ys0UduVi
- Oc5H3ifwNU7DUJSXV1Mwb5jjHY8L9a2A0ovY2SBgEVlJ z98YySPQZrnn7SEveOx1a1J3mtfl/
- mNuVNxYeZbJJulO7G7O4NgEj6Yqo8Ust3H9qguI7cRNI8e/ DOBwDmlS+uYdLEwaKN2dXTcmcq
- ODj6VbW5AneUHzWU5dycqvbb+INSlOGyFGUlD4bp7WetzMN3FK sEcUm1JVOfMJPGOPxNJHNHc
- yxqXUxrCElAOMEngfWopVWV0W4iMboR5Kr8pZAPvcVYktolt7yWOR I4TIGRz3xyP61u+VKw5V
- ZWtG93t5P/MhRre0nQjDsy/6tf4SPlAPuRUKRpDFcpAqRtFIqfvRu3EZ OfwHWh9NWW1uZrecP
- 5kgcnJ4HWssSQNdtBK0qROCyMW+97/jWsIKV2n6nHGFKo5VFK7XxafpsUUv pGs5JGcC5kcYZe
- AVPXj14rUitdqM/mRxJLnmQZLkHhh7Y7Uu6C31GZTbiZRc4PyjCfJkVTleS6nh VMwRbBwx+6D
- wR9TXS3zbaIbrOV+TRLdsjSzaSaSeW3biQMGXAQjvx/KrE87S6dEfs8sMTFpJSSM7 iwGPypFl
- maSK1jdYcKyv5vIT0B96WELKsHmXMUrOmDGoII3HFOTd7y6ETi3UVSotvX7+xUdpZL2F IEUxl
- cxIV5I3fz61OL2VtXcw2T3cTlkwBkEDjceO3rUMsEaahJGjOiRnYCT8yn0z+tMd5I42cNtw 4D
- ovBYkcEe3qK15YyWx6FWDs+RaND4vtJuQiCQIMMrHuq54/Gp5orqG3juoJUAcZdCuTkHkfWodz
- JPcfvlgZJFjw3O44Iz9Kv2yOtlDCY2mWWPdJIDwCOorOpKzTOTERcZXsrPfT/MymuViuZVkYyI
- z8 Ecb8HipWR5Bb7Co25jlyOV9zVyCCW6aZSmGBMiqFGSNuOP51BGlytr5SqMBgzHb2HU/QVXO
- ntudq qqok4u0l+C/ryAx7LMQrAdpypLY3bieBmoY2YWkqzEhWkAYd9wBz+XGauh45J2ltpDkl
- 2R2OVx0H Xv15qSG12WNzHMVhjdlcO4zg45pc9lqddBT7K34/qVEV8W8brIVBJky3G329Oasww
- 2812HQS3GDn APXA60sz25gikihnQsuBKzZVselWYIGlikiWMo0S9hg/T86icvdvsaVKS5HKL3
- 03/wAhslsyRmUR m5BT5lQ4OQM7h7DvVIIhh3NbTtuOZJEbhj3IyOBjmtFmaEETRTsGbfuVsAt
- 3H09qfGfPs3RsIC67 UA5Xjqfas1OUVdmK9rCKlLXzTf6fqZkUFvI7gz7o43PlFeGcMOfw4q3Z
- oWnt4XUxmKM+axPHXJ+n FPMa3FiVSFllSZV2rwWXHzEfhUhZEhuBDBOoklDKzHOAByKJTbViY
- zi7qKavtr177/oTvObi+VEl RLcqGy65OCRgZpv2vyZpEa5E/mb3UpkAc9KLiPmCdXSFnIXbtx
- kfwgU42luJXuJoJYn+YtGWGVyM fkOtYr2elxxhh3JPl07abjVYzyxfaJEETKXJ5HzdAtLcHbe
- wQhmQxHbchTguSMbh6AdxVmyjlJgW 4h8hYY1QxOBuBz1P6fnVRpn826LwmJxMqNMw+VMclT78
- Clf3rLoVU5Ze4nttbb7u4q2NlLGlvDfK h6eYWJUY6DHvzUN1bWSokkxlkUEttV8Eeise571Pc
- rHBcvcRW0hlY78Z4XHt7E81WhuZbu6tY5EW NTC20MnXPGT+NOLn8SlocVVT91yqPkW+quvuK0
- UUCTypGssNoVBkeVt2GB/+vTyZp3jtYwjxPC33 V5HOMH3pkEsFjqEQ8uSXEZ81G5AwOevvzU1
- 0JglognjzKdoVBhpCOTj0raV+YjGWhJO9n0b/AKuZ 8xdYYrcW4WURmQM4zkKORVBIzMUEiLCu
- C4YjAz1BOO3pWg9vLLIPNV0hlO9pv4VYZGPYH0qBGW2v LVJ0aR5IsKgONuD3rphKy03MvbRjS
- vDV+WuvqwB/cRIJDJKG271ON2eefU5oljaV1Y3KhiMbEz8m e39amLSS6pIYo1ZncSAIvAIGMj
- 0FRnfJcLJco0ZAALAYx7/n+lJXuFTC1oys997X/T/IWdbaPz1+ 0JI4l2qUGNyNgGoYXtIMiWC
- YiDzFILctngP9K0YkVoYPMntZHRWErCP7xJyo/OlTV1ghJNuhnLDc rqDksKjnk1ypN/OxwV1V
- qacrfzsjJQ3LXVokCKxjQtgLzIAOQKgvW+06dZtEksYVirFmyE3dAfpj NakkjvcwLKVDquNqD
- aQSf61TffFKiRSxy/eZTjgY45Hrk1rCXvJ21/4c4K6caqaVmvN/1+BDaWTS WxWY+bcSSfumj4
- GwZ6+5psZ+zNbmZ40Z49yx4+bgkAZojkuZ4ijzJBMWHloVwQoyG6VWa3lazjn8 xZltXEG9Rwy
- sTz+taattSZhUqzc+Wc21/Xf9BywRKSJ7gOi4xtyeD1H15qSeSYOBFIkkcb7Q6Jgj Pc8d6Rba
- eC+jVrWWQRoUlUrnLHOMehqKCQxXNtFImxRGY2OMZOc5Nab67nouVNrmT5rdNPyV2W7i ytGtg
- ftMqvHPtCsxOF4yCKmF+2y4RYVdTOroQMEKO1UpriRrqZjGrGRwTxwB3pLaYNbSjHlnzv3c WP
- mZe/NR7NuPvaieHpODcru/fb+vVjb5ZbnU2cs0R3MdwHVsdOKhVjPGipbmUScKI8ZXnkH3qW4a
- G5uVaAOoJwFLZPTn8qqzwvGINknlhv3jZ/jA7j0raGyWwVHVpRcYaXWyX/BKsoKS+XuULvG4YP
- AP +FOAE8YBxcgPtOxSBzyB068VJMrOHmkXyGWQY3dAOwPHWq7gM0qYUEys7benTgY9ufzrdO6
- Odyqu XIne+9/6/UPs4s5BF5ckYYFhvJz79qQW+IzJ83UEDJ47c8VVmtZ4ju84DrkMOxqKS6m8
- mOMqclvL Pzcgk5B+gPFaqLezIdRUItVo25dkl/w7/EvqyK7heCyHAA56dar4e41C1/eoFHUnj
- juKaYlmBcRS xmI4f5uhz0/StBPOuRMkcaRqz7x8oyOMY/Gpk+XUyxLU1dpcvc6kzxrcWz28qG
- WJZEkfqGJHynH4 /pWpG3mQQmW6EkcahNqkjecjJ/PiuY2W4uGihbfIW2qT0bvuFT2nnz+Qt4r
- CSQnyyoADoT1x6A/z ryKlBWTv/maYjDw5UnLlXV2V/wCvQtX0ofVJsGNIWuNy5TOwKBwPxpIC
- ks0kk08JBIDDbyS3A/Gp r0RItvard2rBo3D5HKsPu546mn2N1HC9tbmENcrEfOUoPlK85+uKV
- 7U7pf1/SLUlCgqkL3Xy077C L+5nYSmJDFIowV5XPBz6n3q9b6hdNPvhhhwFZCgiG4N0GT/nrW
- ebiOeG8OzALh4mYZ3KOT9ajjjm nlEgWZ0lQsrJwCAaiVNNXkdE6VGS9pLd97W/QsvA6zSNemN
- FBAcFfusF4Wli1GOE5V4cnDMXjz8x /pTY1trK9bzZzL5kgePechsjgn15p1u/2zZC9j5mMNcB
- QFYvgjAPbpQ0mveV19wOlTUX7SDkvu/B /wCZcN/LdWpllChlZj5qDaoY8Y/Gqs63U7Ov2qKXy
- yqOI1wZCVOMfyqoEvnWPMTwJKPkUrwOv54F OtnYxRhI3JEZeOQHgqp5z6ketCpqCvG39f10N6
- UY0oJwS/P/AIP3FnLpZKAJYW2qxaUkgMODx6DP SpltktxKnm+ZEHMiyAnB2jkn9aSPUHlumEf
- leVJtEe9c4DHkn8RUNzFeDTbuJmQwGThtv3s9MH0P TFRZ3s9CasnKqoysk/P8rotR7Lqzd47h
- fKMgbJzlsc4FURdXdxJE88iZnfd8q4JwfmH4DFEV9Hbx sksDJFISzBRjYRxs+uK1dOv4nRYka
- 2ljhZ0jfYPlj6n8T60pqUE3y3OLEOrBqMY389zKZInupJYy VMPESkkgj1+nNallZLdLIPMh/d
- p5bqF+c992fyqnLZ6emnM0UzbkXbneSORkZ/lTbCRlihk8wTxl R5qx8NjJBGfX/CnNuULxex6
- 1So/YKdK6e2qSZoqJba5w8ciuxDGXOFzjG3681Qe7WS4SKWaKOJVK NuXkNnp+lPu7dEglWP7R
- vJPll3z1I/pTIo7e5nlWGEzskhKqG5IwRj6ipgo25mYwownLmW738vm2 aMtwlp5dzgOTHIrr/
- tY4Htwc1Qs3uprMBFEssMibwwyGz6+pq4bT5o8SBJRbsWR8nb8vU1ammnfT YJY7yxhcKGm2xd
- 8ZA+pHSsVKKVluzCVKlCKT3b31Ms6beJNHIJFbehACLtKjP3T781Q8ue3lQPHN GIj5e2Rid5L
- dB71ckJuILSRROxMY2RqxBPzcH8O9Thpv7VLjExkLOEIyRt+tdCqSS11/A7KVVRjq 1Jeegrah
- N50kKQq0pbc6EfN6EfXHFRx3DW4L+TIHiPlqJDkAE5APvRNGkNvFeOkiIQZAucMB0AJ/ HNTmz
- tDYsHn865Rw7Krnop/ng1F6aW2hs8RF2apvle9no/XU1ofJuMXJkjlmB2qkI28Nycj2NN8m RJ
- WSK5MMucCRzlZPl5I9BzVW3kl/tOTZGV86f5owozyuMD0zVS5Bkt4AEmSSJQqqzffUHlvpXNGD
- 5rX0M6FV+35ItJPvqaVw8U+mSKsyRy2x8sow6hwMfljmoI2nijmb7VbCF5VkPyH5x0Zh7CpI1u
- Jw 8pjSQSZlwigEsBgc/WmzRXEMZtoVzeTkO6kZEQx84I/XiiNl7t/66h7ifI3Zd9H+mn3FZ5n
- GpNC/ +kIFPlugx5g6DHtiqK31vJbrEokMe0ADdnBzjB/CrSytYqktxGZwFAjdejL04/MVntFb
- QxGU200Z gmREy/Xkkg+p5rqhGL6F1ZRnPSGnR3/4KJ4rk+cyRMzbnBKDuR0I/wBnHaqVxeYvJ
- HBiBEWxMp1B HQVduordWtNgaVXDiF4zjCDnJ9frSTiOWZllkgtodmI2dckflVwcL3saUKlO7l
- KNvVf5bjjLJb6T KH2lFdvkK/NIMAZz7ZqlHFFdyRpAkoQfu1lJyHHXPTtipoTdHSrcqYrsvgK
- iryAD0/H+lU2FtdSQ +WJIZoC3lENhQT1B96qEbX/r8Dmw0qii40klq7vXRen/AACYrE5Bjt5R
- MHJiO774IyAfU9a2LZok jiR4o4lki3s20AoRnCk+tYAsnRbeFJGd3XzFKsf3Z6EGrIW9e+4tp
- 1Cy+UUPVdwxz/OirBSVuYur Dmi1KasvW/4kFmYra+zG/knG5GuPmD9Se34VYBSXWFmkXCMuXI
- 6AgcjHt0+tWf7PlWGC2lnhIC7Q 23liT8pB9KtRW26/S2urZxIwKySKSFLA9MY4z9aU60NWbwr
- 05U5N7Ja2aTt313/EzpoxPZwmFN7F clRy2eufwFQW0a3EEUbu8XHOW4CA8j6mrMkkdrqfnxQS
- iFEaNI93UnufpUaRJcqJJpoo23gOo4LZ GCw9qpSfL5BOo5U7JtR6SV2/nYdeWhglaSBmSJ5So
- Jb7ikc5pli91sESQSyoyjaMdh0/OnvIsUC2 tvMLhUQlJApwB1wcjk1es4C9tBK0U+1kBba2Aw
- znI9PSlKbjT946IRnSw37zW+2lv8vzIGt7VIJ4 445IZkfZ5bPzt6//AK6beO0SrAhBUph1IyR
- jFXLyKdlE9soG87XLDJJbrj0xTreBrfUsT27XDoCo C/x8c1nGorcz18jqoVYOldO7XS/6sSEW
- 66XcLLG7AXCiJd2cLkZpgW5aSYxRuZHlaNsfxtnOR6AV phbaW2Xd8iyxCYKTyhB6Gs12vVjla
- 4haRWYAGL5c55JH6VlCV2/1LpxcXeP3X/4a/wB5K8cy6ewR WeVTwx5AB69fypGkvI7RJY/KxE
- +zOwfKP4c+pNVpp/LaRSJDsx5YB+8uOTTbd4pmjtNs9tFO29Wd s4x/F9K05Ha7RpWpuMbzSSW
- r6/gacswiR2tLiFpzPtVduc9Mj9TSRtbpZyrOWO2cEbTg4HGPrzn6 UxjErqIpY5pgU8xlXgnP
- GPTNWVuJWvZGNk5iYliuBken5HrXO9F/SZwVJw5ubdLXsytfXcUjCGCN ss6AMeig8VI8JIm+0
- 3kaNHIURtpw+eCfcA0yKMREXUg2jz0JJ6MO9aM/myRTRLHFIs8m62YgfdHc /XpQ5KNktjr9pF
- SjFO0O779ejMuG5eR5Wn3TgFclTjGOSK1nure4QrsEZKtkNz1G7J+vY1lOGECR CBm3JkheCXb
- jFNurNEvI4FuQsm0KAepULyfwpyhCT7GeIVGfvN2a7fn/AEiSLy7rSjJBNnzFXezH IDZ6fl1q
- nbuzzvPdSrnft3JwBk9Me+K2UMVvpj+W0LQtKMkLjJOMY+lZshimu7mGOPyysvyMfukA ZJx60
- Qne6tocEcXUmnBrQqBW/tLmPgoWLYGQOScn0z1qrb3E7PbCS2kuH8sSYUcggnBHpV2FWMWL Nt
- 8Ux8+QuM7e5T8ccVqwRv5nmeXi3YhlKgAqoH3c+3etZ1VFaoWJrR5fehr2ujLiuYnla4WRVYMS
- 0JP3yehA6ACssRBruNY7iKVwmd5XOOPm/wAK3J7R2vGEaLl2DiQLwBjB/wAaoXCzC8eNTCzsPv
- xx 44z/ACq6U430MsNKLctenW2n4f8ABKLSCOUlLhQxQMQo5BHAWlso8kJMXJdd5ZmJDnd94ew
- 9KvQ2 6BZTPCJJY32DGPmwMhvpVUz/AL6znELxIiPGpY8AHqT7Vrz8yaRGJrQqRaite5OLOJbh
- zbs2USRM k5Vm6j9M1kok93I22PftdWYAc7sHHPpxV1IIoLkJcXiSW4DKux8EnHB6VmxSSiQC3
- lXPy+YwPBIz z7CtKadnrfzM6cHyySfNpvrb5vf7jRmv2YPlUlZpFl+Ucj5eF/rVBEE6wTy285
- aRNqIhxuGeWqK7 uX862VWSRXjLL5S4LEHr07U/7SkmntIWId3Mke04wB6e1WqbjFWW559WUoR
- 5YKzejs/+GZlS7H8p YQ21W27mOSzE9fpWj5scek3UMzssq3SlVTjtyfwqoCzR3EoVQfOR41AA
- 9Tn6Uryqtys8jQ3DHcJE C9T2b6V0yV7L+u4V1pHmWnTvf7ixdXcguwiSswD/AHgfvnjBqGO5a
- DUFlaEN5UhRm25CseckVTa3 gm1GNTMqgLkSHIXjrT98ZeUkLH+82hWHJHb8apU4KNjeEI2dGL
- 5U9+t15vYdJdGRycbd4xICOjdB 9Kngc2kz5AmkjPllfXPcfrVXa20eXypdS4xzkcAe1IrwLAk
- bI0bFiWZmyWPRe1NxTVkVNtrlXp8i NRJyQuF3Hn2xSRSMzxzRoU8sFf3gyBnt+IqeMSrMVbyy
- 2MFduc+oqNTtZ9kZIwCPbirua+xvaD26 33/DUgkj/doXGWJ4XnKj3OMGo5plggjii2Om5juxk
- g45yT61JKJiyF2Csqc7uj/SqSzGO3XzmR5n GFwuQo7g8da0jG/meRjVGLTaej0ul+mxDGxZAw
- lZmxgBzkVOGK3EQgj3s5wvGevHf+dPdovNVAUI ckBkGAB+VJakf2Y0pYk+aVUKcEHHB+laOWl
- 7HC66cbLd6eRatYpRp8scUgOZAxQ/eycjOcVNazT2 rIYkDOYmBQjJGDmlMTSXcUbSJFN5Z3A8
- Hf3Bx+dILJ1+zmaYW3y9G549eK5nKLvzdQbhGnyyV0+i X/A/I1ILjzTLcTKrIWAYxAL14IHpx
- WzBJZpvlYmBCPLj8x8kDPb+dYETm2W4gbbIHlRmAGM7c9Pw NWJYIbm8kSOXbCrEQF24w3IU+p
- J6VyVaabtsv+GPXnQkr2bXez0+W5p2bW6XOwlLto42ZnA+++eM Z9KAlukiXE0MsaPJlmL4Ib3
- Ppk9KswC4trYJdRRvMSVMcahTyPm/SsC5Xy4rOC0aT7ORukV23HPr ntWEFzydmc0IyqVPcWnV
- 7misksOoTeUgnXG0pt4PBGR7CrMAW0tw8Ql2qgHmM3yjJwePxrKQ3HmK U3PtjKlh2B9at20pW
- 5jgumDBYjgg4AO8YyKupB2PTrU3FNvWy6XEV7Ga8InlGxGEcZ3fdPofpWva TW1vqRcafduGcl
- sTHOegHTHfPWsW5soxIQq8OS8be6nGD6k5xU0aXBE8TwziFpSR83KFuxPrnmoq RjOO+nqRKdO
- vB22atu1b5Jq5rXXnXMYtLWKSIxoEBdsn73JqhdDZdLvtZ/lUqGjbCkE8jH0q3bKm 5lluf3kY
- xO4JHPU/oMVnm5sprsywx3LEyAwoZM4Q9c+prKmmnZLRfqRGtaXLFO0dev53JrtIbXUH uWzEn
- zKsQ67QAOPcE1nwb2gzH59xErYlbcSpxzx6etblywnaSEoGQSdSv8IAU/nmsOS3NjqTFJ0j SP
- dsU5O7AxyPfNa0Jc0bPcdCq6vMqkdX6/oXRL9rhPmwNE73CBWONoQgk598d6f5dvHDMiLkM6FW
- Q4ynQn/GpFtobiwsgEk2om2UB+euAfwqEC0jiUBzcGNirBGxuGc5HtUJrZfcKjO14rm9P+CW0i
- lR 2lhjx5ZYx7+Qy9Rx3qFJhL9nlnTbGUIRIxtyvPX6njNWZftFxGWhAj85twJ6Y4GB/OqccUa
- MkM6s 3kB4wynAweVH1J6VMHda7/idtCrzK6WvVrf+vQsLd7olHWWNlTYw5Q98+uKmhv7g3O57
- dJCilGMK hSxY4DfSqdpFZtYrDPDO1zHkqivg4/iB9T3pF1GzhmzHbTfNtYHfnPBwfpmiVNO6U
- blKjFxlFw97 5foy5vnXU4pLu5jWO3BikOMByOf14FaFwM6fMyvHctJIDIsa42nI4/Cqdoz+Ra
- BHiMphdWEi7hIx P3vp2qFp7uCFXlaFsuR5YXDMw4zwP8msHBykrdDkadWrGUZJNaWv2+X6ot2
- 9y0Sz+ZJCgZiVm2fK hHBGPftVZJ4zEhS9t43iiwqkclh16etaXlwvpcESERkt5ZRvmJ9W/A9a
- r3FrDKlnbugjIt3ZHAxu z/8AqqYyg3qrEwq0XLWNnfW39aleKR5YVS4kTyooWjMjD5SG5FXIV
- 228KI6XMsSSBVTqExyG9W75 rJspoIbTyEXzBNIr7nORH/sn3yKtoGlkRmguJERShhiba6An7p
- Pc55z6VdWGr6FSjTlqk1b7v0/M mNgzTwLMk8kfl5DRvtJIHX8jVUPdRNFOkscoCZ8sqSSAe38
- 6tSp5ltbhfNilVmWPc2QU/iOPamS3 9tLAn76FAE2sAOu48D8uaUHJ9LjoYhyTly3js1YZNPLO
- 6mSdENwrSJ5eVGc8KPTmo1vdRt71pDdQ CcJh9yZ+o+tUrgRTw4+0RxRW9wYY5MHBU4yfwP8AO
- p3uJIblg6JcIWKzKBzvzwM84xxWyguW1jvV b2kOTlV+3/BenyC4W+l01ZHZZjHhDtXGTkdPpx
- Vq7iuRNJB5sUku5yVC8lhjn6YqjLI8sk4mWS3l mYsMnCrjhhj1PalTzknt4re5S7ZlJKoPnz6
- Z/SnZ6PTQ5HOEUm2rp9tP1/QXyb14n3KZw6bj5ePk wRwPTNLJLNfRKmIond+Qy5Kn0/SmXwRY
- YpA8tuGGzBcnOCBnjtmr6tNGm1XhnMZcSSKvDt0Dj2Bq XLRS/r/IKtfnUZJ3d9LKyX6FFIZbM
- QTW8yzEKQ6qp78Aj070+zigjmNvI6WrjMjPKudxA4/ADr61 VMs39nzKWDneoZ04CEdAfrVyJ4
- rrUZXLJmUNhT1U4wq/UmtJp8rv/X9andKlTdHmqX9V+qelvkSR W1w8UDPIkuFKh0G0EN1NVGJ
- s7qMzmVwIyGAbB3D7uf51eWae20hYZ03yJgTBePLb+6fc9qz3kjjl mnJ3MXEZicZKkjv9Kinz
- Nu+xGEhupaJ7ef8AkOt9QuttvKWRyLdspt5GM81MLuZ5EV7oWzOglZ3B OCO341BPOsqiFlj/A
- HZCxmMbdy9z9KijuHilD26R+azEssqhtoUe/bFa+zTV+Wz/AK/rY7XhIqD5 o+8/T/J/kTW6vd
- TuYY3ljcrs55OM4GfXjmr95E1rpAleJoiD5QLY+UNyc/54qW0gkMMN0sTOHjY+ XEduMHIHHc1
- SSyYMs8t6hjLYaNiTtcg8H3rBzTlvojgqV5Tqp8ytHbTd9rX/AEH2lqlxdpmL5IlK gD+Pb61c
- kuGO5VgmV1G8KrYEY7KQPp+tYSQxwPDbrclopF3Pz0bPI/KiW6eS5ke0LQws4Zg53FwO nOK1d
- Fzle56sKEaj953fbZGz9pkkMb4E7SOJXSPjac/Mv6UlzHJcT3txLFcRA3QdDu7HsKpQtHai Ga
- VvNSWL7iNhlYnp+VWbNkt7+RmZYwkZCh+fYqR3PpWbjy3aRpSp06bk1Hb1s/Ly9bGhLLA42I4J
- b52QHoAvzCqLuLTWIJHt7hICCVSST72FwD9OaZDDBEY0luUmWJGwqEhgCOpPfrTNzHy4ZLiNFR
- N6 NLzjHQfjUwilpuiKDTTunbqtbfkatrZ2ly8Uqv5RijVCrnPX1qpLMpYj7M8UeTG0mRgE8gf
- kKUS3 S/Pgojq6ncBgHHfiqj3Ey6eiSwNlduG7Ke2fXipjTlzXvf5lfV3Gq5wmn6Pb5lq1CXF6
- sUJJSM7E GPmZD1b3x61O1hb2trdJ507SM2YoxIQ2P61SVdU+1w3Vqse2MAqoQfMucYqvPd7J3
- AtbkzNKMsX4 wB2/CnyScvdehx1K8o1G9LeqevntY1keFCZkZlE0TYWQ5A9/w6fjUsN7I1laRy
- xGacEBFiOCRnOf pWS75RZVkQRqwVYerIOmCf1NXwYotRgu4xJvSBxvY/KTjpis5wVtTLE14VK
- d56vp/l3JHjmhinIu ol8yXfFleSAeo9smnwi6k1GC5cJORHk4XkEHBH1zVY3TJEkkhjKYUnK/
- c7bfyNSJHEjZtJX3JIFj BYncvr+Z5qWny6kVJJUfZyitfx8r6Fa7tEmSbbeQh/NDFATg8csB2
- Aq5bXAnACouxyG3EYJwDzn0 4xUV4tusM7SxPvEmAVbHJGB+XekYQIbSRC63C24V0DYALf8A1q
- pvmgkzSTjPDK9/LZ2+4jur9pYE jYRR74lZ9i7djZ9u3FTRzq8RBLiSaeN/KzyM8Efj1qN4He6
- SzSEQwratl3AJOOQc1EM3Ailms57c ufMjOcbuMcfl+tPlhy6f1/VicXOn7NKCs/l/wPzLGoMI
- JdsbPCYm2F3bhm6hh7YyMVRgkto90pu1 dwpAbaeRyP606a1lkx5MUxEhOZZDlVOOn1zUUkOYR
- HLFHmRhueNAoHHK/oPzq6ajypNnLT9iopc1 79rL/P8AMjfUYzpZU/vCWVgF4bA4YUfaPN1mRo
- mSb905CKvB45I9v8KintFtZLfyNmdrE5GdpHXN Z6SJs8x5o0RbgINq7Sc9G/3euRXTCnBxvEl
- zpQi7LR/12JLlGNvDLHbTJtXIdiCNq9fxz3qnE10J 3xGJHbLsqrg7T1/CtW5Vbi5ubVJS4ViU
- xnHHUfjUVpJL9rnhiWOSc/cIXOUI+Y/pWsZ2hqjOo+Wh LnjqvNozXSQpG6FdrKyxbUzhW7fnm
- s+b7VLd21uiBQgIKgY4IFXkmminhZWRYzIpZCO4yFHtkVGk 8kOqztAP424YbiPb8K6I3XQ4ZO
- U4WsrrZ3aIGa9ZI4js+QMANg5XvT40aOJ1fazSLvUqOUHUj+VQ IJd0YZZQjKd24c568ZoSHfc
- pHuaMlTww5+7nP0rVpJG7nSoWqRTsuutvw/4BGsgZvMV4wcgDCHkH r+VPZHM9zuKkiRixxwee
- MVHYSxQCIXhEkIJ4ThjmrRULEfvbCzN5Tk7h7k4/rTk7SIjKdSak07/h +H6jE8poN+5E3DIHU
- tg8/wBabsRZWMYRlR8x7gCxH5U7f5IUrbqMgFtx4wRg444zUkzvHBtjG0kd W6daFe520IuqpS
- qr4fIjaRPJJmkBkZ+Tjj2PSoQWIcqrJ1JLdtvUfWophlH+YFchskdgOlSgyZMg TaNp2hvSr5b
- IyvUcrJ6emtiBimAZCyqzAwlj0Hp05qDEBL4VyEkCq4bjPXninmaT7OZBGTHH8qgq DjcMZqJp
- CD5UZAIbawx0I/ritIpnjJpycHa3q7siSU8eWyM7MzN8vT2/KrUbKY5W2rjcMEDAB7A8 deKZ5
- EaXivEhkKksFAzhff1471IDEFdA4GZP3RP90evvTk10HScqUuSS0XzLlulxPdvC20XM2WUn 26
- qfc02YFYUSRJPLhYEZblVI5Q++e9BkaJmkJEkqsANvHGOefrQtxgzI+JYM9ckEH0JxXPaTd0ZT
- 9tKstPdW19Hf10LFjD5ksQeVAskLMQfvDB4H41fEq/Y2WNUDyhWZSvQ5xj8qyYXH2BlQ7FTDbm
- HI wfu5981rWLvLNLdq8cZDbIo3X7wPAP4VlWVrtnoxcaN5Tlr07fl+hZ+0XAmfdJtkgm2ByMj
- kY596 seVk2/lzwl4YShyOozy34Zq22be2WScpIFWTcoXBLDHP4UttfJBcpF5AmEeVmwBkk/MT
- 9K4HNtXi v6/r8zOGI9xzhC7/AD+9FOZb6LbJDFiOPKs4UYbpz9KozpHJO7SB/M4wynA5GQfoK
- 6Rrpv3xiwhe RXIcZxuGMVgO8waWO2QOFKljt3Zxxjn0NXQnJ9LHThazt71NX73t9+hMXmSwV4
- kM4bMfmDkMW7j9 alt4Wi1h0ll+VIGBPZiBwR+dRbfKiV7iCeOZnC7AcDr8xA9hVi2aSa2lC3d
- uSg2jenOc8fmKUn7r sZ/WWk409U93tb87lNRZSxbYJJATGvzluE55Dep60t0kEVyTEd8cO4Yj
- GM+n4VO+nziZt09u8ImH miNcEEj6U+VbecRS20se9FZGGPfHPqaaqK6s7o2eJg3abbflt+KuJ
- IQ8skTiSFs7lUn72FBJ+lTS SxbYDbRG4nAK7D83B5BoX7RMJ4whRTOuHbnjGDj6ikSUR36LAh
- SRY2SEkAgrjv6nrzWf6GanC9ub Va2vp8xlwUbT32ziGaCREVdxGc8mo1kPnuFh82CdS8ip94M
- PQ9gB2ptxfSDS4wEgKyc+Zs64P8zT JJJxdTXLELuZtqquMgrzitIxdtTpw8mqWi1d+v5bWLs9
- z51nAI5o/KPzMw4xtAA/CmxfacmFpYJX 83dOQvOf4R+Ap1vaW5sEmiG7KYWHPLjozD2FSRKrS
- MsjBJopEUEcZX1Pqcd6ycopNLockZxhzOCv b8GXI0txEFZXnm+ZlaM4wmcf/WqBo7Z7VLmC0d
- JNu1Ucg7gGwCPahMPfPFBeRbZG8zO0/KEHT6Zp GihlngcXIPnJ5ihSQAB2/PmsldPf8zVJqSk
- m9NWnf+rE0EdsHnRre5gnBBVHk5Ht+NMF4Y55DOFj m+0bgjjOz1U+9VUFzc2UsrXCgPKjEY+Z
- +xYH0FWpPLSzlFrieXzVMRPzZTnnnqc55puKvZ6m8/Z8 /Lvf+tW/8i1GkZCJc3kW4xMV2ccDO
- aoSzwx2qGPzWSSRSjl87VHT8O59akt7x2nhhVUaRoWZztB2 kHJFNmu5PLeIpDEXdQW2DGWHAH
- pxUxhJSsznp0pqoo307X0+5LUnnK26NALX7QzuGRo+BtXqf1ow jXUrXUxEJYCURnaVI+6f8aj
- Zrh7Zjb3UDMXVI1wcopOOfrVq4LyrID5bYYh0UYPy+/rxU7af10Om M40/3fPZvte/4ozw0siO
- jTpwjqrY7sf/AK1Q28V3AJoZVilRdxEoQY5H+FWbx7h9OVtibXj3gquN xzkmoJA4t1ntpQJZV
- Y4c5DKOcj+Vbx1Vu5tKMXTSvu9P+HWxBay+YwkaMRwJgMWGQSTwarz3dxdt cllEXnEyAlcDch
- 5Gfwq2k15HYTqkK+UzD92U+YMcbapXAAntY5VZbkbyX/gJB4GPetoJc17f1ucv PGFRt2u9O9v
- PyHjy5kmuZLjeguMB1PChjnB96kuNh1KeNJo4NzqzScjp1PHpVe3uYhFaxXCbGVGD rjG1s8Fv
- XFPjhVLwOG5VTGHcZV+OcfXtTas3cyp1/Zxm3o+nb8V/n6FQCW5yrrLJIT5kZB4BPGPx xWlFG
- 0LzTy+Y4hYoyI2CSRwfp/hVS4WOYW7W8oUrErhFz8oB5BPcjrW9DcIJrsBont5JfnPUk4+U j2
- oqzajojv8AbzjD4bJ9Nn6/8MZsNxGPPCFVdpMDfyu3HLY/l6VZuTDCkc0KFlkBjCpwUX39ST3p
- rWbw24JELs7qVUDDZHTHtSSXVzZ3LLMYjMuRjZ0J71npKXu6m1Kftbezlfvr/wAC5Zmth9kRYr
- W7 G9SDIzg57/n71NbJCssn2x4ordjuIcZbJ9/bpVBtSvTZrDlX2Abzt5B9KrhkmTz7gtuK4VV
- ONw/v fQGo9lNxtI6KeGrKHLUdr7W1a+8sSafbw2SyTbmMM6wyBW5GfX86fcfYY5CEjeONg4iJ
- Iy3Ytn0q vDdG1htw7xzZlBkVhnzDnjH4VuXFsizSGSB50kfdGo6gL/B9T7UTnKMlzO5zqcqdR
- Nu/bVK//BMm zliS4W1SWUr5ZUNu4Bxk/rVoC4S0QXOxbcRbXBAyr5xz75rOjlRL9XuIWjRjvd
- BwQRkgCn3EcbwB mndR5oyrEkoCMkH1NVOPvevzHiWlU93Vel7Py8wtbdUFz58hZYpRFlcZIOf
- UUkkdvHCTAXZygPzD 7gHGDxgmq+5vs8gglVYUkAAfktuOd34VZiLpLcmUpLtkEcgAwCTypHt6
- 1o73u2dNCvLl/eyd+i6D 2hWR7cJG8sUaESOp4DAcD8aginBJCNvGBtJ5wR2PqcmrMoJ0xoow2
- 4SBtwP3yvUj2qcTndJKIInj c71WNQCCetZqTsFLFOMHKyfkMtrlJbxJQI0kZBu3pkcZBGP1qC
- UGW6MkTJJ5ThFwvBB6VE1s8iZS WPaMqrD+LnJx9elFtcXK292GKRsJApiZMNycjn2xVKK+KI3
- U5ptqSldf1/WhZupHjkWO5jlJ2uAU faOvWqyzQm8iWMmE7SMzHcAfU0rwsq+ZcszFiERQfmI7
- nmq8MipOgYRQptdCzqCfQ/4VUYx5dA9t S9lLkl739bLUs3d4HQNCzo8cvloFb+Hj5TjqT2qnN
- qUt5JvSP7PtmbzVIySemB6cUyW4iW2RLWJl VGZn3/MeeMH3qpbovmK0Nu8jtIcI2DtIAG4+x6
- YrSnSio3aPLcYqMZ1F72t15dNVp+BbVY3uJt4c AMzRgHBwBxn61qaWIIZoGlYxpNgh5D8vQhl
- +tUTazknzjHGU3KQBjdnqfoKlEMMcsMy3EchjP7sD 7uQOWx6VFRqUbXNqtanOm4pu78m/xSt+
- Ben05AytG5khwvybuWYf0piCZr0JPKLYBHDSlTtPcY9M 1FKzXjWKLeRbowwZFBBfHORWnbqiP
- Gcbo0t2CmT5gcc/pmudylGPvasqhiWo+/rLbs/x/wAiulsr RWaq0su+IsfmyWbOQw9hViB/sP
- 2lpraSSScecrMegHUdPeqv9omSCFoY1V0iESDaOecnH86fa3CG 4Xyo3UlcoZm3fKQRj+ZpSjO
- z5lodjUqcJOS5k+l3+n+Qksm6II19ChKjZnOUXrg+pNSz3CvbxTWm ZRGwQAH/AFeTnaffrQ0E
- V7MYFAHzqGcD74A5YewFR+baw2Qmt7qKOSVpNgcEjZjGcAdfQ1GjtZan l1a8bxhFu99rfnZBc
- yywSARN5cBT5CxLbxn7w+mefWoLhg9lFcx7p42cNJsbgFSAP8arRQW02k27 W6TzSqG3IXzuPt
- 7e1Pgtr1Lq3uYgnnsm94yo2jJ6EfrWyjFddV+P9dyKtWnUg5J2a6d/XqvX8B66 hIb65ad4o5f
- m+Vkz1HPHpVK7tWXT7YJAzl4CJGH3Q+eB9QP51Tv2mkuoCI9kUhLKpGTtBwcn9ald PIEodZ1e
- ab92+/5W9xW8aajZowlJRcWn8v6t+QyK3n/dxtuAAJMnTDHoCfWpZLtkWJbiCQXCxtEW Q7T8x
- 6cDrUM81zb3EyRk7FLJhgG9OaheS8jeK4mt5eSGiJHUDgn34rXlctXYdec+TWzXTVrX1IGk t2
- l8pt8YIGd3O0r26UlzNEI0eMofNyYyvYDgg+p96givWnvBPsiL+ZtUEZyM49KhmSRrolYVQlm4
- K8j2+tdMafvK4vbTnOLT9O/53JbadI2YzRO2MjBPHI61WIWS7iKySIUYEKxJL+1SMItjeYjh8E
- gA 9B+X41FEF+yEjzJI0bBOCeT0FapJamFSmuf33o+70+64scFqJV3sEycqSTyT/wDXqzJIMRy
- BhNgF ZAqkbccAn15qgJURvmid95x2+XtThKy3SxmEkFtp9vr9abi27l+1jF8ylZLol/wNSx9o
- R4U+YM4H 7ojoQDzxSODNbyyrIJVSYBSp6++MdKpJMHVTtRWVxjA7/wCeo70Ga5jKhI0Ul8MNv
- Ue1P2epz+2W 769lf8Cz5jQ3ckWCQcgFhkAdc9O9VJ3kYxeTIWKZEiMD0GORx0qWNXnlIMckbg
- c89ewxSJbTxlpk VtqPhi65ww4IP9RVKyfmcuKrVKkLRk/S6X6XKqRiWXJ8x9nKANgfjVeOJt8
- AiKA5PmEnnPXP0rR8 uaNPljYnG0ADnrx/OqkUN3MQ/lnYpJPyHnHJraM9HqcuJpRpOLlfn7aK
- /lrsTNdll2Sja5wq8djy aHx533ldEYBiF4NKsLKqlQZAVynHb1zUaoWARJUK7gZBg5PoOnXFT
- 7vQ6pzqO0pu9/T8+v3lnzld ZXkRn3NncBjdxz26U23knW5R7Et8zhcMoOARwTkY4phgAI2jIV
- Sck579KSR1gjtWjcywSStgISCu MEZNSkun/AOHGQtT5Zu172X6dvxL0QdY4nRSltO3VjnIB5P
- 4VuwK0SWXlobqaRJCuzptJ5H1FVLe 2jlWKLyJYEVsFXblic4I9B6+tKjzRX0UbMI5S6DJ/hJb
- t6VwVHz6HoRm6ivdNx1fb8DqZLiBNOgD W0/3N4QtlmHGT+dULgSXeryKlrKoaQ72BHHy5q3PM
- zJevHNBJKLkALs524AOPQVcijiBZT5gjtmM Bkz97eMj8e2a8yM+RXtqXRrOhBzjo/V6fIzYre
- SKK5K7pJFkjJIPDDHb8OazxJENWQRRysCDhlbg A9z9a0TA1pMsltchZRbner8gkDg/lxWXEqy
- WisWDfKuHTgR84w3qc10wd7s9Jy505R2a31/Ine2m jZJJGaNGO9WkYtuC8n9P51oWSWc10YJL
- iNJpnypwQF44U+p7UyZTBpgtpnZ4TJ8zn+HAwcE9iaqN uDqUSFJX8xsheVwOV+vcfWod5x3OF
- /7XFq+uy5bW+7qW7q6uLOS8gVRGJZ0cOy5AwASKZb6legSG OO2ZPOY5EQ+ZWFSJNNLbRXEtlJ
- GVjxC7kEbMfNkdz71Wmmne1txB5Z3orLtXHy1MYpqzirkKFOpZ cqvt9332LSXsMcNrHcSKksk
- uQBwemPyPaqKTKXYSI4ztHXBQ54H88+tSW9xLc3hM0cYjkYSKSgyF wQSD6DirSzQTyC3t7ORQ
- JFYlmBbAGQM/WnyqDen4k0aCpSlGMb97NafkQRXNm1uiuVOHKlffPDfQ UyRAwuAZFdoblGLDO
- 0tj5uP7vtWtbwQ3FkbmMRCSZ94wByc4wB7/AM6jXSjPE0LZjdxvI7oQDtDe tZqtBN9CcPXpKX
- vJr1s/wsUoo2WcSeTN5fzKpRuF55qHybmeRBb286OU2sHbluoz+FW7eO5srP7Q xOZlMgVhnfx
- gsPYGmv5loPtNzdJu8oqO25uMEe1aKb5nb9T06GLvUlyyS00639UWLXy/IMbyxyJb qIkZOCVc
- YOT35omDw3KQrJE2bZjkL1KnHHoPak8qO3jj+1uqxKWWZV4LNwQPw4q3FZLc6xKFSWRF CsoDY
- OQMkfjWMpJNyvp/X/DlyqR5ZtbW62t8iC2gWztZvLlXDSDG7kIM/dPuc1IkD2m37UrRKuUX HH
- llzwp9TzmrLyTSI8kVsLcv87JKAeW/wAqrdPJLFbMpKxiMhQ/PQg5J7moUpSevU56L9rZuS+/X
- 8Cf7Ozz7YLWTzI28oyqfl+7yD7mqkRuo4438yCRVfy1Qx8jP3c+46Vqgzi3muTIj+ZcRCEJkD5
- vv GqMscxcwW0ihg7mUMOcr0b6YqYSvo/67jptynaMk4+jf37laeLyQyI4WdnVdwOAo6c/rUA0
- y4i85 WaQbztVi2dw6k/iKvQE3ulNmCSVVKMrqcE8/MaiS7mGozrOTNDIjSEIMEkdMegx2rZTm
- rpbo7ozq Rva116O/oUPs4SwJa7+Zf9WpY/dbGPy71bK7J4Ty6hWLMvTrzgemOahnWcCye3tnf
- NswDkZUqOS1 U542ihMjrO6OC0bK3BXHJ+nNbL37XZMqilDl5rX+/wDr5sjks5pXZopPtKMzOr
- gnDYHXH8qdHI9x BE9xDIwYKY0HDFQCvX6jNakcEuz5EYuAxZU4w6DqPbB6VizvNPALiRkK5+V
- QMfe44HpWkJ8+hzrF KpL2bkkl1t+hLdrE0KTCQSo8m3CDB3jgjNV0g3Swkz/ZoIn+TzTn3I46
- 1O1iPs0UUbG4Kk/u4j1Y d+fqaLi0uHtT5YMaxseDzn2+vWqjNWtcwniIygqTavfe35X/AMiuL
- hXuAtsY2LuVBA/vf4U2C6lI j8sgAsIxIB8qgDGCO5680ItrHC8QBtwASsjHOM4Kr9evNWrGyj
- lbcZTtD8p6MDwfpVycFF3NfaNQ 5ai28v1X5DgJFjztnkYErFznCdDn396tpLJBA8UMLvMW3gz
- DcQoHQ5qK7uHjupzg5QADacAZxkfU 1YH2x3zEgUSKWBZQSRngg+lYPVJtHp048y/eR9P+CTW0
- EM16I5CVjZMh84z6H86sTWtpbtFIJBvc Mixn19RU9pmCzcyJ5qtKpDKMDnpj2zU9pbia5ka4j
- eSdZWjQrwCG7j8a451WpN30RksdPn3tFeZk zRpMyhgrMgCqFGCSe31NXZL6S28mMK0jKmGbGf
- nJwAPxrRm02Y7txjd02tIVXGHXv9AKqm2hudWM UW5pnTzPMB+VgP4gPQVKqwktdUjrlWpVI81
- rpev47GHJHLFdRRGJjPuQOvXJB5xVk2c02p3ASFt7 KxQnG1VHAB/2vepp7aC0lWaSVrgbGUFW
- Odx561RS4aTTokjMkchAA+bnqSB9Sa6VJyV4jdWd1Ui/ d762+epbRUs57VrieDdsTdEEz8vfP
- GCcmnTQi41ZxERIFDLtTghgPlJqtOjTSkwxMZD80pc5wcdB 6Ac09LmS0YbZo92xhG2zIKEcE+
- p9DU8r3T1MKlNuo3Tl7z7pfp/w5UY3K6mGwnnN8gAHD5HJA+lM jliil2eVKu1AG3Nna3UH6Yq
- x51vdTxp5UodTuUbuSMZx+GKkmhs3tMxuRcdSO20jn8a15ukkOU5y k1Ui1fTRbeen+RDc3iJZ
- GKGJ+CAHJ4Y5BBH61XV/tF+ZpnEbSTq5XpuIHb25rQZbmeyt7cW+1oNm G2g7jnr9AKkkt2zcy
- xR+WwugVLDIIyDkD0qVOMVbqcjrRgnSirX6tpsx21CeW5DSf6yJiqggYB71 dSe2/dyyyxqyw7
- XQpkk5zmqxgAvJzc8tvdnRMA885+grNG+S5hXckbBfmLDIJA4rf2cJLTQ6bUql J+7ZrdpL+vw
- L8TrPG0Drl2OVA4JA5Jo2xRi2nZ8jawk2HGGbpQbq9itIJnktg5AaMiP7w6Ht26U0 XX7udftF
- uIjKMKU5Y9iOOgotLpt/Xkct51ZabLzf+RUhkZ7MEB5WyUGD09Sa1I7MJZlrtm2jIV0O AVHp9
- ayVuGxIpZWywyUUAY7/AJ1YUf6JiIuxchvLJyVA9a0qRl3sdjhUkoqU7Ly/z/zNK3hdM3lv NC
- ZMHam3JB9PxFTW5mgjt2ifM7DzdrcjaB6fic0yIwQwyzSMZY1cIAhwSSOopJopblLVkhkjkfIV
- G/hHQ5/KuSTu7PY5/rMNFVel7cz6+T7/AHISRJG1aNrcLOUh/gTAI9f6VZtPPMcwA8mMPnEi5K
- sT 8o+lRx2csEqtuI2KY5FHUknIqUlRqIkWZdskbloyeUkI+UGlOSasjqrYihO/IunUn8yS2Fs
- yAp5p ZjuGdozyv5c1WLESsIreGWJUzCSgO6MdxUMBEdujt5kxdWaQ7uBnsPTPQVnyTQTsdknl
- RHIRSclT 12mnCldnDh6bi25W16/5dTQjnjS4FzBhY5QPLtx944OAfp61HPLPa6vKZWZlBOEU4
- yFB5HsDUEf7 /TlkEDkruNuV4BXPzH8Kvubf7T81ldSQ5Oz58kjGetNpKW1ylzNuqo8yenToZk
- UsTWcc9xA7KVGw bsZQkhsH61dk8w3RNqgM2/bGW+ZSmPmIz0xUxgmeximghA3qC6FchAT0Hpn
- FUp3jmjurgJJaywS7 FRm6Z6jiqupPQJSjNJX1/BeqILgo1iXk/eStKu0A+oxn8ayt8q3MeJAk
- gUlQy5zjtirZgjjVNkoX zAGG9uFAOMdPWmG1zPJErRuUb75PTHUE/jXVT5YrUUYqnCUZNL5Mz
- 1MIYdWjcMfLXqp7Akj1pzl4 ZA/kOwlLID2wBy35mmvCFViZELpLtG3ocDn8jTVd0aFVheT5jG
- zZ+VN/c/rXRvqjhqVJpKcZOy/L yJUQ/Yo8FW3qSCRnft7j2qq8saxwqoUOWGWLZBHcYx1z3q9
- NZ7JlC/NalSY3B68/41SCPLKCBGSG wTjH3qcHF63NbOajbWz0av8A0x6/O0qugc4ypjGMD3+l
- DxvGIjMhQtH8rFcbu+fyqZVtzZxjLJNu IJJ9OQPxp6Ry3d38zGQHhlA55HUdhjApc3XoTJTfv
- 2vr13/P8Rm+CaUptjcMDtIOADQLFwltIYpj bv8AOzL/ABKBjC5HXNacOnI2jwSlUS4MZCgLgv
- zjP51cnsljs7iGOQmXzAJEz3HXA7D2rnliYp2R nXxftmlJ2/B/hb9TFuFs57dGjhmh/egRl2z
- lO7H1HHFOhsd90ZXSVLZZNwJbjHXn1rQa5RHiLGJ0 KFgu3O0d1+tI8dv9reQXqfO5KoScBSKX
- tJJW/wCCebWhKEeVrfrvf08zDW5EN48kCrKgOYxt9uvT pVJGuJGUpbyM5UgFF+9z1+ldBLZkP
- Ir4uZGbgQqFAwuRj+tZkQkWBJchZVRQcd92cY/KumE4tNoq MqfMuV2b7oozrKh8uNjt6MMdPp
- 6VDGlyp/1eSTlsIBgVceS6nkWOIM+xdhxHyO/Jx1zTSLhYla4i kjITauR2Oc5Hc1upWVnYqUq
- LqK7s+lnp8hjpiIGbzfMB4Yfdx9MVnRlkTaxWQGTPyjG0Y5H16VMV
- ZpLZVl3hEJOfXng+tTh59q/6tgyA/d6ZrRaIz5FUlfVee5//2Q==
-X-ABShowAs:COMPANY
-UID:F0A6918D-8E09-43FA-9684-226810B8A96F
-END:VCARD
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/F0A6918D-8E09-43FA-9684-226810B8A96F.vcf (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/F0A6918D-8E09-43FA-9684-226810B8A96F.vcf)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/F0A6918D-8E09-43FA-9684-226810B8A96F.vcf (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/F0A6918D-8E09-43FA-9684-226810B8A96F.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,2010 @@
+BEGIN:VCARD
+VERSION:3.0
+N:Inc.;Test;;;
+FN:Test Inc.
+ORG:Test Inc.;
+EMAIL;type=INTERNET;type=WORK;type=pref:testinc_sf at example.com
+TEL;type=WORK;type=pref:777-777-7777
+item1.ADR;type=WORK;type=pref:;;3 TV Street;San Francisco;California;99999;US
+item1.X-ABADR:us
+NOTE: Company with picture
+PHOTO;BASE64:
+ /9j/4AAQSkZJRgABAQAAAQABAAD/7QA8UGhvdG9zaG9wIDMuMAA4QklNBAQAAAAAAB8cAVoAAx
+ sl RxwCAAACAAIcAhkAC1Bob3RvIEJvb3RoAP/iG6hJQ0NfUFJPRklMRQABAQAAG5hhcHBsAgA
+ AAG1u dHJSR0IgWFlaIAfaAAEAEwAJADEABGFjc3BBUFBMAAAAAAAAAAAAAAAAAAAAAAAAAAAA
+ AAAAAAD2 1gABAAAAANMtYXBwbFYcEOZVYuhIRg5LwLIi62wAAAAAAAAAAAAAAAAAAAAAAAAAA
+ AAAAAAAAAAA AAAAEXJYWVoAAAFQAAAAFGdYWVoAAAFkAAAAFGJYWVoAAAF4AAAAFHd0cHQAAA
+ GMAAAAFGNoYWQA AAGgAAAALHJUUkMAAAHMAAAIDGdUUkMAAAnYAAAIDGJUUkMAABHkAAAIDGF
+ hcmcAABnwAAAAIGFh Z2cAABoQAAAAIGFhYmcAABowAAAAIHZjZ3QAABpQAAAAMG5kaW4AABqA
+ AAAAOGRlc2MAABq4AAAA ZGRzY20AABscAAAALm1tb2QAABtMAAAAKGNwcnQAABt0AAAAJFhZW
+ iAAAAAAAAB7vQAAQXsAAAJL WFlaIAAAAAAAAFYqAACp0AAAFF9YWVogAAAAAAAAJO8AABS1AA
+ C8glhZWiAAAAAAAADz2AABAAAA ARYIc2YzMgAAAAAAAQu3AAAFlv//81cAAAcpAAD91///+7f
+ ///2mAAAD2gAAwPZjdXJ2AAAAAAAA BAAAAAAFAAoADwAUABkAHgAjACgALQAyADcAOwBAAEUA
+ SgBPAFQAWQBeAGMAaABtAHIAdwB8AIEA hgCLAJAAlQCaAJ8ApACpAK4AsgC3ALwAwQDGAMsA0
+ ADVANoA4ADlAOoA8AD1APsBAQEHAQwBEgEY AR4BJQErATEBOAE+AUUBSwFSAVkBYAFmAW0BdQ
+ F8AYMBigGSAZkBoQGoAbABuAHAAcgB0AHYAeAB 6QHxAfoCAgILAhQCHAIlAi4CNwJAAkoCUwJ
+ cAmYCcAJ5AoMCjQKXAqECqwK1Ar8CygLUAt8C6gL0 Av8DCgMVAyADKwM3A0IDTQNZA2UDcAN8
+ A4gDlAOgA6wDuQPFA9ID3gPrA/gEBAQRBB4ELAQ5BEYE VARhBG8EfASKBJgEpgS0BMIE0QTfB
+ O4E/AULBRoFKAU3BUcFVgVlBXQFhAWTBaMFswXDBdMF4wXz BgMGFAYkBjUGRQZWBmcGeAaJBp
+ oGqwa9Bs4G4AbyBwMHFQcnBzkHTAdeB3AHgweWB6gHuwfOB+EH 9AgICBsILwhCCFYIagh+CJI
+ Ipgi6CM4I4wj3CQwJIQk2CUsJYAl1CYoJoAm1CcsJ4An2CgwKIgo5 Ck8KZQp8CpIKqQrACtcK
+ 7gsFCx0LNAtLC2MLewuTC6sLwwvbC/MMDAwkDD0MVgxuDIcMoQy6DNMM 7Q0GDSANOg1UDW4Ni
+ A2iDbwN1w3xDgwOJw5CDl0OeA6TDq8Oyg7mDwIPHg86D1YPcg+OD6sPyA/k EAEQHhA7EFgQdh
+ CTELEQzhDsEQoRKBFGEWQRgxGhEcAR3xH+Eh0SPBJbEnoSmhK5EtkS+RMZEzkT WRN6E5oTuxP
+ bE/wUHRQ+FF8UgRSiFMQU5RUHFSkVSxVtFZAVshXVFfcWGhY9FmAWgxanFsoW7hcS FzUXWRd9
+ F6IXxhfqGA8YNBhZGH0YoxjIGO0ZExk4GV4ZhBmqGdAZ9hodGkMaahqQGrca3hsGGy0b VBt8G
+ 6MbyxvzHBscQxxsHJQcvRzmHQ4dNx1gHYodsx3dHgYeMB5aHoQerh7YHwMfLR9YH4Mfrh/Z IA
+ QgMCBbIIcgsyDeIQohNyFjIY8hvCHpIhUiQiJwIp0iyiL4IyUjUyOBI68j3SQMJDokaSSXJMYk
+ 9SUkJVQlgyWzJeImEiZCJnImoybTJwMnNCdlJ5Ynxyf4KCooWyiNKL4o8CkiKVUphym5KewqHy
+ pS KoUquCrrKx4rUiuGK7or7iwiLFYsiiy/LPQtKS1eLZMtyC39LjMuaS6eLtQvCy9BL3cvri/
+ kMBsw UjCJMMEw+DEwMWcxnzHXMg8ySDKAMrgy8TMqM2MznDPVNA80SDSCNLw09jUwNWo1pTXf
+ Nho2VTaQ Nss3BjdCN343uTf1ODE4bTiqOOY5IzlgOZ052joXOlQ6kjrPOw07SzuJO8c8BjxEP
+ IM8wj0BPUA9 fz2/Pf4+Pj5+Pr4+/j8/P38/wEAAQEFAgkDEQQVBR0GIQcpCDEJOQpFC00MWQ1
+ hDm0PeRCFEZUSo ROxFMEV0RbhF/EZARoVGykcOR1NHmUfeSCNIaUivSPVJO0mBScdKDkpVSpt
+ K4ksqS3FLuEwATEhM kEzYTSBNaE2xTfpOQk6MTtVPHk9nT7FP+1BFUI9Q2VEkUW5RuVIEUk9S
+ mlLlUzFTfFPIVBRUYFSt VPlVRlWSVd9WLFZ6VsdXFFdiV7BX/lhMWJpY6Vk4WYZZ1VokWnRaw
+ 1sTW2NbslwDXFNco1z0XURd lV3mXjdeiV7aXyxffl/QYCJgdGDHYRlhbGG/YhJiZWK5YwxjYG
+ O0ZAhkXGSxZQVlWmWvZgRmWWav ZwRnWmewaAZoXGiyaQlpX2m2ag1qZGq8axNra2vDbBtsc2z
+ LbSNtfG3Vbi5uh27gbzpvk2/tcEdw oXD7cVZxsHILcmZywXMcc3hz03QvdIt053VDdaB1/HZZ
+ drZ3E3dwd854K3iJeOd5RXmjegJ6YHq/ ex57fXvcfDx8m3z7fVt9u34bfnx+3H89f55//4Bgg
+ MKBI4GFgeeCSYKrgw6DcIPThDaEmYT8hWCF w4YnhouG74dUh7iIHYiBiOaJTImxihaKfIrii0
+ iLrowUjHuM4o1Ija+OF45+juWPTY+1kB2QhZDu kVaRv5IokpGS+pNkk82UN5ShlQuVdZXglkq
+ WtZcgl4uX95himM6ZOpmmmhKafprrm1ebxJwxnJ+d DJ15neeeVZ7DnzGfoKAPoH2g7KFbocui
+ OqKqoxqjiqP6pGqk26VMpbymLqafpxCngqf0qGWo2KlK qbyqL6qiqxWriKv7rG+s461WrcuuP
+ 66zryivnbARsIew/LFxseeyXbLTs0mzv7Q2tK21JLWbthK2 ibcBt3m38bhpuOG5WrnSuku6xL
+ s+u7e8MLyqvSS9nr4ZvpO/Dr+JwATAf8D6wXbB8cJtwunDZsPi xF/E3MVZxdbGU8bRx07HzMh
+ KyMnJR8nGykXKxMtDy8LMQszBzUHNwc5CzsLPQ8/D0ETQxtFH0cjS StLM007T0NRT1NbVWNXb
+ 1l7W4tdl1+nYbdjx2XXZ+tp/2wPbiNwO3JPdGd2e3iTeqt8x37fgPuDF 4Uzh0+Ja4uLjauPy5
+ HrlAuWL5hPmnOcl56/oOOjC6Uzp1upg6urrdev/7IrtFu2h7izuuO9E79Dw XPDp8XXyAvKP8x
+ zzqvQ39MX1U/Xh9m/2/veM+Bv4qvk5+cn6Wfro+3j8CPyZ/Sn9uv5L/tz/bmN1 cnYAAAAAAAA
+ EAAAAAAUACgAPABQAGQAeACMAKAAtADIANwA7AEAARQBKAE8AVABZAF4AYwBoAG0A cgB3AHwA
+ gQCGAIsAkACVAJoAnwCkAKkArgCyALcAvADBAMYAywDQANUA2gDgAOUA6gDwAPUA+wEB AQcBD
+ AESARgBHgElASsBMQE4AT4BRQFLAVIBWQFgAWYBbQF1AXwBgwGKAZIBmQGhAagBsAG4AcAB yA
+ HQAdgB4AHpAfEB+gICAgsCFAIcAiUCLgI3AkACSgJTAlwCZgJwAnkCgwKNApcCoQKrArUCvwLK
+ AtQC3wLqAvQC/wMKAxUDIAMrAzcDQgNNA1kDZQNwA3wDiAOUA6ADrAO5A8UD0gPeA+sD+AQEBB
+ EE HgQsBDkERgRUBGEEbwR8BIoEmASmBLQEwgTRBN8E7gT8BQsFGgUoBTcFRwVWBWUFdAWEBZM
+ FowWz BcMF0wXjBfMGAwYUBiQGNQZFBlYGZwZ4BokGmgarBr0GzgbgBvIHAwcVBycHOQdMB14H
+ cAeDB5YH qAe7B84H4Qf0CAgIGwgvCEIIVghqCH4IkgimCLoIzgjjCPcJDAkhCTYJSwlgCXUJi
+ gmgCbUJywng CfYKDAoiCjkKTwplCnwKkgqpCsAK1wruCwULHQs0C0sLYwt7C5MLqwvDC9sL8w
+ wMDCQMPQxWDG4M hwyhDLoM0wztDQYNIA06DVQNbg2IDaINvA3XDfEODA4nDkIOXQ54DpMOrw7
+ KDuYPAg8eDzoPVg9y D44Pqw/ID+QQARAeEDsQWBB2EJMQsRDOEOwRChEoEUYRZBGDEaERwBHf
+ Ef4SHRI8ElsSehKaErkS 2RL5ExkTORNZE3oTmhO7E9sT/BQdFD4UXxSBFKIUxBTlFQcVKRVLF
+ W0VkBWyFdUV9xYaFj0WYBaD FqcWyhbuFxIXNRdZF30XohfGF+oYDxg0GFkYfRijGMgY7RkTGT
+ gZXhmEGaoZ0Bn2Gh0aQxpqGpAa txreGwYbLRtUG3wboxvLG/McGxxDHGwclBy9HOYdDh03HWA
+ dih2zHd0eBh4wHloehB6uHtgfAx8t H1gfgx+uH9kgBCAwIFsghyCzIN4hCiE3IWMhjyG8Ieki
+ FSJCInAinSLKIvgjJSNTI4EjryPdJAwk OiRpJJckxiT1JSQlVCWDJbMl4iYSJkImciajJtMnA
+ yc0J2UnlifHJ/goKihbKI0ovijwKSIpVSmH Kbkp7CofKlIqhSq4KusrHitSK4YruivuLCIsVi
+ yKLL8s9C0pLV4tky3ILf0uMy5pLp4u1C8LL0Ev dy+uL+QwGzBSMIkwwTD4MTAxZzGfMdcyDzJ
+ IMoAyuDLxMyozYzOcM9U0DzRINII0vDT2NTA1ajWl Nd82GjZVNpA2yzcGN0I3fje5N/U4MTht
+ OKo45jkjOWA5nTnaOhc6VDqSOs87DTtLO4k7xzwGPEQ8 gzzCPQE9QD1/Pb89/j4+Pn4+vj7+P
+ z8/fz/AQABAQUCCQMRBBUFHQYhBykIMQk5CkULTQxZDWEOb Q95EIURlRKhE7EUwRXRFuEX8Rk
+ BGhUbKRw5HU0eZR95II0hpSK9I9Uk7SYFJx0oOSlVKm0riSypL cUu4TABMSEyQTNhNIE1oTbF
+ N+k5CToxO1U8eT2dPsU/7UEVQj1DZUSRRblG5UgRST1KaUuVTMVN8 U8hUFFRgVK1U+VVGVZJV
+ 31YsVnpWx1cUV2JXsFf+WExYmljpWThZhlnVWiRadFrDWxNbY1uyXANc U1yjXPRdRF2VXeZeN
+ 16JXtpfLF9+X9BgImB0YMdhGWFsYb9iEmJlYrljDGNgY7RkCGRcZLFlBWVa Za9mBGZZZq9nBG
+ daZ7BoBmhcaLJpCWlfabZqDWpkarxrE2tra8NsG2xzbMttI218bdVuLm6HbuBv Om+Tb+1wR3C
+ hcPtxVnGwcgtyZnLBcxxzeHPTdC90i3TndUN1oHX8dll2tncTd3B3zngreIl453lF eaN6Anpg
+ er97Hnt9e9x8PHybfPt9W327fht+fH7cfz1/nn//gGCAwoEjgYWB54JJgquDDoNwg9OE NoSZh
+ PyFYIXDhieGi4bvh1SHuIgdiIGI5olMibGKFop8iuKLSIuujBSMe4zijUiNr44Xjn6O5Y9N j7
+ WQHZCFkO6RVpG/kiiSkZL6k2STzZQ3lKGVC5V1leCWSpa1lyCXi5f3mGKYzpk6maaaEpp+muub
+ V5vEnDGcn50MnXmd555VnsOfMZ+goA+gfaDsoVuhy6I6oqqjGqOKo/qkaqTbpUylvKYupp+nEK
+ eC p/SoZajYqUqpvKovqqKrFauIq/usb6zjrVaty64/rrOvKK+dsBGwh7D8sXGx57JdstOzSbO
+ /tDa0 rbUktZu2EraJtwG3ebfxuGm44blaudK6S7rEuz67t7wwvKq9JL2evhm+k78Ov4nABMB/
+ wPrBdsHx wm3C6cNmw+LEX8TcxVnF1sZTxtHHTsfMyErIyclHycbKRcrEy0PLwsxCzMHNQc3Bz
+ kLOws9Dz8PQ RNDG0UfRyNJK0szTTtPQ1FPU1tVY1dvWXtbi12XX6dht2PHZddn62n/bA9uI3A
+ 7ck90Z3Z7eJN6q 3zHft+A+4MXhTOHT4lri4uNq4/LkeuUC5YvmE+ac5yXnr+g46MLpTOnW6mD
+ q6ut16//siu0W7aHu LO6470Tv0PBc8OnxdfIC8o/zHPOq9Df0xfVT9eH2b/b+94z4G/iq+Tn5
+ yfpZ+uj7ePwI/Jn9Kf26 /kv+3P9uY3VydgAAAAAAAAQAAAAABQAKAA8AFAAZAB4AIwAoAC0AM
+ gA3ADsAQABFAEoATwBUAFkA XgBjAGgAbQByAHcAfACBAIYAiwCQAJUAmgCfAKQAqQCuALIAtw
+ C8AMEAxgDLANAA1QDaAOAA5QDq APAA9QD7AQEBBwEMARIBGAEeASUBKwExATgBPgFFAUsBUgF
+ ZAWABZgFtAXUBfAGDAYoBkgGZAaEB qAGwAbgBwAHIAdAB2AHgAekB8QH6AgICCwIUAhwCJQIu
+ AjcCQAJKAlMCXAJmAnACeQKDAo0ClwKh AqsCtQK/AsoC1ALfAuoC9AL/AwoDFQMgAysDNwNCA
+ 00DWQNlA3ADfAOIA5QDoAOsA7kDxQPSA94D 6wP4BAQEEQQeBCwEOQRGBFQEYQRvBHwEigSYBK
+ YEtATCBNEE3wTuBPwFCwUaBSgFNwVHBVYFZQV0 BYQFkwWjBbMFwwXTBeMF8wYDBhQGJAY1BkU
+ GVgZnBngGiQaaBqsGvQbOBuAG8gcDBxUHJwc5B0wH XgdwB4MHlgeoB7sHzgfhB/QICAgbCC8I
+ QghWCGoIfgiSCKYIugjOCOMI9wkMCSEJNglLCWAJdQmK CaAJtQnLCeAJ9goMCiIKOQpPCmUKf
+ AqSCqkKwArXCu4LBQsdCzQLSwtjC3sLkwurC8ML2wvzDAwM JAw9DFYMbgyHDKEMugzTDO0NBg
+ 0gDToNVA1uDYgNog28DdcN8Q4MDicOQg5dDngOkw6vDsoO5g8C Dx4POg9WD3IPjg+rD8gP5BA
+ BEB4QOxBYEHYQkxCxEM4Q7BEKESgRRhFkEYMRoRHAEd8R/hIdEjwS WxJ6EpoSuRLZEvkTGRM5
+ E1kTehOaE7sT2xP8FB0UPhRfFIEUohTEFOUVBxUpFUsVbRWQFbIV1RX3 FhoWPRZgFoMWpxbKF
+ u4XEhc1F1kXfReiF8YX6hgPGDQYWRh9GKMYyBjtGRMZOBleGYQZqhnQGfYa HRpDGmoakBq3Gt
+ 4bBhstG1QbfBujG8sb8xwbHEMcbByUHL0c5h0OHTcdYB2KHbMd3R4GHjAeWh6E Hq4e2B8DHy0
+ fWB+DH64f2SAEIDAgWyCHILMg3iEKITchYyGPIbwh6SIVIkIicCKdIsoi+CMlI1Mj gSOvI90k
+ DCQ6JGkklyTGJPUlJCVUJYMlsyXiJhImQiZyJqMm0ycDJzQnZSeWJ8cn+CgqKFsojSi+ KPApI
+ ilVKYcpuSnsKh8qUiqFKrgq6yseK1Irhiu6K+4sIixWLIosvyz0LSktXi2TLcgt/S4zLmku ni
+ 7ULwsvQS93L64v5DAbMFIwiTDBMPgxMDFnMZ8x1zIPMkgygDK4MvEzKjNjM5wz1TQPNEg0gjS8
+ NPY1MDVqNaU13zYaNlU2kDbLNwY3Qjd+N7k39TgxOG04qjjmOSM5YDmdOdo6FzpUOpI6zzsNO0
+ s7 iTvHPAY8RDyDPMI9AT1APX89vz3+Pj4+fj6+Pv4/Pz9/P8BAAEBBQIJAxEEFQUdBiEHKQgx
+ CTkKR QtNDFkNYQ5tD3kQhRGVEqETsRTBFdEW4RfxGQEaFRspHDkdTR5lH3kgjSGlIr0j1STtJ
+ gUnHSg5K VUqbSuJLKktxS7hMAExITJBM2E0gTWhNsU36TkJOjE7VTx5PZ0+xT/tQRVCPUNlRJ
+ FFuUblSBFJP UppS5VMxU3xTyFQUVGBUrVT5VUZVklXfVixWelbHVxRXYlewV/5YTFiaWOlZOF
+ mGWdVaJFp0WsNb E1tjW7JcA1xTXKNc9F1EXZVd5l43Xole2l8sX35f0GAiYHRgx2EZYWxhv2I
+ SYmViuWMMY2BjtGQI ZFxksWUFZVplr2YEZllmr2cEZ1pnsGgGaFxosmkJaV9ptmoNamRqvGsT
+ a2trw2wbbHNsy20jbXxt 1W4ubodu4G86b5Nv7XBHcKFw+3FWcbByC3JmcsFzHHN4c9N0L3SLd
+ Od1Q3Wgdfx2WXa2dxN3cHfO eCt4iXjneUV5o3oCemB6v3see3173Hw8fJt8+31bfbt+G358ft
+ x/PX+ef/+AYIDCgSOBhYHngkmC q4MOg3CD04Q2hJmE/IVghcOGJ4aLhu+HVIe4iB2IgYjmiUy
+ JsYoWinyK4otIi66MFIx7jOKNSI2v jheOfo7lj02PtZAdkIWQ7pFWkb+SKJKRkvqTZJPNlDeU
+ oZULlXWV4JZKlrWXIJeLl/eYYpjOmTqZ ppoSmn6a65tXm8ScMZyfnQydeZ3nnlWew58xn6CgD
+ 6B9oOyhW6HLojqiqqMao4qj+qRqpNulTKW8 pi6mn6cQp4Kn9KhlqNipSqm8qi+qoqsVq4ir+6
+ xvrOOtVq3Lrj+us68or52wEbCHsPyxcbHnsl2y 07NJs7+0NrSttSS1m7YStom3Abd5t/G4abj
+ huVq50rpLusS7Pru3vDC8qr0kvZ6+Gb6Tvw6/icAE wH/A+sF2wfHCbcLpw2bD4sRfxNzFWcXW
+ xlPG0cdOx8zISsjJyUfJxspFysTLQ8vCzELMwc1BzcHO Qs7Cz0PPw9BE0MbRR9HI0krSzNNO0
+ 9DUU9TW1VjV29Ze1uLXZdfp2G3Y8dl12fraf9sD24jcDtyT 3Rndnt4k3qrfMd+34D7gxeFM4d
+ PiWuLi42rj8uR65QLli+YT5pznJeev6DjowulM6dbqYOrq63Xr /+yK7Rbtoe4s7rjvRO/Q8Fz
+ w6fF18gLyj/Mc86r0N/TF9VP14fZv9v73jPgb+Kr5OfnJ+ln66Pt4 /Aj8mf0p/br+S/7c/25w
+ YXJhAAAAAAADAAAAAmZmAADypwAADVkAABPQAAALA3BhcmEAAAAAAAMA AAACZmYAAPKnAAANW
+ QAAE9AAAAsDcGFyYQAAAAAAAwAAAAJmZgAA8qcAAA1ZAAAT0AAACwN2Y2d0 AAAAAAAAAAEAAQ
+ AAAAAAAAABAAAAAQAAAAAAAAABAAAAAQAAAAAAAAABAABuZGluAAAAAAAAADAA AKPAAABXwAA
+ ASsAAAJ5AAAAlQAAAEwAAAFBAAABUQAACMzMAAjMzAAIzM2Rlc2MAAAAAAAAACkNp bmVtYSBI
+ RAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA AAAAA
+ AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABtbHVjAAAAAAAAAAEAAAAMZW5VUwAAABIAAAAc AE
+ MAaQBuAGUAbQBhACAASABEAABtbW9kAAAAAAAABhAAAJIjAgAqqcBCT4AAAAAAAAAAAAAAAAAA
+ AAAAdGV4dAAAAABDb3B5cmlnaHQgQXBwbGUsIEluYy4sIDIwMTAA/+EAQEV4aWYAAE1NACoAAA
+ AI AAGHaQAEAAAAAQAAABoAAAAAAAKgAgAEAAAAAQAAAoCgAwAEAAAAAQAAAeAAAAAA/9sAQwA
+ CAgIC AgECAgICAgICAwMGBAMDAwMHBQUEBggHCAgIBwgICQoNCwkJDAoICAsPCwwNDg4ODgkL
+ EBEPDhEN Dg4O/9sAQwECAgIDAwMGBAQGDgkICQ4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4OD
+ g4ODg4ODg4O Dg4ODg4ODg4ODg4ODg4O/8AAEQgB4AKAAwEiAAIRAQMRAf/EAB8AAAEFAQEBAQ
+ EBAAAAAAAAAAAB AgMEBQYHCAkKC//EALUQAAIBAwMCBAMFBQQEAAABfQECAwAEEQUSITFBBhN
+ RYQcicRQygZGhCCNC scEVUtHwJDNicoIJChYXGBkaJSYnKCkqNDU2Nzg5OkNERUZHSElKU1RV
+ VldYWVpjZGVmZ2hpanN0 dXZ3eHl6g4SFhoeIiYqSk5SVlpeYmZqio6Slpqeoqaqys7S1tre4u
+ brCw8TFxsfIycrS09TV1tfY 2drh4uPk5ebn6Onq8fLz9PX29/j5+v/EAB8BAAMBAQEBAQEBAQ
+ EAAAAAAAABAgMEBQYHCAkKC//E ALURAAIBAgQEAwQHBQQEAAECdwABAgMRBAUhMQYSQVEHYXE
+ TIjKBCBRCkaGxwQkjM1LwFWJy0QoW JDThJfEXGBkaJicoKSo1Njc4OTpDREVGR0hJSlNUVVZX
+ WFlaY2RlZmdoaWpzdHV2d3h5eoKDhIWG h4iJipKTlJWWl5iZmqKjpKWmp6ipqrKztLW2t7i5u
+ sLDxMXGx8jJytLT1NXW19jZ2uLj5OXm5+jp 6vLz9PX29/j5+v/aAAwDAQACEQMRAD8A/Ie6uZ
+ LZoIYXSSFWZhjkgn1NZS7Y4I4yQS5woHUnP+Na MsLq7qIXfaMN22nrTWmbzIWeOMzBmckLxk+
+ gr8+hZLRH9Z18KqM3KhBK/wB339SvFbQrNGluD50T ESbj96QHjHHT29qeG3hT5bIQ/QnoQc5+
+ lLayrFJuBUqSWb1PsD606OJriMIiFwpLZA6nv9aqTd9T PDQlBuNlyteX5l3yTvExmWMFjGjbS
+ wORknionheOeFHDNHHGVyvBIPpke9SWrJJp7XJjdCj7QrHI DdTx79qhfe8yF43TswJ556CsVe
+ 7TOujOMk5T1b2a0/4P4knkhYYpX+WM5wpPLDtiro+S2AEqMpcP tJ5OOPTjipZGtBYxwLcKzxK
+ FZjkhjnkj0A6VXnaFEa3KjeAfmLHll5OPwrJSctzahOM1zTXlr26f 0hk8iCYCVGcRqVBU4ySe
+ vTp2pYrkRWwhijf/AFo+Y/wDqc+vNVgPtE8QY5LMNuOy46/gakjAdfOa VDGSRu7buw/OrcY2s
+ zOc6DlyzfL2/r/gFuaWzkib93I85kZyScDnv+lRbBFYi6iuVE44ZCvKhuo+ tU3SYNC/lERwhV
+ nXbycnOKfI5iu1yI2Z3JAK5XCn0pKFtEzCbpxg47rzas/LYfbWtotvHbxyOpLk M3ufT8KhY2q
+ zpGqzOrEklTyxHA7dKgaaJpt5+RnLeWoHOewq8lqy6bFNNxuAxn+PP8Q9hVyvF+89 yeWMZKHM
+ 9dkun43IzEtveq8b5IXdDIP4hjmlDKvlvETvRQoOPXmmTpDC6qsglOSAy5wABn9al8pX 2BiI/
+ l3Kw/rS0aTZ2wjGcWra9dOnn1J2kkmuVeY7nmYvImMY/uipmVv7QYbvNlOQcDA9DT2SBZys gE
+ gLsQ2/G4AduDVgNbLIk+/y1VVDRHJZuOSD+VYOVtkd9GEqauruJUt3QXscZieVHbaXQ42rj5jU
+ gtiqQBra4WJUZt2RwW/+tUhVTEv2dgiIjIzH+Jc/zzUKwSN8gcyHesZBORuOTxSb1vsTWu5c83
+ Ze fT0CK2NytuFOCAoYMOpJOP5VVuYLeLWY4nR/JbkgnlsjqPbNaEUUn9pBY5Nq43ynHCsoPFU
+ mE11J GDJHLM5CxqUx26dKqMnzb6HHUjJVZRcvdtuPsZrpLee7eMwCOQ7kdeST0I9qqJMG0Z1U
+ Au0gOQeB k/NT4nJmW3uGZU34I6cY6n2FVot88oQwi3UDa7MOhxnn61ooq7bOCNOFObdR6rrbd
+ FrdNDFJCu4h ZMwZHQdzUxluC5ijmiETHccjOCKbaW1ybR2mjwu7y3DDLKx6CoYU3/ZpJy0aJE
+ yOgX5icnBz071L 5Xc7FHC1YP2cb37atefYltoZbefa0sKJJnJwTkZ+lOhSe1v7a4BDRupywXj
+ nI2/U0yGTyFikJVZj 91X+br+FaG95VjXI/wBWcpj/AD9aU+a77M6amAnUklF+71v0+5Ic+nH7
+ RB5S/Zl+64kOTHxwD79a 6OGOKz0xGRxFLCqou/n5S2effr+dV0s2uAkCy+fEBucr958D7wPoK
+ fLJcqvmQ2zpmUK5lAYFuAhH tjPFeXVqOpaNzzZJSfspPZ+Vv839xZuLO2llZyfMZzJsVTggdS
+ D796ykkjjihZIZVlaMbGJ4bDcM B6ZrQeeSO2l+0OivDdFXkC4BYentg9Kz/wB297AY3zuG2Ln
+ gj2+nWlSUuXXYKUavspKb93p2fr0L Cwtc6oZWmj+3RAxt8vBPVjiqAkjltpc3cCJJIX2Eck5+
+ U5rcmtjLB5UkkeXk3l4/lOPT61j29hbq zxoj4VsBmbIUkH5frmnTnFq7exyuLk71Hr2026a/8
+ Ay/7PimJlW4WUcg7ONp7VlH7WI0AiIWJGXd jtnk/rWtdpcNNbm3QwTR4S6VunqPyNVru5JulJ
+ ZZHYnzggwGbjkDsK9KlOXr+hrTryvda22XbzMq 4tJmuEKkxwqWXLdx1p/2SNba0eeUwPKDIwY
+ ZJIJAIx2qaRNt8n2hxIkjE7o+MnNSfZX+feduyXY5 bqpPQfSt3UdlqOqubXn19PyK8kCKiyS3
+ MbTkEjjAb1xSwRSecs0Ubl0BGBztLcc1ry2dssrm7R2O wl1XA2noO3bvToVaDT51j8uSZ5Rhg
+ Oox1+lZe393+rDlUUYuPK5X6tW+/cyhCfICSyxxXG7ALZww xx09ahgjK2oiDrEHO9lYcjjH8h
+ V0Q3Hn7Agk3bXDAds4yKlubf8As+/iJQXAZHVW7Nxww9uf0q/a K9r6nXSjTU4xb5uy/rVGZHA
+ Yrf7xYMwKsBxjHI+tWHUNbKwR/KVgScEjd9alhVo7dI5M+Wq8ZH+e tRT53feI3HLIBgLVczcj
+ veHlBLmh+bFgsYLeKMl8ySSZCg8px1NSwzz2lzHInD71/h9P/r0yN3ku HVG3SEEhcdcelVgGG
+ wZYZOWbr+P0FJxcr8zuYuh7kqcndPtbT5WNGa7+03Jk2vnYRIw6Z6k1qxXl ta2kA2s/7pmyD9
+ 0HsfcnpXNWl00drJtVJfMcktjtjFSi6aRSoYDAC4C84/8ArVnPD306Iy9gqqip ydlt/wAAvLe
+ 2UlpHbLBLaOxG/wAxs/vAeD9KoXN5IU8yUhJ5Zw7nH3iARx+lN2ESxCSaPC8hthO7 3HGe2Kpm
+ G4nMD3JO4hmjAGMDkf0rSFKCdzjlhnKahZv1/r8CHYgiKMwUkbhk+nUVYxJcQqszIRt+ cKOuD
+ nj6YFPjhdrYZXcqMAOmcH39KU2zGVUE0RILK3BwR3IOOlbOSNZS5VZp66WV/wAfIqBQqBYZ Bh
+ cgMeg3dqgjU21itvHIAryF5UIyRxgc47DmtGG3fjy9jKcn7oIB/KoVhgMNtGyuk4Y5JP3ge9Vz
+ o4sRgnzRm4Wts1/wxLGyyBlk3Pt3AFeAw/hxUi3ziwt1kGXRtkQAxlfX8DVUKxjjEP7xdwDYHQ
+ DP tShVLlQ4TnKp/Fn/AA71DhF7lV6PtUnNv1/y2LNy141xG926uWRmygxnnBNRW8+9nTzRCjF
+ d4cZy vfFSSTb512QuVOflYA5AHHQVGro+2d1j2wZAAQAMzDAz6ikl7trDqKcafs6b/wDbX92o
+ 15GMwC4E THAyONo6dqRY7aOSBo4pZCrBmAPQjmm7oxb2ykFZo1KsWOQ2TRE4K7MFFZwVJ64HX
+ 86trQmUE4Wd /nv+RHOZGYhV3KuD7nnINJCwi/5dJ0yCSSQce9OdpQ2duOPm/OpCkbuzM7uEHB
+ U8PV30sdqT9ro3 fzt+bRRfyxlllUlgMYHfpUaOzop5aUZ3Nnj2GKsSqYn8wxOQBlRxxjHP51U
+ SR1kDKm4nOSBxn/Jr WOqPKrt+2tK6S7X1JXvGjgCyfvS45UDsTTPs6i7kiRAPLyrHOee/8qHT
+ McYuYsbV5IH3v88U64ig Z90TlCARjPfvmmrLYhqrUfNUs1HZPf7yqi+ZIFOYgSCM85wDVyNFL
+ EBxtHK5bt6VT8lpJ1VXwcjy lOeR9anWSObISKUuQRtUe9VPUww1aCk1PRrXV7r8iUw70kYXCg
+ LtzgdyOn1qtHGk1mJVnHmqzOR7 4HFNDq4fZG9uORhzktgf/rqHygVRh+6KtxnvTjF9zPFS9q0
+ 4bLXrr95LGt9IV/eRKVBLHb0TuPr1 qw1zI4kXzV2bsKSOoqKNlAPy/KxYg+uOMVZtI2kMqYjG
+ Yy4LDjCg1M2lq0Vh7r3ud2e+rexBHKWU +XukPm7sg8DHb+daUkMbM2Uk4J8rLDG09KywrySwF
+ nS3TPI2/eXv071LFvGNisMhiwkOfWlKPZmt CX71Jpv8n8tSREkS2FvwIgwLE8jpx/M08WpQEx
+ sWIX5lU4NNijaWBsFiv8BA68frWgIklbyw5UMh ZnXtjms5TsddGjR9nJvRLs9eu1inLKGg/eb
+ ywA4H5imvPMTkElWAx8vbuakVf3JlVMZkAOQG6DIP Paoo3Kyp5hcbpRuKgfdOeOnftQrEzxk1
+ Fzd7eW5RnEk9wjZSELgNuHOMVLC72+/5FkweWHY4/wDr 1OYraR5GxK64BHPTsM/lVa6WMzliG
+ A6nB6mt4yTtFo5VGUE6yer8/wDgWR0NzDKtzEqGRwpwxwQP XB9+aHTY0KSJISI/lAx8oOTz61
+ El6r7Y2lDGVioYEgA+/HepfJDSKiv5jlQWYnjODwP89q4NVa57 6rQdTmVtfWy+8diFLfeqhxt
+ 4PbJ6dqVHk8uOFoyjpG5XaMFgOpqv5DmGN5oXiMZ5wcZ9PypIJriJ ot77pIQyL8vr1BoautCa
+ 1adRrkitNr7E7TPBB5UaFsKAXA+UKcEk/wAs1I6tPcbIgCXOTg8rjnB/ lVRblJIiSRgHaDnrU
+ sckksJkASEliNpGTgd6ORo3pQp83uy37L+vzNK9juJIBcT2wjhdyVUIARk8 /liqUELXNwsfll
+ XIZyS2dzf0yMUhlaS4WNlYnJ3dunNX4ZYvPE/ktHvJJZSAuPb2rF3hGxzV5+x0 STfSysvu/wC
+ CQ+ZNBIj20QjVkztdQSTjBwfQVloI5ImcYIU4JU4y3Y1cG1PKywnjdt5APLYP6CnG ONt6b4kQ
+ kyBVXb8x4A6dquNkVSpxcuZQWu7/AK1M+B7q2X7KXzFJMWm3c845Ge3FTsXi1AGNcsSf LDYJV
+ MdD7+9TvZPDLEZhmZVcMo6Agf8A16bYgjLNLF9oKBF3rwQepH04FW5Ra5kccYJNuO3boIUk vg
+ w8qFHV1KsqAYAXrVwPc/ZYkkClPLPlnbxyen1yMCoRtW6RpWAZNygKcZLcDPtV8xxxXgidyypG
+ RIQM4bnGOOnNYyltobYOnL23M1a22jbX5FORrpWe2WEcnJURgsoC884qSK2d4kkDAITsBJ/i7D
+ 8a sl7gRIU2RKdqNLt5Ax606dY0+0I86zmObKmMYDLtwMD1zUc72R6MvbRqfu1pfXTX9EPSxii
+ nME4k bGWV93+qA+8G9TUc5R4cpaMkZbdvzznHH4e1SmeGWO3ktzMJlUocnOSRj9RmhIJjLbxB
+ dmEypccL z0ask3e8maU1Je/zX8u3ys/1JnNx9ntwyxw7oecxjghsntUgKyXKvPewIjOXGFwdw
+ 6HgdKinmSRp VEbInneZuJ3bTjGD7e1Z7/Z4SHjDuq4G1m5G7t9RSjDmW1mbVKEpU/fVn3snp5
+ XW/wAydFE9/JK8 6Rxl9sjZwCSDg1lMpjjQEPxyDu5HpV2DdLbPabdpVvMXPXj1PeoI2228hIP
+ mAEdM7hnlvoK6IaNn LOkpTbe9vNW83p1IFKjzGZwjhsktzk/lW5pywSwj7QQ+3fFIqcHcT1/C
+ qyfYjGY8CSUuQX7NjnOK nfLmJ1gKOYZDKqjq5Pt+FZVpc6tax5uIpyxElGKs++zGuzCaexBMo
+ 3bQUGOR3oWKI2DCPDO21pSf +WbZIx9O9VYYdluJhKHCuoZQTknB4q99me60yN4X8yRVyFQYJX
+ uT64pStHqei6c6XKtEu67+Y2K3 SV900tuAgYhiv3wDgMParM9my2SzC6tXULtXYuCecfzqtBD
+ G9ypuIplZZQqIMDcCM+noK6KOH7Pe Szq8XltIpEbrnC45H171jWquL0ZvUquDvGV/u1/yKMMN
+ 1azxMtzFG03zcj7pBxt/HNasst4yCBQj G3baAF5YZBP49QKrTOJHjYSRbURgeOpIzkfSnhkaz
+ WWK5V5JkJi6/dHGT9O9cc25WbX4f10OOpyy kpVlr3tp82tSW6EE1v5e8O+M+XnlvmyxPuBUds
+ PL1k3gh3RIAsW0Arz/APXNR2k8rt5DRb5UnOJV XCuCOCPbrSztLZeW/mqwRHUccHBBB/Wps17
+ ncy9rzxdBR+L8fQS5upbbUSkzoyIQ2AuCxX0P1I/K oIpDqdvcTMjIyEoqx4G8sfvfnUpSwYXT
+ zzK1usoUEtk54IGf50lxcSWz+dCY3G2RWVEwMn+orRJa KK17luMI1oRowfP+H4GVqG+P7RF5i
+ J5LxhmP8TYBPP61nS8Xlw8ULOfMbew5HzDqPQA0s7b4YYQD vjTLHOS/vVgRTNHZKh+0bY9s3l
+ 9ixyM/hXfFckVcqvGVGznBJ32X6/8ABK1rYx3tpGEbLNCzKpOS WBxke1MeEQrFGRLGshP2jzD
+ nJ42kelalvttryDzJY8GIlAo25HO7FVkjEbRyRMXAUlt3OHHQUe1k 5Pt/X9epxc85y5VK3a2m
+ vqI00bM0cwIDL5cbt/c759TnnNZCRrMojMpEqNjcCQCfUe2BXRefI1rF LNZ8gFHZkADMeePSs
+ xLfc6K1tIZBtDhBggnOCaqlOyf+ZrSUuTnlt6poIQLQh4tw3gsm9s47gfh/ Wm/apLlvNkgkmX
+ cF+X+DIwR+Oafd2c8W+JQxij27mPc4zkegqrHPcJeBomBmfd8qLxnHp0+lXFKS 5luUuRw56du
+ bv/W3yBpm8+EzJHt2FQMY3EAqPyphWNre1aMOx2ESknOSOc1OJw6mLCTwsrNlV+YB R1B7DOfy
+ qmsrSBBEqqWLKBjO4AcmtYp+h00qsZSUpOz+eo1EM0wliO11IfIHXB/lVpHS8nkkZ0Tc TuCr0
+ P5YxTJBHFEoCSxuQAVJ7cEGpfswM7bWUBHCsw4BJ5z9M0SaeprSgm+ZRs+uln+oxRmLyYzH Fu
+ kHUckjsPwqeSJ7edQI0CliXjIBZCOACa0bW1hlTzDFIH8wsqhumDwPzp32JZHmVRJGWcNIHfJB
+ 6msHWXNqW051GmtF0M1YVmslQQyi6DYBByOegxVEFTkfPmMlFDHpnk1uJauLjz7aGZoSGaIk9N
+ 3A ye+Kz1GYAtrt2jmZ25GeRx6Zq4VEOKjBpL/N/wDAI4EaZZUWGRwSASDwuPWldpRegSRxwsm
+ UbKjO WGM8DoK0vLZ77yYysUjvsUjox/x60kihYZI2ZfNaUm3BGS6fxNn8OM1HtE2YVKkZ1OS/
+ 5XXmRyxM LY28c9lIkRysoj+9gc9u5rF8qdbjdMgZt2WCDGwZ5B44NakqolqwKywxcmFnPLEe/
+ pUDKqR2srF2 Mineu7lZBxg/zq6bsvU56kacIr3t9NLfjYqrLBDc4kXaG37EHHTpmnW5865hia
+ AEODkouCCRgc1M scTQOrgxyrMIzvGSCTyfpTolKSPFGrozNuScnC7VB56d6ttWfc5+Sm4Nyf3
+ 30/ryC0+zlhBNG5Ql dpDfMAMhuf1qGe0iacFAUhZG8s5zkL/F9DVqRoRLbuCJ0jiKDbxtPO4H
+ jqM06O2As0kadCVUDC9W UAk49OtSp2fN3KhGEZqb+F6aa39Oxm29qGuIlwFG3fhurY6D8aI7J
+ pLnzImSBgzbUc5IHbPFXGtT PbJNbF5WwPlXkg4yf0qncvvKJCSY+gI6kdQa1UnJ6MJ041Obkf
+ y/4BXFv5bKHky8ZJAYE8f3SO5N SYkjjjWMKu6PIUg5FSwzCGaS5kiWYyHaoZchgeC34UnS+cS
+ MDHl48hcDbj7w9hVOTvqHtYU3JxVn bp/V/uZlssghVJ2EkpUkOnAx6fhVeWWTciIhcZ+bA6L3
+ q7HdvLAZEaEBSYz8mcj/AOvUjwB7IzGS N5A4UxgEELjmujm5X7yOOUeaFqcnZ9df+D97KE8v7
+ xkSVX3NtVj0bHpVULJAuwsMMTnIzmtjykct EHhyFLA7euBn9azpYj5ce4kblDDn3q6c1sY14O
+ Tb3a+RXkaeZoyVyBzlONoz/iKcYZ3BPCHO4np3 qd/JiLhldweFC1Ahjhk2K2/erKMnP860Um1
+ ojlqUlGTcndvfX7ug2VIphGGkMuwll2tjk9c+tW4Y vMhYBh8rZJBz26VF9mSLT7Zt8UiHPCDB
+ B9+KWCJxs8ncAPv5PXmlKScdGRg1FyUuT1SJDGHk2ylU VQNuxeTnmlW12tF98jblSPTPf8asG
+ Jhcuj7mKnBKjgelapvJpLe3VPs6rtZB+7+9g5z+PQVzzqyV rHdOmlUXu3bfUznjUlhOhV1lCK
+ i8E8ZOP0pZ7URyLHcDEinDkZGD17dulaNtaRbC9zOvLqSmfmQ5 7n+dTXEMPnrsdxIzl1aTlWQ
+ cbunTNYe3tKxq5e+1O935afO/+Rl7pgv7t4oI1m+8yjAY9R0qREba 77CpmkZlz2Vei/nQ00Vt
+ cFFT7SN3Ddsdc89aspELmWLadiJueUsPXgKPc05O2ttDGWHSrKpb3eun 5ozZo7iWaMzZQyP5g
+ TGOvf6VApVQxjjJ3Nkkjg4PUe1aSxzpOjeW6FZdnzgHB9KrbUQsSDGM/Ip5 IBJ4rSMtLG1ClD
+ mfs3p8tBDG+2TEbRHuOuDUE1tsQCSJsYLo2efx9avWscb2SyPMytliw9QOg/Go 3s5nijZSXVQ
+ Bx2B6/lTjNX3DmpV1zKO3Vr8hwsxa+Vb+X+82szHHUjn+Rp0Mkf7sCKWTcAF57k1X GJCsiyyP
+ s4Bzzk9/oKuRguD55VG88YKJtAU8Htg84pS2u2drxCov91H3enW3qKuILh4Rwwcx4zn5 h0P0N
+ QxnbCfOkCS+crDKkhgBgnilkSWOTZhWfqWxwcUwHfPE0iYc/KBjoewI9am10XK7ilzbbdEP Eb
+ GPcDCEX73y+/H86mDBbxVSSFWdiqq6574xVJFLkAKyEA8N3NXI4h9kb7s2clW9MUS03NFzTg4x
+ W6/rqh8cjtG9w4GwuwXjsOM/nU0RcwDJUoGAQkcKCOatgFLTyDENzMNy4yBn+Hp1NQPH5byBV8
+ r5 tzK3X0ArDmudOFjiPdVTp16/cCRqVaQSBpY1JxtwB+GKZEF+/JltykjAAq7bxKw2yRSO2AZ
+ Npxnn oKkEcflwq1tOjMhYjd0yeD9MVDnZtFRTjUd2/XS/5lCJNojdMMPLLEZJyc4x+NaElnY7
+ baMLKrYK ctyG6kfWrUdvBJNFMFEEEbEAuf8AWcHDD29qYGFsSdyNKiqN7cq2T6euM81hKq5PQ
+ 83EU6lad4dO 2n5f8Eo2FgzXzbwBhfmLj7uc8mrcFrHHO8UiSuyx7Btb7xIJ3fnV9JSzXaxMqq
+ kvljIyWUnAOfal jkt5r3EkcskuDsZeA+DgYqJ1ptts7E6nvSinGy/pt6fmOtDbRrHHbwvPcOo
+ ZyxDBCvVSPXrReQKs LxxWbKJvmQ8bmAOBzWeUul1W3iWCSVcsrNEMc9zn2PH4VoPcXtldW7yR
+ 73+ZYlP8Kkc5+hNZSi1J OLu35nL7KSq88Xd+r/zKB0+G3+zuZ/tEkUg3JESCdpyT9Of0qS4Fx
+ cWqzzI8aJuO9OA+48H6cUye bz7iLYzCRR5hC85Zf6Y6itOSLzdPee4BVS22NAcAh8H+daSk4u
+ LluelSw0qVSFSUU9db7/cY6yKt oDNBKT5jBwpALcDJ6fSniNGkSD7K6xjO1jjI6ZJ9eKrJbTT
+ 3CRKQZmIyp/vdMfXirNxHMk1qof5j HLKMevdffgVrKydk9R4vmg1FSu797WMmRoortpEDBVOI
+ wW556Zx1pbaB5YHlkkjk8qUZSMYJGMkV YkkQwruiGXG5D/snGPxFaGnxOlmZBEZ0dyx2d/X+l
+ aTqcsLnHWnKa5pRtfTff8SvC/kyLIjQxM8L ja65xnv/ACqJpWlhkBjZrltgG09Ox/E1It2SYL
+ nckDueY2XJBAwOx4rQlhkklmR0E4wG8yEAfdAI 9KycuV3aNVUSp891213/ACV/vJVkijS4Rts
+ eydMggHqMUxYxa3H+onit5MpG5bgg9R+dUrBZL9Vj YeQxYtM7/wARwSD+VbxtLifTEjVPMR8y
+ Y7gcEL9a5qjVOVm/U8+M6cJW773/ADRAwu54t62xT5i3 QZQ4wAfc4qQXrLoO+6s5VYS8oxG7A
+ x8307U4JJHDClw/Mo3SFeNhB4U+5pt0Ymt7m4hhuEkBIdZH zsyRgfXqax0k0rdTohCNRwUYvf
+ 0/G4ly9u9wXhjeZH3bERudoNZNoXN0yMjOoDMqqcFV6gfh39a0 rnU7FLOaILvZ5FYMnGD6fTi
+ qktyHnit3j3ZDI+zAPJBz9Pat6Sko2sd1KVRx9+DS23ennb/hgmvF GqfbIZ0MbfKoU8Zx0qe3
+ gi8pmuvNmRGCvl/uE8bT7nrWWI0lnWHymOWEhKnA+XI4+tSwec8zR78v PiWRD1Vl6fpWkoaWT
+ tb+v8yMRTlVk9bNL00JLmSG3ln8sLHtZYyj/N1HX61SuHAiSOJZkaNsYds4 9j645zTZZJZZ5G
+ nCrlt2x1+9kcEVIyxy2cM4LjHE565Yenpwa1jHlSudCp8yi5X8rf0iO3hCSs15 kxE8Opxux6H
+ 0qe3ke10+OdyJY2I+WM4O7ORz9K2ATdQRgWrS2rRF1IwNgHG0++ax3hmMiCZY7WIE Id4OMseP
+ xrNVOdvmOJ0/aybqvRb69P8APuMuLiV5YTcSQGRZnkwkeCORgVblZ5ba6khXlZywKL6j PT86W
+ C2tlupYZ7232wvtEjKTuBBzViDy7ZYonu4XiaEMyDruGcc/SpnKK+FbeXzMXRo0rcsL211T e/
+ 3GNMjzW65L2ylvMBckggYGPrzmrVvZpuVJXklaQMAyn7/Pyke1PuLW4khgRTstjbGQOwzu29cf
+ jUTXoSM3USZZcRkHoCw4I/WtOaTjaLIqxqxpr2Oj9dCdhZxRSW4eUMXVWd2yNxXg/QY6Vlrata
+ Sx BicSvlZB1QKCcH3NLJK0izxFMSkneCOm3p+OKrO0kiS+Y7NGzq+M9DjitadOSW5UcLNpcrs
+ 3vcWd 43uifOjiHllAuMdV9h602PdbWFsgik2BS27HO4n1pscoLtsWN15Xdtz+PIqaKCN5UVhI
+ wPLKD0Pr 9K2doqzOicFTiqmjXp/X3jVndoyskamQ42uUyWXByRx0FXQ7xg7fuugVNycMT3Hqa
+ q2bBp4w8Duw RxheORyD9ParttNvKbpYoV4Rd4zgnms6mmyN4zVKm1ZW31T/ACLtttggZQDdN5
+ 5QtGcYHGPxp9xD Ld3riNDahGJZnPQAdD75/nVRkQ5kil+RZCWjz8xx1/Si2uJWlLgtHbysXO8
+ 5JA4PNc3K78y3HRjF XlGV/v8AwRfSW4tYFkKfIZFj24+4CCSakCRQvLJK8NwrcKsa7cjsaZNe
+ faLVXe3fyVdsHONw29fz xinWcMQS28yeOFBGpbzed3JJxWT+G70NVFSpubXrotV2Wl7kE8UbJ
+ bhJogxYs3tjk/jWbc4F0rvE YoJmYoSM4A6jNaE0LER3EAZvNHmtJwVB3YwKzpXnN1biVlABBG
+ 5eAGPP61vR8mKlF04t0no/v/D/ AIJbmu3jVRHGkkboz/MARzxx6VmWYW1uobggSxpzIeoLNxj
+ nvin3EKeepEu7HIweOPb0J7VJHujt Y55SBM6syIF4PbOOnrWiilCy6nFiMDCUbKHxfL7+q9SO
+ Wa3MyKqM8bI24Dgk5+XJ9afBbO8byS3N vACQ4BzlgDxj27VJbi0EEZO5H+z7SDnJPI3Z9Paqp
+ aWRkhDBV+X5m7EDHp/nNUr7LQinQai7tqK7 q/5ofPIkjj5GQzHz3YdAewwB0pkJuJJGtzJGq4
+ BGV64BPHFX/IWXUCZImjhcuzRg4aIgYwT7dadH BFbtaTupEUkBIlBOOuMn6modSKjYxq1MOo8
+ kdX0/q1yhNPch42lUQu43rsXbjPGeO2KS1ma21pXV YbhWOxiEGD2B56cVchghclppGjBIUM2T
+ 8g+8PqT0qNp7Z2eFYtkLHzowPvDH8JPfvT5k04pDqqMq UowpuzWr2+6+7KbkLem3FuyKXH2dm
+ wcIMk/XNMktLq5d5ZlW3t5SSHcYAHccVoMsFzcREsyokLBC OPlJ6/zqt9octDGq4VY2UFjkMS
+ fl4qoyl0WpwRoVXdKNlbqv6X5lX7DbWV1Bu8uRWi37kHynAOeK ijkY2ashtk3MA6NHkn1q2Fm
+ iKPcwbT9x+ANue5+nFUp9n2sx+YGXcQJVGAcDrW0W5PV3G8LHnfJK +nay/wAiG4jS0uh5DLNb
+ fOiPjPbIyfxqmsiGHe0bsVAR+RgN+XpV0pF9iRFjYmRAyHcBwCQfx460 x4UVy4CoAwkwec4re
+ LVrMKeGrqnZO6XyRXeGH96RMGBm2x4GT04H4/0qiI4hp7+aU84TqyHB+YYw fpg1dVF8yFzllQ
+ ZIHBPOalMa/ahmFmLI+fqe9aqbiclfB1JQbntHW/f16fkZkRCiIRE7Sh68jrkm pLZJJt0bzxx
+ sCWBI4Ix0+uasMR5sEMZSJCn7xmHK56j86cUnhvCjLG4VsEoNvaqlK5goR9ooRbT8 l/wdSdXl
+ +x7/ACnhm37SsmDle/Hr05q7bWjFyTE/l5Ii56joaqWMc00u1tsKLJ8zscgAjgfU461t W1xFH
+ YC3jVpy25vLJ+YZGOD7d64603HRHXepCCs25d30XfyEFxPbrFAPIjiKjLvHuJ/rUy3EYhlL we
+ cUxGj5wFwM7envmp5tPmexEm1vPQLCseMlivVvpioV024W4khCPKMBl29/b6/4Vyc1KSvc7I08
+ O6ammpd9bfcKytNZmRbeABJAOIxl1xlselI1tcNpweOHKSMXJUY2gfdB+lVIrm9hlPAYFhlNvc
+ 5A q0t3cnfD5oCGZcE9MHjP0puM47WMJYerSkpU0rLXr+WpmzWrRXcQEgO+RSVHJJA61DIWm8r
+ DRhWT Ifb05/xq/cLAmqrcCYqYZwqjOQwYcEVmxb0CJcIxw20qDgqBnNdMJNq50Rr1IzlHldml
+ stf8iqlr FHbDe0jyIxKhScEHv+FXbVCjw3QYtFHG3G772Ov86r2sbyPCkTAOww5c8dTz06dK0
+ ZbWQsIVt5pG SIquw8A9wfU1pVnrZsxn7Hk5NEuqf+ZmrFiRY4AQuCXyMnHer0i5ulitz54ky4
+ CjJUYzg+pqOWZ0 fypRFHKVwy7cFjknjHTjFTo0wthdTR+fHIefKAU+gYH0HpUyk9zor15U2nH
+ 7v+B/lYosS37wMVXh T6564qxPsnufM3BXaQnZ0wVGcj2qEmN5dkYMg35ZwcA46HFXkgWRSYla
+ 6lYCTKDhSeCKJStZm8W5 JSk7K+t9iASpHeCbiXcCxwf4j0q3b/Z4woLuWYlXRV5z6gkYxUyQw
+ vMYGi8s7dysewHU1ONOhawt Z3k8xtzPiPgt789qwnUhszSdoVUotpvt/nqWRa+fdokaSGFYjt
+ bPLvjg5qt5ZXTp5HurdyZAsigZ bd+XFaC+ZFe2rElIS2+MZ67Rjd9PWqmTc3DbEAj3h5ZgPkL
+ Lnbx79K5oyd/I6ZVZqs/eVl2/zf6E cBuIzEBC5lAV3XuoU5z+VaNwGup3nWeJpGLNEEGPkBya
+ maZPtab2Te4cttX/AFfy8qfx6VWkDTab GwuIpHjiSIBE27cnv7mo5uaSdrDoRU5+05Xrpqrp/
+ gkMjgDOkiSQ4Me5sDoD2qBZp4oZETyywkCb SgY8DIrVYWK3lxHuMKCQLhj904x+XWksYrO3WQ
+ efHclGyNvYYPB9z2pOro21+BWKxk+b3o38mv8A gfqVI5pDf77qaKOLyysi7QCWYZA4HXOKhUb
+ bS3EJRpFQqx287geRn8aFjtpLfaI5nkfawTdkrg42 n3pLdZf7Ut4A8QdnLkEfcGeVPvWmiuwl
+ WjD95JWXbYm868ljg25LoqeWijBbrg+/vVeYyXFgZpLh ZJEm3LjPzgDqPYGtGa5kN3tAS3ihl
+ LAleYgBwGPfJ6VWia3lkaNgBdyzeai4yBx93HvUxl1t/X9f 1sc8KylO7gl/XdrYRbea8ZmXy3
+ dnDZVcZ9SPYirElpI9zPbDd8qlzluM8FcVXGousKskP2aRiFO4 cBTxVkCO1u7ppneVi4SMoSN
+ ydz+FJ86f5HpUZSbbS9F/W5Xulijt0Rfmnum83cp+6y9vx5xWWl4X hJBH+sznuOxH5VrXIdrp
+ BbxO8ZbIfr+I9KorOXhUpBHAoZ1eMqMtwMHOPXP5VrT+HVXJ9k5Plcb/ AC1CztlfUBEWKQ5Yu
+ SOVUcitezjlRdkcTxxSDc8h+7uHQj0HtVG2aOQB7iKSRiQp8ttuT2P09aZP PLatJahZfJLqW3
+ NnGPQ+45qailN8o8TRdWTj0SW9/wAFfX56Gnd21rCYhK0c7spYqnGQOSRUD2sR kSaKK4ZZUDC
+ MSc9CDz7danZ7K7WRo3y7ThQuecY6j0FRW0Zee1iZnclBllYjGScj8MVzxclHVs8h OolJp7fL
+ 8CS3keA28cKpKoUHIHbsf51pyTyXEf7wtFG0hwqfL9Bx6077IZ5GiVT5bEspXgsp7j2F Qm1Js
+ 4Wk3PCV2OQfXowz71zylCTv1Ob2dPmVRuz9LsqwRPE0Jud1urxMVaVsg7Tiq7yTTTIZZEb+ Jw
+ owCemPwrRFpKkCR3SSXHlSbZcHgDjP07VJcJFKyiGNvJBYNg8s7Hgj2q1VXNc6PauT1fz0sUpI
+ 55A0VrBBKBIfnMYO3GMZPr1ouLeQx3O+DzFSRQ5jUAn3FWU0poneRJZYvKVo5kLnl+uar2yoE8
+ wX gZ1QxyR5J3l+jD2FOM1unt5GuHc4vmUrpdtPxK0lvLJAq2s8AVVwqkfMFA3Ek49aaIJbiyt
+ naWJb jDKSOM7uc/0p8HnWt5aoJFCMuZWcZCODyD9c1anJniMUqCNDIxhIABJxx+FaOTTSHLm+
+ sR5tt7+v y1+8zLSwWUW0ckUjyNESvzdgTk/hU8kUK3EkYAjfysBmOVyR6Y6mrP2NURRcTlZUT
+ CsrEDb/ABVk 3apJFHIgZVX5QSc7we9aRk5y3PXwik5Pm1vtuv1sWDcPAYltn+7DtcY4ORnI+l
+ NWbzdQsjdobiIo wQJxuJHDfnVY5t7oeW3mqA4YnkEAVJFHvWLfLGSsY2Y4IXOf0rRwja4sRha
+ c73VvO2v4ECRRrbJE 0qK4ABDDnOSTz9KtzLasqeTnzZZPMXLZEZA4U/UZNO+zRuiXcJQxrPsc
+ MMnpgn6c1QjJ+1IwSSSJ GYqQCASM4Ge/pRfm1T2PMrynOFqU9F6flpcspLNLexwREh/LZEjbn
+ aoOTn8Kbv0+2tmWVtrSTAWw Y5JGM/j9e1UFb/SbabftZlLMcnII6A0x1aa4gLI1xDFExDKMYP
+ OOSK1dJXtfQxxmHnN8t/d06fkN 2oXSRkcjAJAOCRnnJqKYASptGwIpXkfe96crPJDEpI2rEck
+ d25xUs7eYib0L74wcJ25/xrdaPU9K Eabp2m7P+vuIkSXzG2hQ23cPlGOnIx71YRg0EnnREfON
+ p9BjoapwFn8xmjliZSVYk9M9varMtzMU ghVMrIQM7fTvRJO5FoJxqJadO463kmiYTKECsD8+w
+ cHHAzSfafNtcCFWllbexAA2npxTB5scLjIK LJ6ZAB/rV5/tEVuDFFEsYLbiEHBGOPxrOVr3tq
+ czd6ilKKTv94uniJtTEbqYxjBU9TgZzmtWJpks 7dBp8kplXOQAAOTuFUreS8lkTbAJG8t1d1j
+ 6hhwfwrW+zXIs83TMYfKby9nBU4wM1x15Lm1NK8m6 nv2T7f8ADMryNO8u0252LuCxgcsRgKfp
+ mqLzyS3G4oowAJAR1Oecela0XkLHZKI7g3MSYK7+SoPJ /DrU8ULTSIoSMwYZiwXnIGefr1FZq
+ rGO6NIYunST54fozPupC2kSrJEYsTD92ODnr+AHpVCKVDcX TmLzYyQc+4GanlYOiuFJCKeG5z
+ u/nVeSRvs7wFfLEeA7Yxg4zg/lW1OK5bHTD2UIJOOj316FdTHM lu9wdptlAQKMbwxOSfpVe4l
+ ZnUGUbI49qjHJGeoojbzwH6yMhby06jB5/TmmxTxSFYpo82zOozxu Az1z2HrXWo2bfY5Jyo0V
+ KUE35ak0MkeZGby92Rg7Rhh3A9OKc8quXkSEDzNpiCjHA4z/AJ71TeOb MqymPMbkbQMZIPX9a
+ lku3jRFhhA6bTjI2j72KXJd3RxpSlH2jTafT/h/+CSyXRQb5YpYpJZc89Gy MMfoKcbSeQJbxu
+ ziJBGnPDLn7w9qh27L6QIfOCEomRn6Ef55q2i3l9GSWX92WVSo2gk84NQ/d1Wh MKMoPm0Ue/b
+ 5Fh4YVtnt1kEjeb5ipnJAHUZ/Ws6QK1zFDAylI0ZY36A7j+vpTUed7UxbWWRCAMDJ I60p+zyw
+ l1byWMi+XlumSTj9KcYuO7uEYypx7t7Pt52H2kUlsJUkBjSRWBLDow4xn8c1DOY5LKxi t4XZ1
+ ADuOpOcA59K0CZLQyCQNcoTuGOrMRyR7Cq0lvC0DCJiZkZUCdd64+99KUZXldhGc5zip6Jd tV
+ 8yocbRJMs7gMyStv43nv8AhxxTbiS0MUdtFGTKmUZ853txjAxViaAWdvGVYOrSkgEcYA6H3Pr7
+ VVaO6BVjEwkLgYAAJYd/yrWDT1uc1OKac29Lvq7fdchMLiKQLbywFWKhJDk8nlen41XKFrtdrR
+ gv NhMjjZj+uKs36m4lS7WUu0U+0qODjHB/SqyRRC3hVgytt3Id3Pet4bXOaeGqV17OOke+uv3
+ lkJFA HABYNkgdwPSqIQP5aDckRXLORyuD3OKvgqLXauzcQPnY/mKpyJKbqWPOIweo6DvVQO+v
+ QlKiotNJ bWtcj/dHa3mLkqdpwPWljVVijaR2YFmO3dlgSO/HeqBNulvEqo0pYnc2cbD2zVm0W
+ JI3nfFwqnYN vTd1/lWso6XPKVeFSpF8vvL+t2vyL4hHzS+asfQOGH8eOFq7EBbiWRJbeVklVN
+ q53EdTjj8KoskE lmrBXjYjeGZ8hsHGcU44F0G3M4znA7DOT+Irnacla500ZTqQbk3u9NNf68j
+ X8uNrm3X7aUadZHjj YklPVT7miZpreFEX7REduSS2SrdMfX/Gkjeymvo/OkYbpDJE+cYHdavy
+ XFvLYi6f/SVdi8oQ4Ikz wPpjtXE5NSSauZ08RKNRRnG66K36r9SigvnhEks0LNJltnlgMdnHY
+ e9OvDNDfwybUysbKPl+7xgZ 9acLuSNoRAFGVPmbhnPzc49KbJEkt6R5NxHAJP3bl8/L0z78mm
+ rqV2tA53CV6isu3l6f5alEcm2R 4WaR4/nPoT0/HHSltZrWJth5ZiW8x+QrjOFP1ovnBZYY4mM
+ inAkLcEDio2mFuohlEZAY4UKNwKnl T7nsa3tzR9SMdVahZ3XlbX+vuIrOTa8dysTFEkCgD1OT
+ ileSG4VnzLFIzcgnqTx+n9arW81tLdLP skSLzGZ1EmQrngHoOgqKP9/CgK/Z1TO7J5fPQ+1bO
+ HvXegQU6tS8lq1pZrbz6GvFFE0cklyqtLIN 0WTycZyelJ9ilktoGIKLnCNuwqg9c/U9Kmjl+0
+ QPLJsklLYAjGAqkY6e3WplsPOYQxSbmbHlyk/I VHBPTPJrndSz1djtlXjOKU2k46/CvzKJtTb
+ XEiuMXCOscfHGW7EeuM0+FLWS/aRRKlsofG1iMEDG M/Wp4o5IL9YiDcoW2xrjlmAJU59qo+TM
+ kFukkBgm2sJMjAbrjA7VSfN1HQlKVXlet/67r9S1GzND HMCUeNAkasMkoc5PvV+2hU2tlFIk5
+ dly5D8dSQoB7kVnpHJJLBGIGeRY8bB/F74q9GYWvkdUaJwf vGQlcnoMY49qzqbaHozpTjsrSX
+ W9vvS1/M6GSa3ZYEdPs6eWzs7444+UD60RLCliZZQqhx80fcn/ AOuMmqFxcWElmu6XzH5ZlUH
+ tjFU2aP7P5zQXEbM52s8ny4A5HH1rz40W11Rw4ag5zV002/k/vZam J/syVxEAIpcwuy5BRgMA
+ +tTobUKGkCnJLOkfy+UQMBD6n0rLaVZpPMtnAkPDRMchmA+8B0wBUwu3 ijlNzBujdgVZV28Fc
+ YJ7n3rZ03ax2zg+XkW67Wu/vf5MkdZFtN0sahAVZWK8yEcbvoKczxPpbFxs Mh82EoNpKrwc46
+ 81nm+jMIhmk8mEYVA2SSepUH16VOpguppNolKgiNFB5j4+6feqdNpXaMnCjTs5 PVav+r7jUuT
+ au/kRlmLqised28ZJH17VCt5NOiReULhVkCymMYZjzjBxxU1zJbx2Qs4cxRRzhiXO 53wMBgfQ
+ dxT7SQwSCQxh4mV2aVFwsjAEAj0Har05XLl1CU5Onfk19df69BzXIZZo1jZAXy7NzgqM jP1qB
+ Im8tp/9ar4kV0HcenHQZxVa0ZLiNVtw80+zDqB1cHOfoB1+tXfsssUzfvY9oYOQM8LnJ/wp 2U
+ XY3w8ox0ST+X+T/Mkjltb54zJMsMxZvMkIyvboParvliaWVYVF6SwkRVHKY6g8d/xpdOtIbbz5
+ ZXgD7225HAPUDFWItQSI2/CSSzQ7n8lMFuTkj0xXLOer5NbG6dveUW/L/g7/AIg63LT75k+xwy
+ Rl 2yvbODiqKafFbS/eS6tG3eW45ywwRzirE+2GysRHdLcElxIxJPDHr9Paop1+zRQxSSC5hOW
+ ZI+CC Bwc9hSg3bTr+hVOcqkE1Gzbta35dn53K0QzayS3dpLbSSTCTOMDaOoAHTNOlNzNbQyIy
+ PG43QIEB cgHvxzj60599wYlKSsVQbjnjI54+ucVatoWbTVmhdbhxIFaBOGUnrjsBWkpKOrMMR
+ VVCC9o7u+z1 X/AK8QK5lcRw3KSqjnb8uepbA6VrQ2aILqe3LEzsZEJPRB1x+dY8KzxzRTMVtk
+ jUoWlGQ/PLf0rc VVluPLWZZ1IwFjODj1Ht2+tc9dtbM893jUdRS0W9rteny9RcC2tIlfzSzQg
+ xtuxgZ4H54zSuYRE4 a5T7SsgQJk4PGTx9avGWC0y8LKYdshIk+YrnGOvY9qzphfNBb3cDW8gl
+ iZjGIvmVhwQT61zQbk+x eHqutNRS18+v3iR284h81JCzyjzPmOcBf4ceuajSNXggmlmCOFDOq
+ 5G3JyeParP2KT7Ku6R3ZIvl CkjgY/nmrMkyqkkRCK+/93Ht+Ypx19ap1OxvKpUWy2/rzKSTxu
+ ZE/eujtlpFbhyTxj6in3NrIoI2 R28Q3I7sv3WYgKD9R0rWSSC2nkZ7JsKzonTGD/8AX4FVZp5
+ 005V8n55CDscAlNg7/Q4rJVG5LlRn GpUdRcqt266nPGG2+2tbtdKDCGjZiSdxwTuHsO9GZoYL
+ FXkRkcI8cuOFAznP1qY2sBtt5mElxMFk lAODnODj6iqk9vsD+W/+iLMIclskZ53fQV6EZKWlz
+ 2pRhJcs5aro1+Rblle5V1itnYvKF+U/dX+I fWmynT48JIjARM2Bu6jI5/Kp9rCSdzcRSQ+cpP
+ lcc9x7cVRP2qRxNtWGFV8phIuTkkkfj0qYJGbi pPlUvdXqtf68iu09ldmZYfkgDERjv07n271
+ Elq/2dZdwdfJ4IGBx1FaVvYQPq1pCwGDGQ+3gEjOT VieCW2t5Y2ddvBicD5dg6jHrWrrRTUYs
+ 3hiIUpKlT95rX7+3mYlnPHBJCBGZIirRsoP3i3THv/hU l40IsIoIyDtx0OOB/U1cRIDI00iNJ
+ iNgI4yFOOzfgOaoMIfsX7u2mXDKvmO2Qxz249KtNSnexk8P TlX5nF/kr+jf5FNba2k05RG0xu
+ dwZFBz8oOCKlKFbu48wbRvwUHGCeg4qyzrCiDyg6OCTEgAeNc9 z+tUrqGRYxKQ8WJQAGH3gRk
+ NxW0ZXerFRcIcyau+m/53/T5kSCORLYQ5FwM4GcjaOp/DmqbMHkjV B8zKc8Z3YJ5FW2VwyyKB
+ kjAK4AUNkEGpjCsZjVA4kH7tDj7pJ+nvWqkkzKOHkpTnzrRdX/V/Qpu7 CYrEpJbqCOMcYP161
+ IFltpxLFNCzLvAyM5HQnB9c1OHhkihtMjcCxMv949KiQPIYklgk2qjN8pAP HWlzdzk9tKpU5Z
+ K3l/w7/wAiXzJTakzwmNFHlxqy8kkZ59TVyWK4mhgkkyFiUqwTAJcjg/j/AEqu giaWykWG4eK
+ SBiPnz83IzWhYi4iEG2WMIsbb/MXIbGcEficVhUlZXQ4zStZXaelv+A2MSG4WKN5x IGfBGw4D
+ Kv3jQ7xxXwkVLs2bSfIXc4P9DzWtAwEdrJdyxSvEhiUJwDuHP+FMkmE2nLZxxH92/lbC MsnOS
+ T+Arm9q3LVBh8TUlV5asb33euxT8m8EE7tguW4bpgnqv5Vntc3ioVVWR3cbmBwMjgCtSFd+ po
+ QXVjbsyBjxgevqfeqS3EUC/ZmI3MQwc8jI6D861g99LnfKNvdSWnlb7xN1xPcKkieWrg+awXjd
+ nH4AVmY2xlZFke6kbEW3ptHUmthrgs82UYTuMMOMAngjFQyW8M0zRRxyRSLOq4c8qP7v145rSE
+ +X dWMKuIdLr+GnzM5FJtIdqsswT5COAVLYz/n1ojhjOnzRxyQNmTJGPmUjoM+laRithfOlncR
+ kM25N xztxzj+dOmFnHKZ12t03qhxlT/F+FP2t3oeXObcly/l/n0MiK2JlxIWZv+WhU981sQW8
+ P2gSCSOF 9sikOM4UEc/TNU4GtVeQRbxznc7E7mydmK0bS3lFqs8zxowOWVhzycE/QdxUV5vq7
+ GOMrVeXlba8 rb/mTRxQT2StIV8/cdxXjJPRh6DA6VjrC80aLbxyywuAyFT0Az19T1Oa3IQwin
+ id4UMMwiA2+vQ/ lUkUQtL53hdXBXb5S9ieN30A5rnjVcb2ObDV504Slzarvf8Ay0MKS1X7L5s
+ iSQlFKoCceav94e1M t4hJ5DRvEspiIbcuQG/h/Tiumh+0rai3eKOchRl9uQ2OGIHYYrJs5Ekl
+ VAUWOIFUbbjKnOOfXNaR rycX5FrFVZU27Nta3vp9zRHayTmBbe7CyMs6NtAAZB3BqvcOm+a43
+ KjSox2Y5ByBj8OtWTAQkW+K V2f5iVOMDoQffNZsVrdjWIZnhaCJUYnzOR0PP8q0hy3cjOlUVF
+ udt/Sz/r0K8xMcbxorg798xfnD EYIHp61EFWOWErKTcZIB7HsCMirdhaXV5BAgG1Sh3O4yCSe
+ /4VpNp8SXF2qIT5NxGQp6njgA9vet 3WjH3X/XQ7VXpezcUtd7Xute9+nkcrBbiaaG33GBXbd8
+ /Odp5z+tXbpITqNxJ8oCy4iRBjC4/pXQ XFoilfNjadt26UwjbtJ6AemaomLakl3JbneGO1Wx0
+ BwQfehYnmdzmwlV+2UpL0WhhNG8l7Dl0jjY cs4yvHJNOu8C4HlkbJAWDHoR2NaTWbSSy7djbC
+ qsRxyTxj6jis6a0uA9xGbeYMk4TaRkpz90+9bx qRb3Omo2q0pup8tLFK3tlkhjnLQywxkiTYu
+ OcVGJhEYkiURKPmCvzub8qsMpS6kjdNpUlWXbj5um KesYQrJNE02wFVAAwMD5gffmt+bW71PP
+ +r+z95Pmb/ruRxXM/wBsjlgMTMqlMFMqAe2PWrtmZBIz Twqux/njdASRjBqg6qpDoQGDYDKMA
+ jtx61VkeO4uEKzDOeSWzlhSdNS6EVKLnJzmkm/vN+Ke3MjP NaSiBcquGxtOOn40sU32p3jZfs
+ 3ntuDEfKpVTx+NY8UUwgdDcIq53iNsknHXt6UxvMW4jaKZDEWD lcenb61HsU27McY1FTk4r3v
+ np8n/AJGr5scqwzufvxElAcfN93P046U0eU1hFIrTLIpKMxYlQCM9 KpzRzSSlslNzl3UJjA74
+ 9BioxLGttNHvXLykxtnqOAKPZ3WhtWjJxip6d3/X+Qj+btj3FZFcH5mJ xxznp+FNQN9n2idEw
+ /mAMcnd9aTyn+VA6yMUJXC4GM/1p25IzIGUPwMAKMj1GfrW/Qv4qvLPX1e3 3f5j4lX7PIjsFI
+ +YHsVHJH1z3ptuPJkjmmYSRfdcDtnlQfrVYqk88Kq+QB7/AHh0H4014pXVCySh nYn7uACDiny
+ 30b3POxMruUYz+9/18jp3kWaQ/uQVlQunlrjgde3rWlayzskUslxAjMykJswSB6eg IqtLHHK9
+ vktE0RaMoAQV3cih0t0tmWfzEkldCDnG04PFeVPllGx6VdqtS5Ze75dfxJkETass0roY mV3XZ
+ kEgdKyrhrhdhIJBHO4Z5z2+uanlh8uKOMSNiFSu7P8AF1P+FQMJpLZJ4Y3eNADKSc4brW1N Ja
+ nqWi6XPUfvPRfIdsdpynmbHXKBCuSR3H1q9A1zBC2yIRr56nzZFyAx+6DkemaoiOfyY7mRQjOr
+ FVz8ynoM/XNWIA94nlSloZhIBMjnqQPvD09KKjutdjDFV6dSmrbLrq/vWxoTrDBDgx7xvZMgYK
+ rk bgff09qdJa2ksRgtGkdDKzJls5GB830ArLuiJSpy6w7Mx7j6kA5Pc1UlX/iYeTE2ZtxETBs
+ Ar+VZ wpNpPm1M40aipKftPhd/I27RYY7RRJeWse+YOx2+33fxFOknnmMETp9ms2jMmXTOecVi
+ JDGzhWmK oZBtAPL89R7dqvTSlNTkbdhHVwFbnZjjbSlSXNfcy9lBt8z5uv8Aw3n6oqIr5VmKM
+ ypuUFOw789a dHalfKMTu5MiuGTPK5wfxz3p4SP7AjXKSlAzKhVsZA4qRUCusdufs6Bm3PK2QC
+ P4a1c9NDrjiadS m7QSW1/6epYuZo2S980pNtIUYGCSTnIPYDGKyreIJeJCvmnLsGXd0P8A9f8
+ ApVhJlurdZLiLaytl BnGFB6H1Oe9OAtpNRaZo5WJjYqySEAnHXpSinFNWJWHrSpX5bX22X4bl
+ WNZliVJZ7dJI5mjGxMbg er8DnkY9a0Vim2F5C8sgmwdvRsr0/ACqkMP+grMkQniU4Y4ycn39j
+ WqlmWkRWDJIyu0gPQkHAYeg 5pVJpdTWhenFKUv1/KxtRW9td6OptldoU4J3ZJ29TUEbyS2Bhi
+ tDCrsCrsBlcD196itre7+zkNHJ FEiBNq8ZOcdvzPrU8kMcUbeRFchoWKySM+V3g+noc15+ibV
+ 76nbQahNJT5ne68isZ5Xs4RHCIwRg hhn5egPTtzUjqs9vEk+54oodsUsZxnnjOeuenNW72QDS
+ kSRV4DIrhQOMjp9c1T3I1hDE1vMdn7p1 U4KkHdz9P61UZXiml1PRjiYSoxaWrbX9aXJ2s3WGd
+ mmEN4swxGTzgDmmxTyfZ5P9GkRJJUYEcc9j 9Kie4me/kliUsJJo3IbkqT1X8RUV0MEpLDO1vG
+ +yMqSBg9BnoTSUW9JHn4mPM7TSdtbu1/0uWzbM YLVrhiiuT5ise+7GParwjtbeK8tkSa4/eM5
+ eJsFsEYCnqOKy7V76MbPIllPyZZuQmeD/AJ9auvDL l7dbG68hA2FVhv4Hc1lNO9mzyXRl7ZRb
+ 09bfk0XYY1kndyrWSzoWQXHzcD/OPxqxaTPPLLLG0TSM p3QquPLOMH/GqkTWsVis12ZI2mIZQ
+ 7Z2jGDj271E8lsmnLcRCSRVYKGRsbgep9655Rcm18v6/wCH NIRlJtX301S/Pf8AE25ZrhHjLy
+ ReSkflEeXyCRgkn071UgLSXiFZ7e42RnYyJ26H8T2qH5o79ZIp cIVLBJMtkqeP0Jq1CWMjfab
+ SZUT5AUO3JJ4P0HesuVRiarDuCklZ6b6L8P8AJkl5cTWkFvtMe2OB hEWGdxyMk+vXFZsswXT3
+ BYyTp/AD8xQnJb6DvWxcQkaYzNDJhm3BmPygAjcPxqjPeCYrHHCsscpY 741ALAfxA+nalRaaV
+ kOjOm1GSXXfb+vvI4HRdTu2u1WO389Qvy4OcZUZ9Kx3u72K/QosbGYlinlg hevBrVMv76DzZE
+ jQJudnHAboD+VdT4i8Q3/jHxZFf65FpkDWkEVsiWFqtuAiLtUMF6uepbvWvtHC fw3TWvy2sut
+ /U6pXpzc3FO6/q3Q5ZYYpkg3zxC5dAzBeFceoFRm6d52QGI4beFC9SOp/CiSTN3bC GeBh5Xyq
+ F+bAyM59M1lXE9wCkV7GHSSMuTF8pY9Mj29q1hTcnqKnH2tS1r+vT00RYvPtKRxRTKQX OfNQb
+ Qy9Tj9Kou58yYxszTAmXy2+bb/eHPGO9RwSzRs7NBJOGkOHHRQvWrEV3K6TsQsgKbnVUAJY 8D
+ nHT1FdcYOPY9DD0eXmcEml1/q41ne4t7jyImn2sE3x8A7un4cVUihvVgkjXC+SwCRtyWz/ABD2
+ FRJNtj2tIYycq2DjeR0I9u1NEi3VmrfaPNmGAMKRsAzkHjk1uoNLyM6vNLRNfd/X6DSN0kj7wF
+ En lgs2cf5FV/LfyJICSqGUMDJnjAOF+prWjb7bvgMPkOGErsehIHyjpxmobYeZdxy3qhJGd2Z
+ TwMBf T2NV7Syd+hhXxajG01drorFQrAzRqqSR7o2KKxyeT096favJaR2+4qrYZmMq5Bx/Dj19
+ 6cwEssrN Iql3jKEcAEjoPSrMk0T2MisFndZMsmOVPcZ/CiTurWuY1o3tTjC73+/8DOMwMTK6R
+ 75QpUouGDZ6 UjtAjHG4uEIY5wc5ois0KxP52+Jst5YPz4ByRz3xVi2U/bk/1ckksLMAy5GTn+
+ VaOUVex20MQkpK Mdkt1/X5BbRPFYyyfIdj+WqkZ4I5I/OtGW38rT4EM8TQZDK654H8Q/HikUN
+ NeQ/aIZJ4kQKwh+X5 m4H5ir0KhLQ2UUbW0gJJW4+Yrg5rkqVHf+v6/A46mLrJRpu1k79NP1/A
+ pz20K2MhR3Yh94TdkqM9 D/OrFxaSqUEUcgdonkJz6c8+9Mis4LnJM26V1ZwVzjGM8ipYo5PtE
+ RacwMSI1aQkgKycnFZuduuw 1VlCEWp6pt9fyX+QqW8k9jbF5FWCaIOnHzAjgjOO9VntIAj7YJ
+ jJJJGQWIJGc8D8qswGaJbYSq0s PkeZGUGPlQnj8aVryZmjOwQQvDvLuOxPUVKlO+g3iK7Tell
+ 20X3LVsUrFcakIyyRyRMqoD1MZHzH 6jjmnXJtZroQNbS27RyHZuf5mAXqfWoY3ni1aQCNLiJZ
+ FQyIv3jj5QD71cYvaaotzdYTcjK+R0cE dPzrN6NWfTQ4akFzys29LpJ316kUEUD2UefJWZURW
+ ULyCW6H3NVTbPDeX108JeIuUVcdeOMe1aSL Msk5a3Z9rEZUj5eOAffNZ8mmKloUYXMLljuWR8
+ 5I5/rThNXd3uc9GvGDak7J6ar/AIYZDJCIgJrV R86bNqgEjPzfrWxPBFdWoCoVMgxkHqWOAw9
+ sDpWRAbgXkm6LzJMqWCjox4WtG2tLpZwv2e4gVNzT lzkK2CcD04qa2jvc5selTmnB3fS3+TGS
+ abc2czxwSpJlXZyR95h8u76YzTbe4RYLQ/ZpHMdpIEYH mRT/ABVPHLcLZRSeW7RKilNxyQAck
+ E9zin3+pNIs8lvp06ySI3kEKCAhwCQB2qFKTdpK/nsZQ5qr UatrPrdL7zNigjFtbsxuHC8RMs
+ hHmLtzke2f5VCkc6G3uSYVE0Bk+5gADjH+HpWjZyRWFnE7I00a ryueSR6eg56Ut3uvltjKVSF
+ yrIqDBeP+LFaKo1Jp7EwlKDkm/del97+SMCGR0vRFAx5ZWjEhJ/dj P6mrQjhmljt3uVHmxMYz
+ k/KobOD6k9M1oLPDDq4RZLUwsrC1cr98YyD9M1E1sIrEzPC0sjgeYU4w cY49BWjqXfYqTU6qi
+ pWb2/qyKCW0RvJoWuVt4ZYw+4k4hbsh9yaty2c62P7uRVLSbpQc5VwBx+Iq 2sYVSuxduSjhgN
+ 28Abfz5IolmllK20REUTYkkLjJ3LgYz71DqybVjSpKTaUXtve3+VyKG1uPtVxk E+bOoOP72Ol
+ STW8htPL86JDtZWZkz83p+NSPqFxFDMUMcryOXLBeFY8AflSCzuHjSOdxK3zBgowX C87x7Vm5
+ SveVkcVab5lUmkn007eRTeWCNHcsk8oYIkcYwX5yD+GKzJ7K+utQW4mKwS3SvJhsjDDs evJxx
+ Vy4LzXtwi+Tb5dE8wrwWYcEelQz6XcSPKnnSSXUXMYVjhwp+Yiuqm1CzvZs6G1F25kpPurq 3z
+ 0RhanA8X9nv5ixNJC7sHGfbH1rP87zYlXz43iUlSdvQngZrqbv7LfwurFYplAMG9v+WZ5YfXjO
+ awo9OMO1YQZxKPMBjUk7R3PFd9CquS0tx4NX0qaPXW23muhRnja38t/LDtuKsATwfy9KSCzU3q
+ RK 0WcsWJXvjIH1xSvL5IgguonVnmXMnQRqMHJFW7iJjdA28UwV5XZmAzkge3TAro5mlbv1MJ1
+ 5TrOm 9+70/DZjBaxMiFpViATDE5Jk75X2xS21rbPKZJX2RYKgZ/iI4FXbWKGfT0kuHCTFcbW4
+ yp6EfhSC ezhv4yIGdFjbzCG4Jx8p/DpWLqSd0ric6klJQk5SW+iRHaWVwdQWUxlNwI2OM5OOR
+ +FULjS33Isc MrIEz+NaDwzyNCAjhERUc9dhbqTn9aIiI/tQWUyCNwkTA8eWec/Wmqkk+ZMuTk
+ m+eV21tbRfO5if vE+0bFaMNMp5PTGcD6VDsZ7uNFDlmfn25rTktRJM8M0iwsiliTnsQapSNBL
+ f4EjxIkvX269q6ozT 2KxKp001De26RIYdt5ttiJWCMQ4HDL3I+lQLDLuj4LKEIGO4PNSnzRLN
+ dW6uI5C4HooPUfrTreZx ClrIishkDBioyBjpReSRzVJzilJw5l1t/Vjoo2WFZRNDLPIpJZkOO
+ x6/n+lKl0o05UDI7FSU3ruK jo2ffuKWJZYZYxZzQ3SKGQsFz19c9T6VGVZd7CDJztI9Mfxf7o
+ 7+tedaLev9f5HqNUue+7Xz/P8A yKH2OST7ivLtycqDg46896kTYSoEcqpySmeT7j8K1bKFbV5
+ TvYxk5jyc8Y+99O1QxNH5m8XEO7lm JQ/MMcAenvVus232O7629Woad7bfqjPe4VYY1w6AnCZP
+ 3u/6VE5jlvYIbeKaWQuwMitx0yrU/Y82 9vK+8+MkfdOMkD3p6Ru/ARP3hVSduBnoBWqskP2F6
+ XxKPX/h9dRi21wumZEE6qhA+fPOTkkflQyc tIJI2ZgXUqvKhuMVrS6eVmERkJEX7uUjpuHIx9
+ adDHYpFlGIlSI7AxzvB5NZe3W61MqfIkuV3+Vv zbKAhaWz852RCZTuGOd2PlUemetNRFmkPnS
+ eSzjIdhwQBz071opa7cyAbWaQDY38Bxnn37/SorSI PMkcEYkVshsjdgnkip9qrPUKUlaT9orL
+ 5fduZMgMssWJgwALAAHAz1qytuHcRI5MahpA56YyOTV4 2U6sFW1Ko7MyuBwoA5H41FBE8a7wj
+ hFAzuPO09vzq/aprRmsOSUHyavpbo+4WvmQC58qS2mgaQfv dgKg54HIyAfStBLRDqrH7JOCJN
+ rMCNoUjP8An61YOnXMt0IRAURiTKw4UPj5f0qUtFbW0c3mkSCN g6s2S2RgfpXHOtzP3d2cc5q
+ o/Zp+8+3+dyGOO1kvmtYLaZI2/eKQ/G0DJNPjt3EZuVlDSvEDAjdR HnBz60skUz2hiSRAY22L
+ tHKpjJU+pzTxCstrD5LFInjY8nJTkArn261m5ab/ANf8E6acvZpWdk9G nr+ZYjdbaOCNmkkR1
+ aRcN+hPtUd3fGaZY7VsNu3SHqrYHBotrNBqcoecRbJESOR+VAPUEepp89s0 GsRfKixO+zdjv2
+ /Lv9az9zn8zVKhGo03dpXSWiKpu4pLCWafMkjMCyLxliOCPQVG0s0dpiQGCdDg hhnb7H1LZ6+
+ lTysgvxE1v5hRC6hfp0I71nG5ma8C26b5ePkZd3AGST64reEU+nmelCFoNyVra2bW v+ROZLgX
+ MjxvEpEnOV4YnoR7CmXksL3scEXmLsbYCzZDZHB/nzVy2e4XEpiDo0ZwwUYXJyQff0qm BBMI5
+ VIgwmArDJOD14pxaUttjlnyufa+3LqWx9qh2Rf6xHKEt/eCnPH4VoSXCxXUoEc6QOzNlnyR 3H
+ Ppms4AgQRxP5mCrq/93HQH69fpV6G2mbUpI3zKjNwR/Gx4JHsPSueoo7s55UotynK2i9H+JLNa
+ pcS6fiWNoxExkBHEpxwV9ADWnDZW4s08vdFIIh5ivyAMYPHrWMj6p9sWWONYUkUvGjqDtUHbnp
+ 07 1agsbydSJxPGwIThsBsd/wAa56iaSTnoJRcbRdXTfT+kTSQQ/YvJjkKW6jaJHOccZHP51It
+ y7Jh8 iSR02g9G3DHH0qF90sgtzNFCHbeA4zn2H5VdunumUvBHGjRnKhkBJPHSs30TO2jKo4KE
+ bX6Xdvns UpftaxwWqymQRwlmQ5JXB+YGsZjapcQyvDcRxyFWQq/3B3B+tbRe2iCefcDznYZwc
+ bs/0NVJoLFp mDzbreL5AFODnGR+Z6V0UpW0aOuE1GMoyjb0X5bDnezyrBXdVVmZN/K4OFz9Ot
+ VJnH2yIMJ5AsJM rI+N7HoPxHSp5bsKVMcaNM8ZkPHB4/lWU6B7kvLdxRK4UHk8Ajn8q0pQ6s5
+ KdKKg5u67Xd39yHyQ TxQWrW8MytJ/rGLD5WB4PsKkiu7d7j7Net5tugJEkfB3AfLg+h9Kg+e2
+ tt73a3MI+RyufmZjxj8K bPEILxEGyVYlAbaMZx3/ABrdJS0ZFKopR/ePbZrf7/8AMkSSBUR4h
+ Ise07gWyQT3NSiaTbAITBLN 5LD5ExvP/wCqiO7T+0vLlEUapDLtAQABiOM+tQRW3kadHcQybn
+ ZR8vfjHI/Ok0upM5xhpb06/wCQ k0MyMjq0EXG5mKZx/wDXOahkVSWhsYMSiTMRx95F9fU5zQR
+ fQSCCQ+YVBK/KD781FDZ33l+aI5F4 YhvYDkVqkrXbRqqa5b1JJN7Wf5rqbNhCqvLdX0yW4zsb
+ K4y38B47VlmW1ktlaRvtjq5wYiU5z93n 1q8siR6DA6p5zDHnBju+Y42j8R0qoklsbM5gWJ496
+ IwGA248Hp1FZQT5m3f+vxOWjGpXq3ld306J fduR3MsX2TDhI5pGDbMYKEev17UksU/lykmLNw
+ N5A6qV5btxioZUNvPChKOpQhWxncM849/etdl0 2f7NbJM6LIS5YsSenC/U1rJ8iVtjtrupQgu
+ TVd0k7HNSNE3lFElBYMwYNwoPH9K2IYX/ALOuUlgA Pmr+8AGY/UZ9D1xWhaWtlFp6+XPFG+4M
+ vmnJPoKjF9b272UkmXBiYTYPDMCeMetTOu5aRW3/AA55 31icpJxhd+e/4/8ABKcDPBOtz5c21
+ GA68NjIP5549K0Q7XVwqySLC0swXBHzFtuV5/n61Etw7okk M0LRoNgiK5O7OT+dRRy+bqMcty
+ 6w7ZVZVxjnJx0qJJyu7am1bmqRlVWkkun5WLkENpcyRI/mwZfM yq2CWPQD0HtVojT1iW3Nyok
+ 3CRy55BXp+GOKQ3FnJLJvI+0vJn5eNvzfrxmq8UUMetXW+VIVWb5f MG7dwSPwrmd3e91Y5XBy
+ fM1LTp1v/XkP1I28sLNB5iDEjDL8YYA/l/jVeBUazjEV5BGpj3kOu7y2 B4Qn3OaWK1tWhQzzG
+ QeWNwVsdG4/StV4Izdp5Nvtt0RhIMDOTz+lN1FGKj2O5csaai09Nf6v/wAA iLw3FjKZ5IiJSZ
+ AkY2lZOyfXjis1YruCWFJZo4z5DPJ567xGQeQffpWnHst7ASwQm8uVKxIV+64b knHr6HtUBkj
+ eKJ79SvkwmNU7shbkn1IpU5NXSWn9dDKivdajDT5X+4UTi4sXjSTzD5uAyt94dd34 d6zgt3Nc
+ h1n2xuoYGQk84p4FtHE7RhobJpdsTMeoPv8A561KtzYTT7lL7zJsgTfjKnqfwx+VWly3 5UcU5
+ uCbjt3texEYI59IneJnL5CmTccOc/LUstrLb2zol64mllJKsxJ4A/TtVuRryN2jIidAyliq YB
+ brkegqbzWuZ5VIQs0jYbb1XjJHoKzdSS9DzcRiK0ElGWi1/r/IQDybKaVCzsx2qM5CqRjp61mN
+ p1uWiX7Yw8iFlPzngjBA/Gt2K3DObUP5bZZstzuCjlh7VB58clgrHy8SHc5C92GP5VnTrSi7xZ
+ zx xtWPurV/13RTSVDv3xtgkGfn+NuRj0AHUVox28kkcSYUMbeXqPu7e/0PpWWl0rytawotxuc
+ bivVS B1+mBUYtGW9h86eSOCQFkBcguvRiD6VU4edhzhzTT+Hl1tvclhsYJ7a2aZWYKoZGU42B
+ hgA/zqGC 2uLTyh88ttvKyEnOWIOOvr2psFxC0zRafcCNgQEEmWAC5wD+taSu11bRpcN5CMQQR
+ xs7Kp9STVzn OOj27Fz54NX+5p3/AAI547f+ybOQ2t0ZGg37RJySvC/iKiu0uX0qNoF2zDBkBG
+ S+D1HoKmNy6acI GUCXcpYMOjE4yPRfatW4kWRIA6BmS3dJzHwNxIxWPtJRa0vqTCdWHI3G+vm
+ 9PO/QxozJHcySXMO1 SWymANxGMEegFQzJLN+9SO5S4L7Qd/G36dsmpQWuPMjmkWRWxKu3g4x/
+ I4qyVuis0ovbZAW4Taej Dkf19q05uV36luElNyej/roVioe2AUrbXBlDCKUZMij+MY7AVGsar
+ MxDvEsyFkmckrg9h6Z/rWi2 nXCabJFHjyovlSRlyVJwApPqajiihMslpdB43J3MWPAK9VHpx2
+ qVUVtHc5rRgubmv5X/AKv6HOrD LHJGd0UNsnlj51yfmBHWqsFkvleXH50siy7XMb4A2tyPoRX
+ YyW1iZxGzGVZlDhA3Me052H3rDl0u 7N5G9gsjrL+/YZztC/wn3NdNPFKV9bf1/wAObTxUHzJe
+ 7deiOXuYrd9akZFZrUBypJyW9D9M1XIj /wBHdTLcKIyJdrEbH/HsRWx5MCXMUTXUeAGEJVfvL
+ yT/AFFLbRRR3RWK6gLcuQUzgqO/rXo+2SR1 1J03CLT97qulvWxVtQG0iGWCWMx+YyTlueB9xR
+ 6HBp94k8iaYosZWLQFGMQxnH8Rq4Z0h0qNE+zy 5bMaIuCFPUH1J5we1QRXFu8Rmj89XTPlKZN
+ 2E7/lWalK/Nb+vwOLD08To46a9X/TQQWs8GnpdR3S 8qUdG/vZ5FZptoVuIlMjmRVDPg8eZn5R
+ +XatWJ9Rlt0DeT9mYhg4QYw3JX/ePrVOeCC1MMyMW3vv VS3JX1zVwk+ZpvXyOiKqTj7ObXM9r
+ fkyvI9xBqVx5iDzBMFbeuduO35dqpMVcvLtYj+FUAB29d3I qwytOymabLq6KBnllPLE/QdzU6
+ pYvcXBG7yon2RsW4K//XrdNRW2prVcabsrp9bLf+vIwS0gjVd6 hd7sx7EEDA/SrME86P8AIgV
+ iMfdB2fXIpzAb3aOARxsC2G56f/XqCU2yGMs7NyG2huWbNdN1JWsY xpclOdlpvZu977nV+ey/
+ vjCbeRWw644OeCMeuMU1LZpXUyXUaoke1m59eB9TULGaZTLJDIY2kfJH GOBkH9KsW2nwskZkn
+ 8tZIi4BP3mB4/AV5ztGN9vxPYp07U+fl+e6foxxt44Y7m3hguUkeUMUkfJQ D7yn86pysn2ZY0
+ 2sm3ggdw3StMhRCnmXQllizuYA4Oefx9KgaQKqzC1ISZc4OOcnGR6AH9amEn1/ r7yqClF+/dJ
+ bX/4IkbCOe5SaNoUaQMofqBnn9KdcRRhVS3vIZ0dtyRqp3DBqWG1Z7g26ROAhMcnm HJDt/hir
+ C2lmtr5UxaN1kwWz+QqJTipXKqOmqi017KzX9fMyY5JItzqsqSyTfNubgHnK49T2qdLC Z75o/
+ KcquUAAGV2jp+HetARRRlEuNsaBC0bHtzxn1I9aoXSXcCQkM6IhaOVgf9YzcnH6VUajb93Q Kc
+ p0pXste/6alo+QLCOOC6UOpG5mBwM/1NPtxE5lMit5MV0kYKNg57UyO032FqwiZ1TdgL1ODzk9
+ 8fpTpd0V2LqFGW3Me5Q3Ic4xn8DWWjukxQhOrTkqctXfdobPdXIubl1LCUXB8tQM7lI+bA9OlN
+ gQ vBgAh1mX5Tz8uM5PtxSCCK50yCaORmuIyqRop5zz19STz9BVh/OsQXiCh5lBfK5OQcZHsc0
+ 9EuVb m1Kh7vJ7ql6b+r7F+Nbi41WKQMxWWIzvsOArAYIP86ovGZLWFEdJCrRh37A9vzqSNY44
+ 5LjModJF hChsde35VcNoq6vMbe0n2Rt5TEtkMSRj8RXPzKL/AK/rqcdVRUpS5tvS1/1KUpeyu
+ yLcGbe7Z74P /wBfNbUy202gBEXaXdXjQH5sAgH86yJYb2wV38vlZdh3DOxs4Cn3INXY72eG8u
+ FdoGaORYV+ToG/ z+dRUi5JSjrb8S3S5oxmkm12/UqTacvn3X2iYv8AvWYIpweB15plrF51t5K
+ M9u0jCZTM24v6sPYY 5q1LcGaSaORhJGjlXCDBBPyqCf1qhJPqfkJbJCryoTHKQnOTyNvoMdq1
+ i5yjZs6I1JWSlq/w/wAi pJBKwlVJBJhhIze3QCh42guy8wZZWUqxBxtbGT/hTZYHHm+ZIseJA
+ oUn+8Pl/LmtNU+xOIgyzMbZ pGLc5YHtmt5Tsl1O6eIaS5NfJ9f0sZ6xyywRP88Q2qPLz90dCx
+ /GpJY8vFbBg/kxhCyj7xJzxx3F IuoTDMgjX5Gx0zjcclT71bSaI3SGSJ5ZDKCY1O1gV6H6c0p
+ Oad2jnnOtGSlOndLVa9flYpKftF4E UNHC2CyA8kdCwOOgrZP2y0nKQSJPHEjCNgPvA/eaoolH
+ 9qecAsK7hjI/hwcj86jkWGSBHlEySu2A Q/ypkYAP161jKXM0un3i5ozq6r3X87f18x7yyT6dJ
+ P8APGhQqi7uQuRmtWG6jjtXlR3ZhIDMNxON y8/TFZsyvYwxqtxGfs4MWCvXuDj9KjjSaWSONf
+ 3jyxSSlV4C4HQ/SspQjKPkdMXGVNc2iTeu34WN CyDfbE8ueKdI7faqheQTnGc9TQ2o3NvJyVe
+ FU+UEfxH/AArIgSTfZyEOiOo3Nnjdnj/CpXmtE8xL mKcPJligbBjOc4P1qnRi5a6m8nQdVqKU
+ o2XTX8CXD3jW80sLzgR+WqxnbuOSQ49h0qC6n8prmJRv 81g+7HA9f/rVZN9eRNE9vD5cQVsbl
+ BwT1qv9pVFtpHmguyImXCJjGDwDnv8A4VUVK+qFOrONT3o8 3az/AKRTgvIxdsXBEewqrE/dUD
+ gH60eXNN5TwvHc4wr7V+6cdDS20PmyvEGgjRW34K9R1Y/QCpFu YY4XELBVBJVQOSc8HP0rd6P
+ 3VqYShOVW1Na9nt+S/MSOdfsxhnaM7ZVdV2YLHPX8K0poyUkaYBv3 oRGXjI75/GqqzPvmmkjg
+ lRZUCzKoAX0B/Oo2e7acLHmSSFWDei/XNYyjd6aHLLDSjV9yKsvMfOYD BOwifzFbYwJ5XjnPv
+ UVn5n2CadEcohAiOeCpzkfnV20t02r5sggjnchRIc+YMYz+dKYfsf7h0e4D KC5jOApXp+FL2k
+ UuVGk8VFRcEua3m/8AhiB7G4uGt55LS7CheTu6nufoPSnyQTiITRyB/wB2+zaT gjcAPzodVkn
+ MdxfGNXDSSDJ+V+m3jpx2oQWbWlsod32QjIVyNuW6H370Xlp/kKMKianKPytp+JGI 0uJriJoJ
+ UeNkVgGwN3/1qrhHt4UTyGdJXDsx5Abv+FWJIrtpXhhglUxSBBzySfU9zT3tmguJUnlM NzGxG
+ 1+ijAzke1UpdL/I2s2/ea9N7fdr+BC93JLFBJ5EbLFCyA7BkE0QiOSGPcnl8rIHHADDoPp7 Va
+ kM0wjQRo+JMII1xuU9G9wMc1Qe5khnYh4nzvIULkegFOOqskHKnDljDX5/1+BKJIJZDhZJpZ0M
+ gCHG0joPxqOFQlrEZZraIxtsCyLkgNyW/lSB4Y9OUKcpImFVRhg4PBz2HtV57KFrSBZ8xyMuAM
+ 85 B5B/CiTS3Oeoocq5m/69V/kZlvagsY3u4Y5Fk+TKn5gvJNMVDPdIFni8uQfIzDgnnA/pWt/
+ Y9tIF WJLkKX3I2/kDPANE1rJBptzIwV44rkFNgxz7HtzT+sRb0e4lmE3VXK7y87aencdHCEtT
+ E0LtKG38 /eIX/CqcTyvdxzzRtsZQ5PYFsjn61Izz2YBk3G7hLQ5J4Ibkkj2BrRSAzpOks0f2b
+ 7pVRg8DK8/h mseblTb6mkW2nKfXrbUpLsZo7Jka2BwJC55YL6Gp7lk8mFIZjEs4MsgZsnOMCr
+ o09pNl07G5GcAx 8csOn14qxHYSpdwiJ4bciMszzpu2kgjb9fSsXWhe9yY1YSd3K/L5/wDDlBY
+ Z5EMsz/Z4QAPMHCo/ YfWopJFYsH2tkEmU8qB0K4+vFb9pbpILVJp4ii2jq/oDnIz7+9ZDrbWt
+ v9r86J0kZXRPRQMAfXPN RGqpSsTUnRrVLNO/TRpf16/cZgR28xoGGwrmONlzgjgjHrUFsY/7N
+ kuJNqyLdRgR4wyZ6j8e1XpV sp51tjMyTPIrK4PGf7v49qfEl8k7JPBGQWbJ8sAZ7fjXVz+7qZ
+ 1qXuNSirrdbO3r1BHaW4vGTckS TKrlj91ieAamFmWBhRnm2EqZIzjeq8kj2rOkR7cXEW9UIkB
+ mLDhmA4H1NadtfTJAgkKJbylWJK8s xB4B7D2rKpGSV4nJisNVjS54Ky83+Xcc9pmMsPPAbLJ8
+ 5zt7nPpVtLCOK4ciOQxhWIQn7uB/jWDN qL+WI/MHmJMgGD/B/EKtz36Q3HnQrM6P5jBi+R24/
+ nUSpVdF3OCWHxEYqM+o3Z9jVZoE8xvLAfaP vs3BI9uaikSSDT9tk27a5WPzDuKqpA7+pJofU4
+ /tEJnRijQvt2nAG3kVWW7spktWZpOYnaQq2Aec n9a2jCejaOinh5qpFqN/lv5b9CNobiz1CNV
+ 8mK7lTKoU546j61dg1NEMSbo5C0LhyE4V88Hn0p01 yktqZhBJ5o5GW6ggZx9KC9v5VswtARty
+ ygANuJ+UE05e8vfWpo6ejq1Y3vp9lbEskkUmlWyR3UT3 L7V3f7p+9+Nab291Lbunn273QmLsE
+ TGOO4+nNc59lmmSRmhdZy4VEHHGecfStbF9BNEjTxSRxrIu FHzEHGGzWNSCVuVmVWF4Rakm09
+ tPxtuiMM8QSQGKVUX5nVcLg5H5YrTtoLY2hQ3UZZXAZM84Xr+h qlC8Zhj80jAAXbjG4A4Y/QG
+ iKAPBcNBcxSqCUBC8t3Y1nU1XY5q9GlK17p9+j8tjbu/LNpJFbziJ GcMN+Tz/AAA+/FZs91cC
+ 8t0uvIjmdW3Hy8Yz1/Gie12QIzXAOeAvcn1psTyGYABGZEKjeufcn9Kx pwio9zNYei6Gmthqx
+ piV2Yu7OigIeQDkVjX00sNsypKyyvIrlgxAGO49j6Vsz3Eql3sgiYmViWXI A25C/U1lSi5ur8
+ XEkW1JIWE8W0ZDEcY9MZrqoXveWxjTjUvepJKPbS/3GM6tJPdLCiSThw0WxfzI 9s9qdYi6m1B
+ xcRiJ3bcCUABBBBH41pW9w4s4EneDEEZjDogBfnKnPfNKLiKW3ISeBJHkWSMMOUDD kH6YrslU
+ dmrfM66s+ePJGHz3t9yMq6hiHlGJlA2oIVJ5jXP3W9W96Y6TQ3U8s9p5UXnbAdvABBz+ FX3m0
+ 8RP50Es4QbY2jbAZM/epXjmuYJjbRud02/LjI+7/wDWqoza328y+WomubVeen4mbDOFtTEo aS
+ OOZDHGAQxABA/WqEhY3dvAdsPyoR5o3K2CTke1PFjPHpcUhWWJ3XchkP3gOv5npTyInu5XEcoU
+ YMbHoExyPc11pRV2tTR4ai6TdPr89fLZfgWzY2cjG6muI3UxsrpH8pDdmHtisqxjmitlkZ4VZD
+ 99 l+V/7px+OPxq3cLHLcwJHMGhZiY0UEHGMDJqrKhjmInSVUQgZDEA4HH50oJ2s3uedChUkn7
+ XV9v+ BomRMn/E4jKTxiZgcRkE7F53A+9Ytu8ciQvHE7jzwgfqI25xnIrTiEswiAbzp9rLiNcE
+ c8A+5qqE tnvIFVGUq3HO0BvU8V207K6ZXs27yjNbbNar+vQ69EfzZYLp1QkhpCeAp5yMe4xVi
+ ya5tdMtQiCN VG4TTKGDDJHH51npzJN5jGSYjDKOCzHPTg9Bip1lkiSKOOTcN6J5bjJ39Mfj1x
+ Xl1INqx7+Nw0ZR Ttp2dkvlrf70X4tPeaGUzMke1xHwuO+cH396jha5SCWMrEVEm1Y9nIBHJHt
+ irV0oIZZ1mSASssrq 2Ar4ABqnpSRrrM0qrJP5J8vy92S+c/N9MVgpNwcmccZynCUpu6jsv+CR
+ LbRxX8TefldpDndyCDgZ PrWhDJG2oIrRmSMMYwndSeASe9JdPY2/2aZVdlH7tOcgqR9/3wahQ
+ tbWyRQ5kKnLS4+8V5yPqDTl JzVy515VlzRur6a2K5Rft4hc8xxYG8ZGz+L8ff2rVMFt5KpEzT
+ 7QykhuCeAD+PaqZilvEku4k8mP 7sRPO5cevvVtZLRtJFrLOsCYZgp+8nQqCe5qakr2t8zsrVF
+ OUeWWmzSW36lM3VzDFJGzRl4pkUoF xuGeSPqcVaRAbO4cQSRS5KoHOQxzxgfnUbXVtDBGZDHL
+ NMpaQIOS/wDCR6D2qMzyRQXNuJ1GJV3b 1yV2j/E03FtaKxU6Kk06Ufwf6Fa1NoJVd4J4xjBw+
+ MZ+7+INWrXylupH3M4EmxtxyN204qrc+fKF lDxOFdRIij7wAzxWhYvZrqs2Z0jjdsszAkZZTj
+ FVU+FsiULU5Sjfz3YBGmsxFFGRKoy4bklh3/Kq lrhmPlvOTsBY7+CTnn8K0YI0tIZU+1RyylQ
+ 5IJGQvJx9QarxSSSXAmcIttgKsaKASDkdfY4qFLR2 2No3hBqMdPuf/BFFxIsHlqrXYOG9c4HX
+ 655zSwm5+02pWEP5sJldccsQfvfQUR+ZNcSolpMm3AVw funHQ8VbAgaye6j8yO4OWUbs4QcEf
+ iOaiTUdLbmNblppQS1l6P8A4Yql2aaOVraRYgmM5A8wHv8A X0NZ6ht8ckU32dZWEhMhJ2gkgZ
+ qxDZJFELh55JogNqKrY+Ugn9M0yRVNtI6sBHHLGiuBkFdvWtYt XsjalSja0Ov3fjdDoLJo3Ju
+ cs+4GIZxuVev9Kp/ZpzPsiLNJtKBGOSB1wfrmrsptiLUO0rJEjRFg /wB71Ye3OKkaWM26xsCY
+ 9oLqp+ckcDn3pqc9+4QjiXUb5r/LT/L8iKa1aO4txBG880oO1E7KMbif 6fSnbNt6ELpEspacF
+ uSdowpB9D6VPE98LZpoyEk8xioIyVzxj6E0qRX0EOJo1doMxu5XhQecfmah yfVr+v6Rk+aMru
+ af9ea/IhguS9r5k1zC2844XiNccofc9qpxCB3iaRmuNvEcMZIJHPJ9/wDCtCIx PbbruB2DMPM
+ WIhdj7cAH3pbfy4rRHjtZCIEMZbgnceMf1p8yV7IrDyTbtdv5L/L9B8eoqsTuY1XK kuHAYhxw
+ p+mO1SBVlaO4BaOQx4HOCwK/MfwPNNksLh4YmBhdUQJt2ffIGfzNOlsmlljPlTRYjJkB fgEnh
+ R+FZN0+jsdsoUo2Stfv/Vv1Ik02OBY5ILpZYo/unJwSejfT0pLxmaWMyQ4LfIjEffPQn8Kl W1
+ thPKheSSKOVUCq5BGf8K0oEjkmEZUYRmMbE/d9vqTSlVcXzPU6lUnQleyb726fLr6mCtrMyqsQ
+ eU8MNp4BHX9KtpaRSWTGJlbegJA7Ed/xFbHmmSENGyW8jlclxnkA46etVZZCqzg3EH3FZgq4Mb
+ Lx il7ectNjSOIqVpXpzSt5P8d/zMtkkRI4/IZnijCFl43buao2sSTamsbQecPLO9F6lgD0rqk
+ W2mtA 0kolU4VVj4OV6knuBWBMr7A0A2mSTcSBhk7Ek+hrSlV5rrY4qVf2kpQuvx3/AD/L1Egh
+ zo0weRYw XXMZHzFiM7voKkja1SBPKLzuRhirdTmrtvb2cd5cySyM3ly7FGeNnf8AHnrUdxDap
+ GVs9ysMxg7s 8kZFT7ROVtTGEoqpy6272svw1IkPm3O7y28hXEUS55TOf1pz3rxXS+aUAGWOR9
+ 5upH5VFbxyR2qP JMjE7QmB1ycZpJLCaO/YFhiIFCjDJJxwPrVe45WZvHkm3GaWv9biPKsN3I0
+ USSR5OI3AZjkZ3fQC po2m+TMSyW7EPHsXBZQPvA+grQ0+1EdxELi1fyXXMsh7seQo9Owqw1tG
+ JriQxyRuJgjjOAoK/dA7 VjOtFO1jCq4RnyqOvfT9HuVoLW7ju8bTPskUbh3Pb+dPniie9Jt5Y
+ 3O10eRzuBHUfj1qS1g1VILw wzRySLPGxbZwxx1HtWc6Bb9JJLmORRG0aCMYABPGfc+tZq8pt3
+ +70MvZSlVclJJ22V7/AD6DLd2n v2mbKR4WIdssehHsMVWubX7E3mN5cx3FmOMhW6FTn6062Js
+ J4mgcMUXZKrDIVycg1PYRwPqEg1CK dwiFDhsDPr9ea6m+VuXT8zun7WLdSWqS7a/iVZl86UeV
+ EwjbcoYDAUgYqxbRytZ28iyLIDEVbIyQ xOB+NbEVrZ2sNvHCZFghDIzu+4OzHO76Crwi8nS4V
+ YxgSHzCVXAADYK/WuWeJVkkjzp43nSjb/P7 ys8LQso3llBEaheoJYYH1zTJGnk1G8hRUbdMpY
+ be685HtxViaS0MkqxOZo/MX5VbB3bsq2fSmXZE TXEiONwlIRh2HXJ9ec1zxb67/wDDHGkoSs3
+ q9romhMEs8clx5LSujOflGMMcAY9T2qL7MhWGAjaV gKMo4IZgc596lgaRdUW5uBHIkgDKqJjH
+ GPy5rQjTzLhUMkYDDCjHzEDOTmspTcXoJy9nJyh0676/ cUxMILRIYInQLtyXwQWA4P4c1TSdn
+ uZlvsw/MH3dAxAPT2reW9AXHkp5DBWUlR05GTWRc284ijkK faFjxG5jGPmB6fiDSpyV2mrXNa
+ Lgm+aOr/r+tTLgk8/U3CuIVnj3srf3iDwPapFht0RYmQzyZTcB 068ED0HeqlwCJhOrxxkxsEX
+ HKDODmtSE+QUJuIEcQsWZlz05/rXbU0Sa/qx7mKpS5VKOqatbX8dD NvYkF+QRGJWJkV1GAxXj
+ j0z2q5bWtzC0zTl/KaYNGp+9gDJ59s1dxc3tspXyc5CBwn8JGWA9+ODU +IntAsjOojhIOW+5u
+ HAPvWUqz5eU86opOHJHfr1+4zZk09beViVaMMI3JOS5bgOPpmufuIpEkFrb q89tAyJKQejD3r
+ pryWb+yUeIQyyFMyAJwxGOnpxXNrLfy226J02GRdwVeTzjmujC3te/3sinBql7 z083/wAAr3q
+ LNJA0YVNx3NIPUnCj9Kr3UTK6W6brkmRzII+MHjcPalNptsRERJ56SFZELZJJbqPQ YotYHi1f
+ E2YmCPh26Adya71ZLR7XM1ShUp817OOvX/h/wEaCGKFraOGZ388s6k7iowMYx2NU4rgQ X6FoR
+ sUsNu3p8v8ALmrhCSQtEiyxyFQDK0hOG6lTxRDp10IZ5FgZQrAAyDPXqPrirjKKi+ZmkaEY 0+
+ Wa19df8v1FsxbmCBmdxJIFI+bgbc7l+pqxIheyjiEgLIMiQdAo5Off0qC0tJFnxBPC7biEGM8H
+ gHp9abIJbYykMrEOEx6gL1+lQ0nPRm1PDSlP3anvdE7/AOQsZ/cxXKzu2Y933iCoJwc1ofaobP
+ TG kjkaR7h96MxyNq8HHsc1TMN6lmit5UhZy8qqmCAOgHoO9KhLwnzot6rGAAF4+Y/LgenHPrU
+ SSlq9 UZKjOrK+um+yv5FqbUvKMvm27FpLhREQQMKACTTrO+ZLy4le1kjJfDLkYGec/wAqz53k
+ dJFmgYkY AYcckZqeAieNPOlVQz49MBeGB9zxSlSgoaowxFKnTu59e2/4HUrMJLWUSeXI8kgMJ
+ UdVC4JHtWRd XHkRRRxlmJj25zzu6Dn8zWBc7AxigklZ41P2dt/ATGSD6k1fS2mGirctMAc7lR
+ uSuR3/ADrCOHjC zb3POU4UeV8y1e1v8i5hcRRAkSpGomdj8rYOCR6VkXEzwaxJJ9oBO5kEXOe
+ V4oK3f9ns81xHOIX2 yGJcbjj19OlVwjw2dtJPIszOm4qB83Odx/DiumnTSb1uY02nPnWt9LXu
+ EyW6WFgziaGRbfy5AzE4 cZPb61bszbyaX5kxEUrsj72+78vBwPyoSa3TS2McJuYVkTzi3zb27
+ EZ6D2qCKySa7bbLHKMsXSMY J56j2q27xak2v6/rc7aNKUqbjUbVu35f8PoX7+8SeKZIkU2yvh
+ 3QDqcHA9KLGV5IZZ5C4tyQwI7F eMceo5q1cxzGxePfBsZg2FTB7Y/E4qaGT7FZ3LXSKsTT5ZA
+ uNu7jH4Cubmj7O0V/WhrUajh+SENP W/3eZi3TtLaLNENivbFo0bnCg4/rmqE1shgQXE6IgdeB
+ wSMYrduWG2BbOBpPJ3QM55Vh1Ax71gTw PNPapPN5anBG7sD/AIdK6qErrt+YkvaUbN2t56/dr
+ +YnmpDHIyPDlWZYjt+4meR9c1VJkltfnEig HcxxnlakuIooNyAhgh27gcjcecVSkhlSOWRUkC
+ LN5ZQnkMR3rrhGO5FfDUaLjOc4y9dikHmEbCNA 0gl3NlR3GT+tRiUDaTbuzseCGwE9AfXip1J
+ YyseGBAPbIHBH1qooMbqgyoeYAE89sYrsVjilNU0k r8t32t8kdUlsVuoTFIrsVMgwSSCp4z7V
+ rLJd7fNktlU+YJTlRyQfmI9MVNGIYZVkluLfzIFZBIq4 VgecY9TniqjLci3h8y5iTbuEAI+8u
+ fmz647140p871Or2sK1W84Kz6vS/wB3+RrjbdXL3AIOSef4 BuPcepHQ1mTR/ZEEkUiiV14I6A
+ c8mnGd2sWZZY0CzqsZUYyjHv8A0pzzXCNeQmBrmB5cK6DoD2H4 1jCLT8jehDVtPTte35kqzQX
+ GlqLgxyRYUKFXBHOP1pkd1DaxxIwM7S7nB7Haxz+B6VVd9Si0+YS2 pikZ/nwoAU/T8qkdYY5P
+ MmXIVGTav8DZBCn3NPkXXbyMuSHKnJ6Xen/DbEs89y/lSIvlJIG3KBxu x0Hpjj86gjtjbT2kV
+ yBI6RO0idw/UqffpVNbuGW5SZkkdBjzQrdJD938u9aE8SmCKaVi/l7gxHVn zgNn0yelaOLhaL
+ 0Npy5XGHL6f1djkXT8xyurhpommyTxnH8gaqbrIyQId8L7P3nmtnJ681cgtylk rzj513QyA9m
+ J4x6Ais+aeaeaWHy0iRiWO5Bk9Mc/hRDWTs9vM2oSUnzRbSXX+ty3Pp7SqsxVzIso jmVCOWPO
+ R6ACoBZzRSLcW+4mKfayOufM54K+2KlijvWe6ltyZMlgWB43Y5wPX0psEdzdaLApdonY EgN1I
+ U9aalJLWSsYutPmSdVcr6FgzzedGtzayMLdtrquAy5Ock+gFWpbotcyLA0LpEJN4Vfv4+bc PQ
+ Cqcs7z2NwVkUtJMrbMfNntzWmjKkJk3RGV2Ln5OhPG0/XmsJxSSbRdWlTjFT3v0u1+g2W5uJ7M
+ +eq2cciBmlIAVm4wRjt2/GsqQSR3LIgkdWbEqqeYx6VekkkUvHLPAkQO2NWTqfUe2eKpWN3Mmb
+ kj lNsUgYZwxyOfenTi1FtImkmqbjHTy/4O5enjJZ45InMaodgQ42AkZU+pqaC2jjm2CN2gydk
+ TEEhT 0z+NZzieHUVXLeWu5CzfxIR1+ue9ajR3I05fLXdK7AiMctEq8EN6+tRO6SV9zplSbik9
+ O3b79RiI 091ZRmHASB1AI68/MTSzwwErLaRSxXEKlnWQ5BLDjj9auLcTW86yybIpnZkJdchi3
+ TA7ZrR+1JHY sWWMTMMsuAMkYCiueVWSaaRlKq1OL5fknb7+5k2EktqC0xTy3ljxuHscD+tRTX
+ d08hjSNsl0EvHD FuSfyrQkiZ70htsbZLbSPvFeQw9h0rHnE0kUEsQa2kYiYmQ/eG7g/QVUOWc
+ rtDTjOUpWV+/b9S1e 2rnT5vtUUkSrJ+67ZQYzn1PvVV3IsZBHG7xKwOFPOAQeT61YlZ/LnLah
+ FLILjLA52gDoMe5p6wRz 2Db5lilcgynsrD+H6mqjKyXN+peFa5V7V6X81+DLVvHFDdTXCOVDh
+ xtds8tjBH0pzXsLWht5JoVu o8KzEYDH+8PbFTzuscSQMgcxp1AwAvGQfU+9VGa3m1a6FsiCNp
+ N29hnA24Fc8fe1ZpTpSm+aSbt9 35CNawy6a5jcKZpRJI+eFI4K/U0lrFFDujTeu5SGLN/Gen6
+ UsM/kwKqWsv2lwHZnOUGD8xxVu7jR 1JVJJELFS0ZwNxI5/pVuT+F7Hou7ioN2+f6f8H5EFvCp
+ W2XzQsW0Nk99ucVaWN5pJDJCiGRlO8qM L8pyD6msiWXy2ZYZAkkZA2NyV7MPrVyOWZfPLMSIp
+ ECj0OO9KcJPUzqwrOL2+4pzz2tvIVKSIykK hDYByOf8aRgrDG4FdodGUcNtHH5mtNprI7S0Ik
+ kdPMjPXCg4OfU1H5cCXieQVEKIwjY8hkB4P5mq VTTZnEppL4NvPcrxNehIpZXt4WX92WaPhj1
+ z/SoGs70xx3KyxS4AyEX7rBsAH3NWJnuY7xZCnmoI 3RgBwG4yfwFSRqbgHdDLBChKMd+A/fIp
+ 8zjroOFWUW2rfd+BTdCxuJLf5JxcFIg/TB4PHsc1OGuI IQ1yBNtc5KjmQgfKwPpTzaW7Lvkkd
+ pXBkYK2CGX0/PFNvINwaVp/KcltwOSFJHTAo5lJpHVGPtOW Mne/lr6dyT7QPsCXE1vPGZXRkZ
+ m+UBe+KmeSO43TAMZCxdQD9854bHpnis37ZdR2Sh4lmIdCrhfk K9CMe5qSVriWNo2spIY/NVV
+ wRkDOSKHSs/8AghUw8IRTqaa9/wBN/wATShvN9rcK6skr3AcDONoz jafcnpVRUabUlSeAr958
+ jgAZ+Zfr3HpUEccaXL/aIp4l3n5WfkAg4P4VU+yqpRFuDCxUMA75Jx94 frmiNOKvZi+r05J8r
+ aXn+mv5izI04k27VgKAmQD7zZ+U/j0odbopbSy3lujPCxddmDt3YYdOvvUr xRpZKFl8obtrK5
+ z8x+6PwrTginuojJdQlvIfYuBjjow+prR1FFX6HVGr+6TWy6dfxLtoQ8c8X2eT 7OkyeSx9O4P
+ qfeprmNpLxljjkjWM7cOchQT90+5I602SZ4QzbGti6kFW/hHcmrlu9r5EihyAWDBm OThTwf61
+ 5km0+ZI82cb+9GNr+v8AwSsjW7xKrPGyhz5aIMMFByQfU+h7VBHpc8eru0dxHLZyOZEB GTtUe
+ vv/AErQC3GLmQJEYmkykioMHI5xUNtdS24hi8yJkRSACvKg+tSpySfKcSlVgpcuvf8A4HZ/ Iz
+ 3sofNtpGlYxmDLqGOck4XHpWqQ8MUv2dhIS5IBGSoA6VmNcSPJKlvbmWEbTkAdc9KlN1eTSNLH
+ H5cu04GODu4PFaTjN2uzqrUas3FSlp6qxpwSvI4Wd4drRMyoqYPHU/TNF2sc9rCXmEbOvmSkHA
+ 3q OMCoYpDECJ4jE8b+XHuPJTbV+Kc3S26SCPCxkAhcY9j7nFcsouL5lscqoTTvHZHNCKMooME
+ ql2LO XIO1gM49quNaQTWSTw/61lO5Sc85A/KpL+5jgjiT7LNcJMPMZozjcemR7Y4pbeGO2mmd
+ ILhTFMUj DyZGGGTn8a6nKXKpHe4SVPnWj/P8b/gEsSy21xAW8opdGPepwDwCuKx3EFpcPDNI6
+ CYhiznofT8q 1kHnRndIibdo6cZHc1HJN9uecS+SUaQFDtGFGOR9cinSk1p0NMLGFNuC1XXXVG
+ UjTzvPBaspjiKq X25Bb2qi9p5UYt0mCxKrHaoOTg7uv6VrhJoZWeFQ2VLsij72Bzj6ViPE8ln
+ 5aiTMLADD8vH1Jrsp yu9NEOrDnqe7FL03+exERCZVu1czcFGjB5y3IJ+hqfylluXdgyxRYTzT
+ yGJ5J+lQ2lpH5zXMtzGk RdQuc/Nuzg/hSSSxQ3JktbpWicFgCchmXjj866Hq7RZHs4qbhGV2+
+ rv/AF+ZJILf7SjxSLKm1o3d eiEDof8Aa96iS4nuZbdSxiTYAE/vAn5vxz3q3BePFgxRRu543b
+ fllJOCwHpiohcyGVEW2zFuCJtA znnHNSr9UDpTjFxdr921+H/DlGCxVbi6E0rwx27vlsnnnGK
+ c88aSRTiHzI1j2qpGdxI7+pAqw0N0 IpYxbSZZjLIzHIBTsfrUbpujhuVjK+Y6FUx93J+Wtea7
+ u2b/AFeDSd/x0/r5kSNBLbRx/Z7rzEQq H8zO7v3FSxzTK9qIpYtz25Z0KZbIGdv5cita68spc
+ KIDGiSq0kx4USDgL7fSpY7ONIZLiUokxLsr Y+Xb04H6Vi68bar9Tjq4hNezcbLp1/H/ACOUMk
+ LF0McksKjEJDc7cZyT3piQGGO2mjk/cmPzJmxk K5zgfiK2IraBr6KQvHHaNAAueoPZT7ms9ra
+ K1tJo23AvMHkjLcgg/d/CuqNWL0RTdNz9lDVf1tcv wS21vZtLIY8TyLhyPuqO361QnfL2Yhcl
+ fIkJzyCVOB/Om3TRXcSBW3K8YCp6MTx/I1SVolZISxQC N1aUscHBzwO1KnS+11PIjl6hLmb6/
+ wBdmagF3KgiSMQxMS0e5c547/Ssk3l4t5AMQSBLchMx5BBH NaNtcJ9lZ3nWSE4MajIOAefzrO
+ iWWK4SR4XKgMoyc/N/hz0q6Ss2mkY4WlBVXF2t06a/PU1bCVfs QlaSJi0iK7BPlj/ugj1P9K1
+ ZYIbi4SNJovNicmIxjb5qLyT7jPrWJBNFFZkeZG7W7hCQuAQPUdzz 1rpDOv2bzYZYJFiKoSq8
+ juR+NcuITUrouvRlF7t32v8A52KsmosWeQwqULBlPGNp6j61UWSG5SdJ pUw5fqc8Yzj68VOLy
+ T7S7xpDGrqzr5iZDJ3I9MHg0wWc19bQrcyQRzCMsDGu3GDkqfU4xUxUYrXQ 6aapQa5vdfe//A
+ /UZAtumnyyW7MqvIsgBbOzj7p9zUEtrbzxu2HkllK7kDcqWPT24zUsL28jQlIJ STGfM+bhSTk
+ DHrxVi7zbTlkhIjfDD5uQFP8A9c1fM1PTccm1L3b3/MxbpbizkzDHHIj3BJUqDtbG Bn+dUpIJ
+ YLaWQyorMweTcM/MOnY8nrWkZJbWZZfNRhjzkjcZJAJH8jVR57O709AxaAq2JWz94kHa foK6o
+ OStpocEsP7KSm43T3dr/gc9crbgL8yKwQDk9fr7mqgjWBZlIyBNk4PtzjjiraqstpLK6bTH KA
+ QcEk9M9OlV1imS7BWFikZy4c/qa9WLsmrl4qUXeUUrLa17/wDAOhS3idrkROWiVxtYEkYx9PWr
+ X76ZImeVJV2NIVHDe/bip5kWSKFrGSKS3ZGxtTOecdxTzb21rpjSeaY7xNqKrHOVyNxxXmOpdL
+ v/ AEj2VOM6NN21u/Pv5XXqT/Z4r3S4HRXiR2yuT1wMAfnTIEjWRPtNpd206Sqkm+T5ckHAx65
+ rVilS a5dFjK2aTO7ODgZHQD0HtWJqCi5htsSMJJdshwenYA+/vXJTk5Plei/I8K85N05PS+//
+ AAb/AJml HLMk91ujIlR9knmjI3EZHHtVFoVluoVkP7lwfPfP3ZAMqPxqovlSPKDK9xlT5SKfm
+ LN6+uKvwyQW liqG1uHR49wkL4Dkdx+NaOHJtuegsBOLUqT956WVv1ZVaW28i38qLY+3bMM9WP
+ U/hxTGtpDFKEn2 Iku0yMcqVxk8fWpzfxNaR+TAsOSnzNzkscGnPLK32y3NnO7gtsVT/D0JP0q
+ 1zJ7EUlUhO0rr5jJm uLZpJAj3KSyqUYH5Qdveqtuw3Ol+6KisAQPvMcEjB7dKfarC58ss8LjD
+ RSO+VwMgnHua0T5uIoo7 ZZJYsDO0cHHOfU9vxoclHTr9xq26TcW/efy/HYLWZrlZhBiLePMVs
+ cLkfMD6nikiGYo7gMDH5REc SnDEHjP5k1HFfvLeQSyRi0gVDHtK/wB8kfpTbuxmmkhCo7xJhE
+ MZxwO9ZctpWehjFxpNptL+u/8A wRIbG6MqxG1mhEaFXY/xMORV7SxdW0EqSRF98yy+WR8xJBy
+ QfQelJDPMFkR4pogTtDs+QST1/lVK ZRbXFtLcSSblRl++QGcHkfkaG5TvF/1+JT/2jSdr9P6u
+ aEnzTQ7UCyJC8RmZco2DuLY/SqSQKsfm WQaSVnJXccjGe47kc06K5dZhcX0TvG5DeWnGMcMPy
+ waksJY5dQwgEEQlLgscjOCFH09qVpRi/L7j aFOvCLVk4rd9PTq2W4kuPtL74SY3wVfHAIOAP8
+ afLcXsQe4VPNLviUovRs4A9qtQzvDb2gkIklEB 2qB1HJb8fenpqdrJFbspCRSr5nPYqcCuVyl
+ e/Lf+v+AU61Tm5uXTyJD54M6ytC3lybfucqQQSfwz RPLK80uQgWONozIVGCx5zj6Uyb7MZLlo
+ JDKJL1fM+bOHHQfjVq5vHgtC/wBncu7FipUcnpWPVWWv 3DpzfMmrO/TRW+8pvJE+nRvKHcfZ8
+ 4DbWfoAQewqhNNcxDy4YTIF+UbhnIzyw9h3qaS6uy7pGIuQ BsMeTgc8ela0cSzzM5jc5YlwP7
+ uM8egz1rS6p6tXHUgqfxK9+n9aBaWVv5k7GLzlUlUcdJVB+Vx9 ecVcT7Mt0ITblRJHuweoxnm
+ pEgRYbe4wzDyP3m1sB+eCPQe1WDPFJamKKFt/mNnJyRnGPwrgnUcn 3PLnV5p3SuVv9FaCAvE5
+ +T52zkEnnH9Kom3RXXZbyKBGfMjBG5DnOD9KsraSnWopI5VEPll2BHCl e31q39pkkmW5ZFMTK
+ wIC9WPb8TTUuV+67nVGXsp2i2/M5mC2aW+mwWI3bkTPO3vWza27PNOyzKFM 65UjO4kdR7AVY8
+ 6B7u0aTbDstnjePock8k/Sr08scWnOPNgMaDYsqDAYsBz+PQVVWvOVlbc66let Oy2v3X5abmN
+ NCdrPMiXAlPm5iXBCrwxPt3pqxzM8pBjWJ5FBbb98spwR+lIryqsizXturyEMkeOY x0Kmo4Hu
+ ZvLjjZRIqcKRnvyPqAK0s7FTjUS7tddkhs2ltBpA+0TIo8xVQnIPbP4UwRLE26FWuIw2 GZTxk
+ HgfjUbSPNeQFXaZvL3pk5UYORkGnia7hiWVpYHdgIwipw248vWiU7avUuMJxjac1d/L9P1L hJ
+ DpI0J80Psk5+UsevH0qA3EM4ZEtppoQwSNo3xjjv6mrsYSUtCt1DJIkiqzgcO2cAj8MiojLEru
+ LNDglh6hecY+uM1imr2tqYuUIVLPf56GRskMikSBlBRDgc4P3m+lW5IofPWNLa4uY5d8keHyCA
+ cL +dF2YUQhd8RiiKKCfugjjPqafGzC2i8wNskw0YU4yE9Pqa6HJtJ/1/XU76kuSKneze3T8v8
+ AMy4I HjkmsGhneUKTjd90jt/WiyEkK7r5njVxvJYnHcZ/lTb67vWuDdeQ8UUoYqcgHAAB5+tW
+ bcSJbtdS zQsPLKiNhk5BFdEnLku+v5ndOpJ0730fTe79dbfcLFfLIzSxhXjT5UDc/eHU+vNVj
+ cSmfa0lvMI1 2nan8RHBHHTtSKY7uSeGYpA7EkMi7VA69B6GrL2MCWMsjh0MjAvLu+VSeSMe+O
+ Km1OL1WpjH2EZJ TXvPyX/A/AzUWzaxjlh3mdm3bGbOzPAU8dfetlPMjs1DXCgRnbK2ThzwQR/
+ KjTbRYZWmdR5ckqSJ nogAPyn3FRy2NwLxXhmAKYJLDKvzzx9DSqVIyly3HXrqS9k3dLW7tf0v
+ 5fMvvtvrq6kmnVVSTZGT 0Klen51Zt43kntRNGwLxlpQBjaQpB/AHH51lzSWa/wDLUSyqGEZQ4
+ G0EZB9W9DWnAtvA8k0XnhGy V3Pu9MD/ABrkqK0fyPOxMWqVrvXb19ehBsd47Yq0hRYQow2Bub
+ jNPQl7x4nXyGVRtZ+c461LcP5d vKDHJGuVeJ88bVOSPeny6fmae9SUSjcVDA8epI9qjnVtTOD
+ hHSW729fkTzf6FaFlCFuhIHTnJX64 70W4jw0pwskThGTPIJ6H8jSP5aWocXMUgB29M7icfNz2
+ p7wbtQQgNcbFIdYuCGHTPrxzWKempnCE akL7W666+WqJhMSssZ2OQ6lGPPANSPKRbuPMiHmfO
+ WC42lT938azYS0gQnbEDAWkdhw2T1FXH+w/ YZD5hKyN50eG+8FHy4+pqZQSZ1SioOLStf8Ar0
+ K017KJPO8geVEAhXH3R1xVhVluWX7RGwjdgz4O CpXufYcZrGaRRbiZriL5tplXHQt1P4VoQXs
+ sUyxxEeVENg387s9K1nSaj7qKrUajheO/4f5El3Dt VHBG9Q0bleFZ2xgj+VZPkJbm4nud+2OQ
+ RuqsRt68fU1dlmil3iSKaF3YSmTf8oAGNuPU+tVry9jF k0qrneA0isc4f0+taUudJR7jpzlC0
+ Hd839b3IDdwR2iraXC+YWG4P8xGRyPyqvEbeJ4zI4RPn2lu SVIwaagt7qEGK3kSSRW3ZP8Aqy
+ TkZ/n9KpRXzs8CPEt4FQjfEMAnJOefpXVGlo7fMyq4eN2438+9 vUW53w2UfmhZYQyD5FwVOOh
+ 9+lVbtPPs9oEcLRO3mLjld3JrWVpZURoYtxjDFiVyuT2x6nt6U5Ve ZSssAgUx/vC4+9Ifu1pC
+ py2b6f1sFCm5csnDZ91+Wv4FO0lkP2VXgLb38zCjBYA4wPTHepLu2kij gMsn7ndIHK8YPUH8e
+ gqMTC1dI5UfzfNUO3/PItzj8qY0t3Hqn2lIpJYd4yjcg844p8rcrrQ2UHOq 27Ra1t39GyeOcm
+ 2lHzI4uN7BznjHSrEt9bHTohKoDbQItoxxnk/h0qa6aKG4LDYYEYxbx2PUZ9fe sc3MDCRZGjd
+ 5JF8jAwGx1I9s9qiMVPWw4uEoqXLZL8zQdI557p0JCGZQqs3bb/MUskz2LGUt50Yj 2FevIxx+
+ tRRPFdGVlina53kgxnClj0IHpxUM7XE0GZrWR5SNshU4G8cjj3AoUbuz2FKLqe7KVl2/ pjEu7
+ c3XmpbTJlsIHbKsDxkD2NYz2TmeYtdR/upk80nPLHIJ+ladzZRTzI0YkjGd+C33EH8P1PrV SS
+ BHtZ3Ln95IHIzjO3Oa66TivhZxR5HUcU3d6bf53IpbB1uYYUBlUgHzE4A2nj8+aoKlu08quJEh
+ MxdVJ5GOhz6Gr/2KYQLMPPMTAMpD9CORz6YrMuMpcyTtufcu4lRwpbtXTSfNpzGcaaqzbdS6Xb
+ Rk DLEsyqN0rvtKqnGWJ/kOla1zqEaSToskasrOrgjq2BjH+e1YQh3XrEllKOwVc4YbR8p/E1a
+ eKSaB ZWCsWctK5GAGx0rapCLauY1pxqVouT08/wCtSxZvBJd2ql0KgFWXH+szncTV1PNKyeQ2
+ 2MREyoeS GHGfyrIitXe8hd5UiD5ZXxhffFXY2FrGro/zAb2JORJ82Dj2rOpFX0dzedOTk2nfT
+ b+tP1OgWT/i UZbaJFJjDkcRAkZQ+57U37JCHRI7o70jZGXeST3H6Zo+8t/JJGTbNciJE78/xf
+ nirEXltHKsIC3O 5TtbrgAg/n1rzrtbHBR0jyp63e2y9W0Urf7SQscEDKe4YZyOoNF7Yl5PlMx
+ hRVZZA/AHX9e1axVZ obHZPH81u0jEcEkfKD9MdqghQG4Ehv7cRFSAMZBwNuKSrO/MtDojimov
+ m/BfqYd3HFFdrcRuHiki YZf5s5xjHtUVppsrW0bQTWzlSSVCfe54P4VpzW9tcae9ujDeg3I2e
+ EC9Aapz+abBIVYBpR58hiO0 oRyV9s10xqPlST18zgxbVSmrXu9/T+vMyXu0uXmkcxQRhXQpsx
+ 8x6E/iKyklZZnmEgEj4Vwy5Bz1 NaV6sKziRrZ1idcnkDg9jx1B71nypHJYRSLHIGUlQikbj7n
+ 6V6NJRS20YKFKlBK10/JfodK8UElq 6wyOJlYsoUlRjAHSmmOK3voBJvS6nQvG8pBQZ4xjvnmp
+ FW2jluZEY3OHOxIzjC7evv1o+0tJZWbM Ix5UJGxly5HUHNcSb2W3/APUbcY+yp3t0vv8/wDgj
+ jBciM2rEq4bZtBwQBySfXHeljHmyC4Ks6j+ FTjfk8EegB7VKfKnh/1jtO5G1QeSp7/l1ojTTp
+ kSKJblZTGH3+ZwNp6EeprNy01M4RjKDTbv2tt5 7leO5jZoJriKOO5j+Rwq7QGJINXL+xiCQfu
+ p4lUlGLP8vr+HFNheT7agMlvGFVivmx5xu9fUj1q9 JmJGMVwkzW6FW3DIJx1qJzcZqxnKFWlU
+ XItfV/5f5mNcwSWlmiwtDPBvyzBejLyBTS+5JJgksDOn yFnPzBjy30BqxMsdxYqROFgMy/N65
+ HLfSmzSrbPFALZ4oWdkidzkBDjJ/A/zrWMrpLqdtFJRV1eV 9/8ALTVjEV5LgrZKtxNHIVXAyG
+ Xb1H41YRJ2iImtrlZpVBQA/fUHnHvUaJNCzSQQyOJmDxlDjIGf 5HrVxms4LaLzXlW5KsQ5ckL
+ xyuPfPFROTvpr+ZH72FW9PW/rf8HsVknji2IkLRQXLh4Wl5A4OB+d TNpkku1pPNG2LaUV+WOM
+ 7h7DvTIHeS1SV7dzBkLGpxlM/dJPsev1qKWVxdXXnmSBvMEMkjMdqk9T jsP8aWvN7un4mNOb5
+ 2ua3fS//BIY3ubuztrfyZHZoWmQr/EVPX6DHNSyxOGEs9tJciU+d8hwMjg4 9BTJT5AFta7yYV
+ MW4HkEkHH86da29691GtuGh+R4nMnIUsen171b0V9EjWtRhThzSklH539X1RXc i41S282KWNm
+ fIUtwTjnj04rahtrIW7SSwyxRJOGbEmMjHB+maptZXH2qO3hbNwsR/fNysijhmHpU TxXVtEoE
+ c1zBIygOG+VQDgZ/Opm1OyUrEyp05zT52vJP8f61NcLHBYxRQM0beUwfzTuKv1xn6Vbh UCyt4
+ /s3nTGAhGQAKQei49TT9gF0yNxcrIFkJHEjjqwHYAcVPPNGixi2t5S4Qyrls7AOgPvXnym3 ZI
+ 1dRNxhCN7df8+5UNusskccLranIEofna2MDPv71PJbX9sSrJ9sXaQAg5C45bmkE/2lIPIZEO8j
+ btyQgOSSe5689q0rkiJLnEFzK81wJoiH6xgcge1RKclJImNZKdlq/P8ARsWxs/tItjHGViWHy3
+ kb uwPyjPvXQWunTRzKzRs+IWLIvWIgcq3qcHNWbQRCVLiCVIkGY0BGQQ38WPY55qaEiHEby+e
+ w+UFC RkdyfWvIrYmUmzzMbj5Oo4rbtZmb8jWcSRQMsbIrKzHO7HIx7VOkAmWSVmiEm8A7RjKk
+ 8/4UkxtV sA2HTceDu4UHinXSmYmJZo3hDEsYhgnAGOai76aEwckk07fL+v0GSR2ttdTPIrJbq
+ pC5bqvr+vNV AlqYQouYWCgcgdxnJ/rUpWEW6y3SycKQkecce/8AOqRikayUzyQ7eMKi7TKAeW
+ B9K1grrVnXCPNr KTv3/wAt9fmVWgjlulmKtMoXbD5Zx5ikE7v05qpLdwtHG8bKWmgEkkY7t0y
+ PQD0rRt1s3uTvuDkb hEVYgHGaq3AspLm0El1axRtCdyqMEljkYPpkdK64SXNZp6HRCvFN8zbt
+ 6/1+RmKYpWCHKMw2+Y/Q DNWJLSJLhHt5wylJPMKk8be31NQrLIbtX2KiKv8ApBYZCzHoPb6e9
+ ayXcxiJuFhWODdHIQgGGYdD W1SU4tWLq4mtTa5Ze6+l3/X4mNbllTdEolmli85IQPmAX39B3r
+ TTyzbNdSyQJCZBIGK99vAH41Ri vrWOGJmjaFzHtyT90McYNS3rF9IurePbI0UwChR0A5A/GnN
+ SlKzVh+0nWnGLjp3f/DWNKSSBHtnt kVSYnMuRnbJkEA+9c7LdzvqLRbNiDIdlGMs2MVPdS3U1
+ kB9imYyHO2PghiOW+g71KksNskSzTRNM snlv/wBNGYfeHpkdKdOCgr2uzeKjSjdLmf32K/8AZ
+ m7U0Ed0paNvLlDZbkHvTZmSOW7V5N/lyHy9 v8ORlR+dXLW5Es/kraTlMA4Vvm3DI5PepBOss0
+ YjeAyLCQYyuW57n3xzVOc+b3tbG9T2/teaetku 2n3f15nOzsGjh86YMnkgTqB9yQcge2atW5d
+ pJPNVZGbcz4XhG6Yx71VdbaNykkcjR7Fw4PDHPytV a6kuBEQJBIZZczsgxsJ5H0yK7VDmSS/r
+ +v8AI6qlLnSSW/loWIzNHqJ2wGOMxHzN3Pl+x9yeadvm VY4btWMUICMc8FuWH4ntS3n2iF5Xt
+ juLzEqMZyu0ZP4UtpPGIkjvJY95ZDAccMncn1PvQ7tcyX+Y qk5L94o3W3mrFkjcYE+2Rwqw3r
+ GwOSMZakvZE+wLIkrPsXaWV+BuIIH1xmrS3cRnjMUSFER9pwCT u7D8OazJZbCKyYW0oeR1IjD
+ cgLjv754zWNNNyV0zPD0n7SN4vXorfj1RPbJFNqiyFCsETPGiHqAw 4z6mty2sc2v2eAtNswy4
+ J5VRkj6msA3jtpaRRQP9pRhuUdcFeSfpVmKaeyty32pIs/KuQSX28hhj tg4NTWp1GtHYmthcR
+ J6aNuyTV/yNCJctcrIkquVLDzGyAD7VMZWOivbPcwxokiguQcOeuR7VSGo3 E1lFODEkyREOpX
+ +8ePy71NLeyyQwwBrc+c5ZW8vggCsJU5XV1/SM5UZtxbVrP8fmhyJPsYRlZZnQ sPl+UnOOBWm
+ rSxyTysrsqvh0XggheTWa0837mfckcaRNhQvLBiMVXlCxz3iolzJL5n7tRJ95QvJ/ M1Dg5bly
+ oycXG6Sfb/M2J5IDY2j7WMZhdWRTgglhg/TFZsdzG7/Z0Xc0LbAfUPzx9MVFDNdyafIk 15av8
+ 6gFUxwetVJLuK2uCgdBKpKSP/tbTkfUVdOha63ZFPDxUOTVy+f9Mv3cy29rEWaCaRxtjVVx lS
+ Rk8+lQ3c5js5oyvy+ZmCQHG+IEEsPoaxnhiFnBPHeASBBhXJJx6j9aZDe3YuUjhQ3LDOwlcgIP
+ Ue9dMcNomtbfI0WEcEpJN27ppFm4u9QDgxbPlOCSuQSev5dajeW+NySINyZVJSRkEZxkVHNKUn
+ ka WGVYGLgNu4JIGPxzUFpBcRFC8u6dkYFM/c55B9z2rdQSjeyOmc3Gm5NJX7f5pluVVW9lSW9
+ S3jjk ZIgVOcDkZx19KjkK+UrmaKIgESEJwHOCBj6VTa7K3EMXlIUCfMsq7jke/wBOKszzpc2i
+ iXbbAOzy REfMxXoM+pFVySTVzmk5qUVKL+VvyaL9rLEhWZYpzFMTKxD8DA4B/DmonKzhV81ow
+ ULIxbhj2/Md KwBeh3jyWWIRsAo7Z9cVde33WzSrcoYUKqNvcgcfzpvD8ru3Zm8KPLJOctXt1/
+ ployKlt5nkTIZH BMjtlY8cBT6sfWozvkgaSAkSmQgJnOwD5Sp9z1zSERGSzHnfNtj3J1yDnKn
+ /AGj2NTbYrbVCZCwb DeWOnGO/qQTnNLb1Oec5xk1zNW/ru2KLq7j1CO3kUJbxKxl3LnJXpz69
+ Kr3ZkS7t5YIQ7EBmXbwp H8HtU0rl7WFVJDxoVaU8qWHr9etTKXWwgxLHGOfNaQZw/b86Sai07
+ A6iSulZy0tr+lyzZpm+keSG W2mlckZbAVQOePY1SVIoLiKeQzGMx71ctlW4NWR9rNgkp+aSWQ
+ TFwOEwPmX8ahlkQ2wQxyIrI24u 2QpzlR7VlG/M/PQ5YSnKTUldS00/4JRk2/YfMS6XmMKUYnO
+ abb28jwWrNPHejawW3hPzEfxdqc8g WGJnUGT742jhR2BHqRmpJlu/sYLIsXmHMRjXaVx2/Gum
+ 7tY0rKrLljJJLpzW/TVfeijbxzPcJaZk gRUfaHyfcCqSR+feO9427LLuVBjIPf8ADFW1t1Qkp
+ cExfKZctzkjgA9uajuZJEt5YJAg2kCQFeck DC5roi9dOovYTm3GLS032t5ruZpk2agZIysqI5
+ 8wj+LnsaS6jl+yvMsqxQs4XY5OX78fSpIYmkae IzRQqCMswIAI6DP51AiRlBGHMs/mFozngdu
+ hrp0T9DmxKUPdWr6v+tCS4kYmyWORXYKWCqMbue38 6sCRTpxcTRyTEkyYHA5xgenrVaRvLuFO
+ 1WAUJuC8Zx0+tOjjWR1jlkS1cgMA3Yj1pOKshQpRp8s+ ezWun+a/yNpLactG9pL5yR7lcjoWx
+ xx6/wCFbKb5Iw8vyXQAPyjHTr+lchBJKJZwJFjZiX9MgdD/ AFrRg1OOGGAtKX2hg8hP3nPQj2
+ HpXFWoze2py42VZNe957a/MkDpPBFvv7dEKFIwoPILdP61cgtJ bdblYrqCRIpFiHy5+90NZNq
+ VnhdECl1faJAOCO5A9hWr5klhbpOgEqJ+7wf4snhj6nHSlVTXur7t DpqYia5Y3u+istya6S4h
+ uJIIjGY0QgELyWQDjPuaw5lultvlcCbzQsh28JuXhTWq2pG8SVLUqzRz FI8jqrcA/hg1hNNIy
+ l4b2CaKI4AUHLkHntRQhPqtTzpRlN++kpf18iCeRxbrHNbSkhRGxLdz0OPp WSscsUJaUvEQAO
+ mMcnj6966S7ufOhVZbZ1kSQkkHGQCOPqKwl2/bY/OLtCGLvHnk7T/9evQoSfLs JSbV+W1ul7/
+ dZnTNeNFbzKluqBZgWyvP+zz6ZzUzXl3EZHkaBi2cBYxxhgM9OlPuILq3PlWuyfzX V5AV3E7T
+ x/OrOWnlMV1ZOJtkjuVwNhB4Brz24WTsmvxO6nUpQs4QvHu7N/dYoSTiRproJlWlYAqc BvlwC
+ PQe1W7KSWOKNTPaW7RgQMkiZbkZzn3qNXtLWeASq0YaFlkLHKrJgkcf0pVFxENNkhCSzqg8 9C
+ ufMGeo+lTKzjawSq3hpv6f5k5trmCCI3SiS3SIq4UYYMTlVJ9apmK8DqL0eSGQxsQuOpzg/wC1
+ itIkuJiyyMokJJ3cSFRww9u2KYriW6HmyJmJVKKe27hgfU1EakrHbhq1SVJpfr+C7+ZS+y2jzz
+ /Z pGa383aELdFI4P50XKQpaRBVcDzfLAdsjkjj61ZkjhuLF2bFuMt5YPGUUjB/GpLSzmVXKXd
+ pHhw6 iRMjAHT61TqWV29iKtSUaampu/8AX9bGfHebrgxIDIomC8H/AGuMflV+5lFrqsTAQfZ9
+ hDiVd2/J +8PQZqjdtb3Ea4i2Y3LG6KAJAT19znimxRQPdRLdPJBbDCLFK3zg+59iPyqnGL1a+
+ RlUnT9m5TXy 6l26kIt5S6mTMhwsfy4KAY/Dn9Kz1xc2yKjCOUxESLLyGJIy34fnS3ZjEc6rIZ
+ WlYPIQeFJI/wAK fHbbyfsuN2HQhuSpyDg04JRhcKMKdPDqUtLvT/gq2w24triSC0bG4lmMmwY
+ MhHHFWLWKMX8qXCXE McUy/MZMbDgkbv5VVNtqNvbRtLudFfIC9cdx9TWtJcTXFlKgVI3aQCRS
+ vzO4GQ4/2QO1TOT5bJ3X f5hTs6Tho0+q6fn+Rbwz7rqBiwdhtQHDE4wCPRR3Hesclor17e6lY
+ pEpRsNgNkZU/hUTRPa2L7pD LEZsIQxAIPP61baGCS9SRrqG1x/BMCSejDP1NRGKj1ujehTVKi
+ 6knfpp5en+RHAL2c2+JhPkFW2A 7sHhjn2rZjuXs794IrW4DJKGVnbIwOpPtjtSabBDLJLMZ0d
+ pJgz+WdoBHOAPT2rZM3kKZLsxxq6E oxXJA6nPrziuWvWTly2uc7xidRwUbp7b3v6LcoXzSyvu
+ tzEkqnZGqr/A2PzPfNbCQusUJuIZv3YZ XkLfKp4AOPSs2RJrmzhbg74S5KDBBBBI/lWpDIjrZ
+ ySCXzZo3Yqz5BI6ce1cdV+4l2v/AF+ZtWqR 9lBJdXtv8/8AhjchV7e0iRNqiMlWZhnacZwaUs
+ 32i3DK5LwuGkXgZ7H6VB5T3Jji83ZuIwSevHU1 qOYIwY9rI4BIDH7hIHJ9h/WvLbs/M8rSMny
+ u79CnbQw3CwbW2ybD9/kA+mPWpt9qrojq4DIZDhsH J4FV4TCb4EMZMHcGXjtwKfI+6RGCjzNm
+ WyM4A7USu3qQ8M3O8l+NipaBJJYi+XdI2JBOeQDz9Kbc CCaCJF3uJ4TchlOAAABge2avwm2ba
+ pV03LyxPXNUL0R201vFFHJEOdjPyAgH8s1alep5ik4Kr1uZ MMNs7uQdkisNvPG053Gsm+2iJl
+ FjcSSkB7eRWGAg6j61o3UN1FDHPbzwgxPHHuZMg7vX6Zrnr2S5 knMRt7hzbuYo5I3wFHUqfUm
+ vVw8OaV0z1aVPmqJ328/+G/MezQeZcl5hHvl2xqxwWyAA34VKY4Ss gSSaeRkIWNZOZFAxn3Oe
+ c1VeaRlt5tsUMMhxh1yVBGAM+oq/9hhiiWKKC5M7hN0m8fIwPK/1P1rp k+VK7N6mJcY+9Udvl
+ /wH912YTxQp5dsjyRfu8yLI2SzAcEe3t71bt7jVPskKRRrMWPmSAL82BzV5 NPsY7tzFvlAZzE
+ 5bOAex9cnpTo7M3F48ipM67P3hjbblm9PbtVyrQa1/E7KTjVpvW6Wuv9Ir3N5e nT55HcRbrg7
+ WxjAwOntUomhvrJ3MLN8xMu3jL8BcelTNbuLefbNCJBMPvrkNxyAPyq6ZIksWa4tZ Cjy4Pl4X
+ aeOPqOKxcopKyHQdKnytxW/R2/D/AIJSVL0pPH50UUyFTKoTlcct+XeiXYtpcNBPDeP9 oBzCu
+ 0le34A1ckslkW8uFuCG83ZKcn5sjjHpVBfszaYvlH7ZuTa6wHaykEYpKSlqvyN6kaU7TSe+ 1v
+ zf/BMp7l2ZCQqqI5I5BjqScZHsDT7UyyaUImjMilt7FcAkL8tWWaKW/eNDDEFQhXddyk5BNaIk
+ Z55YkeJEO5lkC8bSMhfrXROVopWOuvg5WjHlt1/4bb82ZVnbWkNva25lmKMkjuWk+6SeM1cY2t
+ vJ bhZYXuVRvlIzjjj+tWo44IUiZ9sJWHGJOcfWnC4tLnTZVKpcZUHMQwxAOKwlNt31aOL6s9I
+ 3fL/X l+phxahJmOQRoFVc4C/eOeSPapkurZp43lFs5ZWCgRj92PQ/U9KW9dxO0Nv5SBZEEa4z
+ letTXTXD q5MMIlEu8ARhcjv+VbtRdtLXOmWEp3Vla/oVbeKK3vspI12PIcuqNyNo4qHLJZW2Z
+ o5IjGu4Hkrk jI6dauXCEkTQARylW3EdtoHB+tWYy7i2nmmtWUl96rFgYI6/hQ6ltdxTxnsVzL
+ 3umu+nbQozFRP5 CwSiMtu3bzztH3enc+9RLcyQPFdXGxp5l3sgGAOcDA7Ag4+tTCVmW9uIyAk
+ ZEa7udw/vCoDBMzQz KyS7omH1PqB6CrSj9o1UKUvebsn66mo95b3GnNFsks13lSJD9zJ4U+9U
+ Li2eOxMyieLy28pizZJG ck1Tfz3t4g0DhRuG89HJHWlW3mYuIZfN+UERHJyD+nFEKShsycLQk
+ knGdopu63X3/wCZZ85/s58p WKvIuQOoyeB+VV5rKb7YZXhkjZZywRuScEZzV1rfzLGdVtp7aS
+ OT+Ju4HyilslmUu80wxIoMu/kr Ic7R/wDWpKpZNomcpcjkpK35kN4Ynu3+zxqdoKy8fKWzkAD
+ sKqo0RDeZvtJScE5OBxnGBWtcRySF EaAzziEtOIgV3MtZEu0DYCIzNiY7xnHHNVSacbGkJNwU
+ NYrq9391/wA0WYXuHhiijMcjIoA3LkZJ ycjvVhLWV7QR+Yq3CFwXPTg96o/ZX3maDe8Xlneqn
+ 5iD0IqOWOSG2th5joDAXkVmJJyfvfShxTfu s8+vh0q0ZU/h9F+P/BQyWKKDy4/OWSeYbSpwQT
+ jqOOlUXiuIxEzxtDuGQHGenWtT7Cjo0rFmiDKE cH7q5xj61WvYTFfqJGJXy3CIWySO9dFOor2
+ vc2dSU1yOrdLo/wBGU5FMke7yEy2Wyi4BPXjj0q2b HzLT7XGXSFkZgm77uOn41LBaXEcW1yLh
+ iFK+WMcdhVwQNLa2xt0ljlEJhCO2Qx3HPH0zUzrWtZnN KrGEocsklfW7v99zDmgkRIudkkij5
+ Ccscen4VLHHJjO8SERs8Z6/JnB/OrVxZRqrRs5meINgjPX/ APVTrG2tklijIfa0JIJbqcHd+G
+ KuVVclzrrynGl7WesfSxBCyy2DkoyoB+5Qnk5P6mtQWUV1b2TW rneEIcFsjcTwf51Xdke1DR7
+ PLVAFA6/L1Oakyy20zRo1u0rBuT0GPvD0ArCbb1WhhONSdOEovlV9 P+CiFI5YrxYyzyuoDqgY
+ /dU85+tNMspke8iZZirmJQBlSCOuParUVs0aoySiNVh2O7jOT1BB7Crm l2s728s1zA0ZNwGQb
+ cZC5zUTqxinJkYiqlK8mn36X/Az0s5fPgYSJJCIXjOOrEdDTjAJtJjledky N7ZY/K68Bf8AGr
+ Lia/ZER08txuEqDCqc/d+vGKz47e7lmkS4heIOCyjphicD9KFJvVuzRwztOfvS St+RkYJExI+
+ 8CWAOPmHQ/Spr0rLpY8weY7Yw68BwMfN9B0q08KxQxpG6MxXLA88/wj8cGql9HOsc DiLYjuzs
+ uOIyB8qn9eK7FNSkjoxOIpzUPXt/wCO2kaC4fy/LZgjbWIyGz/Fisvyyk8RdigjHQjBb vitq3
+ eb7RGSkNvmNkMjr8vzdO3aqlzHi6toHcOIoz5jr0c47foK1hK0noYctJtpU7d3e7/y+5me2 Gd
+ S2A75YA+vYVMd0YlKxsZAWBJ5wMDI+oqJ4gm0eW8cQbcN/LD2zjnipXWUopiZtpGVXGSFzk545
+ rdu9ipSc2o29F3II7hpNNWRVUMoCEEc4Jxk1ahGx3DvCPKk2hSuc+/4VTZ7hRbCEoEU5YLH97B
+ z6 e9PRJGkcDLbe/qDzQ46M58PSqzvSl8tv8jQNw22NBH5MRjZ2XHJyMf8A16FuLeK1iAlaVlK
+ sNxO1 yO/sMcYrMaHyzCil2HIXqTgj+vNLBFLhEdssdwVyhC9Pu8jrUOlCxDp007S0t0v/AMM/
+ uRpxXUUc MhiURL9xR1Lbjy34VD+5O6CCPyZTIIyWOQxPcflVR40WLdMjxSGQbEJxlRye3rUQz
+ O0g2NGjOZWy cnIHBz6UKkt1/X+ZhUoJtSp38u33atl/UY2F3taC4G1Ww27jkDBNNhndrUySRR
+ 25B2EPHk4POfxq oEu2AnLGVGiJEnOMHqPrmrMMUd0lsJJGjRQURievPQ+9DSUEn0OWrKmoJz+
+ 9f8E6UxXE3l7bedoo Ytow2GLlsg/QVNCJtt4tzOvmtdRsXAwAuPm/Cp/PaZYVEMqp5Z3OrYG4
+ N7e1XJbaJrFgQxdSowDz IMckewrx5VLaNf1c2q16Sdp6Lytp+hmtBFc3F7AoMkTzMysOvAGBQ
+ v2iK1Vtiyl90nAwV2/0NF3b CJZJRcK1uJsEJkEKRj9etMtEktLW4lcPOgk8uJgeCh/qau946P
+ 8Ar+rFqKqJO910TWj+b2JPtZgW 3lLqsc0bSlWXO8njK+gHpStcQtbyOQFdJApI4LKAAW+maGK
+ TRM4TyYgrIyyc43dh6VWNjcCKOYwT LE6AYJ+/2BH9aIqHXRnRTjh1G7XK3+I9s3F55iSI0RBw
+ gHQHj9DSQx3dtYBQ2wAgNuGTG2eAT7jm pWiVdNVYg29JgiAHkqTwPc5zzWlFKsOoSQzxO/mMz
+ uc/xDp+VTKpZaL+kEsReKainbo7GMtpIyTm R0PlyKvTGQOQR7VFciS4lYW0BSNW3NK/ONw4Br
+ d+zQX7xzoZCZox8itjBIwB+VVFO3S9szxrBEBG iDhmBOOT3Poaca93fqbOvGpO6d5Lps16W3+
+ ZV+yeZJEv2CdyYyqMrAAjPBPr606KL7NfSzStl4Hz IRwCcenv2rWtdPEVn/rzAys+0OxO0AYG
+ fpnmsxnKpJi6t7oIEEjKuQzE4FKNXnuk9PmTShHERlfV duv46Ed1IL21jMKzRSlfkQv91Sec+
+ uAKvxQwCNwjklgWiycliThWB9B3FV5UmRprdpIpFJLkIuGX aMbc++aUxI3kq04Bh3KQDjaxxg
+ H8KUrcqSehM6UYQjGKsr/1qPkVlsV88xvIG+RwvyFc4JA/l6Vn K0kdxLJHcW0zF8RnZn5AOv5
+ fyqNvtMbIRHI8Rk+6ecqOv6mtC7aJrSKWKMwSJLznkMuMHj9K0UeX Te50wpSppQjeSl9xHLbk
+ WqZmMhAOHh+UMBgk/jV6CeXUbeaO2uIQN4ctKmQSOoGc4Bx0qOwDzqkM 3yQkboF7gLk7Se5NJ
+ FaOz27xK8McsGWI6Bycjp9Kyk1qpbo1pzpRTjUa5lt/ka0dzcW+l3Ml2wLz TiTKrgRZxiP6mt
+ VbyZYonkEKeXmNFKcjJ4B+tYqW7pbK0sivMXHynoDnOCPUVqPb5uLpnmVZPMCz AjgucEMPQAV
+ w1YwbIlhFOC1vr2NNHZoFa5ilTZlTtON7Zzx+FTvcNJ5hjOzfIo3OM/Ke/wCFMAkT Jdh94EKe
+ oPTH1q1LHILci6XY7ZI2jHAIHFec2rnAk1Llf4f5Nk8sFjPaMWYjDARbGI3ds/nUcUip qEIR1
+ /1LLIWGdrZ6H8KUbftJhgjM0ZZCSO2OlRSyY1DzY0CiQFkz/wAtE6Eisopv3SIPSVOW3mST Qp
+ NIB5ylSMDbkHJ+6ao6pc2lvpUYnmywATGeQ+4YFMlu7qJbiDMWxv3ocJjaF4xXOajOGtZTc28z
+ FZVWPBweO/vg11YfDuU1d6BDCSUlzPT+vQdPcSyambWK4hDh2ZlK+nP6npWXqbCNEkAkjuJYsl
+ 85 UKfvZH973plyrTtJFbRskrMm9mPPXrn071HsL3jQteW7qp8v5kJ3qf4h7V69Kmo2fY9CjRV
+ NuprZ bq2vrdfqadnHC1pbOZBcBg2xR3Cjj8upqKNorS682ZLqOB5QzSOxI3c4/A1lCO8SaOKF
+ gsRxJESP 4AeT9McGtC7R73KJBLNC77wUOMeg/Kh0/e1ejMYYHnquMp3Uvlp+RaE4lkjlALIpO
+ FXghugz9OtM mj8qAyMJhsmVXkSTCuCOmPfPFMjt5Q0TXFrPFFP2VsF2z1B7DpWhb2dsk0xlt7
+ gMAymKR+hH9fSo coQPXpLD0tnt26/NP9Bkd3FcXC/ZoHGwFULHIx7+p4PNVo3luY5gbG5a0mv
+ FcuDjbt64NXVhSyWW 4t4mcB1Aj79OT+RNOmeWOCOa3jeG1JLxs3IKkYH61Cav7q+/7zsjGLl7
+ iTvs3vf8wNvEJsySPty2 3BwHXru/A1nTW1ta2VvLHvjZ0LNh+M9h9TWxG9vskSW4jJU5KnqpC
+ 5C/iahYWz2StOrRuQSu88Ed ScexpQnJNbipynGcd7X/AK8yOGFpIbQ23kRz+UUdXQEq56Z49K
+ rPZyQSSz3cbsiKqkRnG5/4fwqe SK6WxthCyuoIMkyDhH9D9at/vCtyslrPKfNPmANx04FPnce
+ v9XNqdWcNJWcX9/36Ff7d/aMqI6xx Ha4nDJnLHjj0HFNWzhs7a3fyZplERiIjfGSx6/hmpbZL
+ f7GrGNuSiZXgknIzVhp0s9UcypIY1RlT PIYAcmocrPlgtOxhKUVJxpJ+hlXFjaNZNEs2y4jkA
+ XLHI+v1pk1sJGH2i1u3kTduCuBuyOo9s1YX zJ0haaCUwsyGIjALjPJz6dKgiW4bVnZSY1eQzO
+ JDncBwAPTJNbRlJLfYHOUYtb8uu/4aFk6cIo/k LQsQQpc5BzjNZ0rRkqr4uWCEEQnbgnp+gpz
+ zyxXxDzeQ0S+WEk5yxOc/hVW6uSZZJfL2TedtMqj5 VOOBj3rSnCd9XcIKUXy1Hfqv+C9x0890
+ LIOFjhSVWcIUGSOBTIZQIWnSVZOARCv3s+o9sVOJbkWp a4jWCWP+KRMhezKR61ltcWjxOkI3l
+ 2+RhxtQdsetbQjeNrfcOKiocj0T6rVP+vvNW0uGOXcZw+UX HDKf4h7VSniura/iMMqyuh2r5Q
+ 655GRiq0RlvrjMeLeJAUXPYYxj61K5uI7a3jvE3iL/AFbr8u4A 8/0pqHLLp6GdKmqc7Rad+mn
+ 6lqS5eCIqm9Z/OIfec5J5B/KofNEeoQXdxKHBRjtUYBJ4H4gc1CJR 9rldv9YV83cem4f05qae
+ 9lURxrai5TecPtBCjjFPktolua4mCp+4o6Pf/h+xctLi485jHukVv9WB 1I6daoPGkljcmV1W4
+ dlZFIOVA/xqSGWyaISeY0MiuqMNx46+nvUM4uCqS7oJnRwsgROq/wD1+lKC tPsZ0ptVG7NL5K
+ /zehfhaWOAXqlXckiRVHHIwuB29asQWt6tslwjQzPFGsWGTPzMT+lQTLd3d0Y1 hMMsbkwxqMf
+ u9vcdyPWtQXUwt7dIEUJIhlcgc5BAX8u9c1SUkla2p59R1U+aMVq9U7PT/IhW2P2O WCS2mLx3
+ CK2DjJHJ/DNNkFzJDPdYt4WSRY0MkYPyH7xq/JBcurJ5ojmy2+T+FiMEn8R09KrKUl32 7N5kY
+ K4wedzdF+tYxm3qRGnOMG1bzVtv0Bbq5jKIYEmiThVjQBip5Xn1qvIkLWUN1tktVwNqueRz jd
+ +BPNaW4HV5tP8ALbfEjh8dcheCKzLiS7fQ1iUwlJY1kjhKfvCF4PPoDzRBaq2hn9Wipxjbl792
+ n96+9L1Mq5EVru8yZLptjJhDjuM/jzVKRo2ztjltyIyAWbPTA7CtWYJNLa+SVknjhbY2PlZj7d
+ yR mqK2zz3rIwChFIOD3xx+dejSmrXZ6mBr0uWUZy+Hpf8ANbFhPLngi+0yI7mItiJduARjB9y
+ cVHm0 t9Mtfs63DTq3lyo752ueRx6Y7U6GMtEv2sqilMxKowSAc/0qKXbPMbyaQQeajOIuc5PG
+ cAfjSS1t fQ5Hg4ydoS07a6/gX7v99by3ShirLkBTgY9R+VRxXBh8u6inJEh2APyEyfun3NZst
+ 5NFDH5U6yDf ukbb8o4+7j3qAyXLmIACNkBYqVwFx1z7jrTjQbjZ7GijWlScJfj1Ld5DLNOjQw
+ zR4J3Kp/j9MD0H Wm7oRAkrPNFPGpHzvkE9j9M1JYSlQ1yrgySEsqHnPBGfw71EsKC6j8qaNZl
+ gK7XUkEEdfQ1Sdvdf Q56dSq4OLWi8nr5PfQqWgh8xEnma4d4z5jJwEY5wOR1prsoji8xAHKAY
+ 6FCTwD71opFFZwRtJLEW GSQqcleq/wD66wiEvLyWSeQRsvzI3bHpj16Yrem+aTfQywmIxKlKS
+ b5f72qfyJyRb24QRPK8bkTf NkcjPSq2+MW0iKj+VIysjM2T8vUZ/GhomEy7ZlmcHJRM5bA69O
+ g71JCY2aS3kPEkm9SOMe341tZJ XM1FRnOa+etvwK89074aJkFukwBjYbmORyM055fMkSOVUIi
+ JBKttBU9v0qwqqmpxFoluJGjO9FAG DzyffHNRm1EUfnTwyiAnCKGwX4659BxmnzQ0Oe2G5k38
+ kna/4lKJRNI5iljK5wWJ4XP8PTrTBFho ZGE7gMcBWwpI7HjmtRYZora4jUW48pwrIke0u7dDU
+ OEtmiWVHklti0b7ThXB6kD15xVe17GUqtSS tKN/JP8AXT9SsYrhhHEyPnaNmFx8uc5+nvUQjZ
+ ZJHaKRDuyqsfu5NaE92ViEETboADs7sAeSCfao Y4ITbs84mCK642MAWHeqU3a7R0bU+aUfkv6
+ RQnVLvUINs3kRQlgTIxPHpx3rQEMDoLhJVMCtkr3y ein+dVldBcylFDJvLAMoOSO/0pI5YpXi
+ neKRlX/WbDtDHJ5pyUraHD7OpG9uutl/Wn3sluJIhZok Uu/BBTbwozyRj606NHmvpTHZy7PMC
+ yIpGUB7fWnWEjW2n3Ea+RLiUOPMjDYAppdUu47i78xi+XUx nYGJ7cdajVXS/r8jKNOpGjJJW+
+ e/l/wzOus7MJp0ixeZIxw2N2c7W5I+g6094SmomeKO5Z5cyQLv yNgGCMfjUtlfMdyOm0ySAEg
+ YwRk4q5JeSyzK3lqfMR3aNBgxkAjb7fhXiTnUU3dBPFTVZwcLrvff 8TMgVQkRW3uJo3UEgtkf
+ 72D2FLDFcG6SVpo48IymNh95ugOOmMc1FC1wLHTGETRNFblgG5yc4H4Y qUGVbUywXEE+2WMBA
+ uTjNaSvqd8ZVG37y107/wDDF17aRNPYD5jHLHGXPRuOWqjZ4F6Uk86YEAjD /Ko54+uOaciP/w
+ AJOss8jtAyOxKkhVODwaWFbC1tLAQ+chlx5ckj5HOQT74zU7Raet/6/Q5atWUF yuV79uhKkd3
+ DY4DoEaPzPNZcjjgfzojSN5Lee6V43+6zbsAuQRinmS2geO3aVihVgYyclcj/AB5q m9kiWFqs
+ cdwU8wOCzk5H8X45NJa76XNIu6UZOz7liwSYR+T8wuQVBUdQQcEfXFab2rpGXkEca4yy uuSpz
+ gfzzVd5UmuoYoYpPtETFpCD1Yj730xmoJrqYwRea4QIyhdw6oTzn1PpWUlOUr7FShN1NF+r Jp
+ p3j84SsjvGGWRlGAC2B09+tZkkVrEGRZ42iwpi29SByM+vNaBhvjqaNGiz2sm4ZC5x2GfpUaW8
+ 1u4t5/JjWKMIJ2ThiOeK0g4xWjO2lRp0k0nr6r8inOQs0ki3CCTC+aSONwPI/GrD31o09y80DO
+ BI AXj4AJGQD78Yqd7WO5lXzHQs0ZZ9gxtIOdp96fmHy41Elqm8MQGT7xxwf8KfNFpXX9f0jpl
+ UozUV JO/zX4lL7SbnSwY1VZjggY+6CckH3GM07yppEl/d+aZW8yNsfKB3OPSpbO4MUojiaDbI
+ A6ErnAHH 8+Knm1HyVuEmAkmjuFAEY24GOR+FD5lK0UQo1qErU43e+u5Xgto7j7T5k4VlZY9y8
+ cdTj69KsJdP taG2lWN3kEr71z5RXov4+1QEW7SuIHKTbTkseGwMjj3qSWayfSYEMiJclFM+OC
+ xJz+lTK8nrr+hb nKrK87y+W3quxNb+fPPubduddxQ9UI/qauJJILyN5FcxFOWz39D74qOGIqZ
+ 5/MQr5pVUxyMjufQC rd1ZgWg3b9kUgSMqeCMZJNc85x5rEPGUvhlK99Lf5GiRJFEHO8RxsIyz
+ HPLEc/hT5biYWz+a+6Np C59QAfX0rBgRZFZVeUxqwBLOcepH196vXepNDYZihKo8wLhgCYicf
+ Kf51zOi+ZLc5IR1Vtdf61LI uWEe8kyRvESfLOCrA8DPvQ1zcvaxSJJHFLsZY2dcqysRkgemeK
+ r3NwEucMyy797L5Y2jAwMfnyKy p75lSBDGxZVXcB/snk+w9qqnR5rNI6KaVVXiti3dXl9BaTe
+ W8IAk2zl484YjhR6Vg3OqXQCRsUUR 7UbcuSSfm/pinRxFtWcPDdSRyKzyrv754qrezRwxsHgb
+ Nwu/LH7pz0r0aNGCaVrs74UKDajKF330 X3EjMY2vLmOZJy7bmRDyoPb8BVZd8k1zHGYSyzHyW
+ VeXXHHNPvmlNkPKeApLlnVV5LDAxn61Xtp5 I7+2id41jcHeuMEBff1rphH3L9f8hKpTnTc73c
+ f62tr+BMHYWsAuG3ThTDGqccnn9D1roEuZEt5g GR8yIr4XuOCR6CuVjn23IaaJg2QSM8hs8H8
+ jWw7izM0VtOl0FOAByQf4s5/Gs69O7SK9hUbUJNWe 2/X7zciYT3TW85aOBJcqSeSo6Ee2aVWt
+ 767ZEaWUTK8uxW+Yds5/DNY0d3NJbiLz4IkMTOSV5+U8 Vr2N5i1imgt1ZlQwhlAGd3f8O9cVS
+ lKGqNlhPYvnvbsr9e+xJa3VqscEUayxlIvLZ5GyMnpn9aYt y0dqq3iCZVUoNgwGz90j6d6zry
+ 3lS8cqymAMhyv8ZQYGPqTVqG+hEszzrtd3aSNW/hA6j8T0pulG 3NHU9OGHp814av1Kaox0mRn
+ iaB0b5pHOQCOin3NaOnqbiPZclYFQSZDDJOB29BSkrd2cdoqlWkTe M9VwclT6sexq7a2lrKsm
+ Le5Sdd3yGT+A8mlUqrld9wdSMYrm3T+75sq2e7YjGOS7gG1EihOC5x97 PtSzRCKSVkjui7Ocn
+ zPvYxVcLcWgVYklKgBh/vZzU8duLi7tnuZJYTKhlPzkdKUtG5X0DESnTi56 Wfa/5Ji2V3BPMI
+ Zyo3nzC6cKOwH48UOIJLZxIXgUPuxM2SATg/rTJI445MleH2EsvHQ/4moL82ro jSvJI6kRNEh
+ wST3pKMXK6urnH7OCkpRbjf8Ar+rlt4cBjch4LaIeWjbsD5iMkfTp+NZv2WRvOXz/ ACssuS2f
+ kAP3fqeKsPfLaXEttK4CKxYGQ7s4GSP5VHDBazQGdpn8zeCYtxyR3P4VcXKCuzGvOpQe qtf7v
+ +H730KitZSah5kqTTu7F9obG054X3PFVpjE07QpulM8hdwvTPYj2qW5igi1jz/NCwgnao6q D6
+ +vNZ7Yhu9rSiMb0eRWHzAjrz2rrgk9U3sdNS2s03ey8/yBp7pdyPE5hYNnfyckcc/WqSw7ba0d
+ WQlCBIoGCAT09yTWhPHKYJ3U+egnJUqOpPU/SqmJkZQmwNghht6HHGfc9q3g01oRS5J0vaXs/L
+ b5 ixTyQFY0yrSMRhgMBwOn5U15JXaLfDNIWwqEdMHqf0oFpN5ClsfvVEir1OORmpGuStpbRtE
+ 2Y4yu fY1Wl7rVlTV7ONm32Vv1uS3VwpupYo4xGjlipI/h45z6cUttJI9kZ3eNwrrGCBwPUH3x
+ UMZZYoXh ULGCysZOS/HQelSxvCmnMu4LKrKGP8Jbk5xUOK5bJGVanOpRUUrR+/8AHUZLCU1MO
+ kTGMneExkbj 0rQF8ZtqzoE2D96AuPmBqratb7fnMpCsEDbuOeoq99h8lk8mRG2IRISMh/8Aa/
+ Cs6jW0txLbklHX +uupYslukl8zaSjLvZmPKkZyP1rQkgMttcJgTbXC7Ivlbpk49qpOk76SjzT
+ o1tlVAQYOWPr+ArQl jdLuY4csx+cqcAHHI+vSuGb9699TKVL3lNNL0/pMtfYJobBLhJ1DFirh
+ 8nlsc/QiqzWrjUvK3Kxj VsKg5bA6/hTysUVpBteRnO6VAXJAjyAQfU56HtSCG/uppoXUpIJGU
+ soxhSMkfXgVzxbV22cULwTb ehVMV5aTwlXCMIWR55BnzM85/XFUJIrtQC0TBYSpAPUKOo/Dqf
+ WtcXCo8AkuEEJXEYfnIJ6/nUrg LctJI32gq58x1OFZivAA9/6VqqjT1RFWFnfkun/X9bnOIsc
+ tvJE48uVbhRFKpwoUjJbHoT0rQnlk jlgVogMRMLiRVADknt6YqSWXyNLjhuERhtJfYuCrZG3P
+ tVCETSaw0jW8qouYrhmOVVz0I9Oa2Xva vZXKo4eLhzVFbsuv4W/UzJIXFhiKGeMpIoRpG3bQP
+ vmoY/PjVwqLcxSAyoSmc+4z/CO4roGit/IX zEuA7OFdA+MsOCP8aqXPlKoVIpo4gQjvu4zt4U
+ enp9a6IV76WNKdWHa/9etzHlillZVkaGKNxmRg mBkjOPqccUSpcXNnKRKmPNC4C8gEev4Vdku
+ VuJYoUs5mEcWwLu5B9/cUW81tbARTho9y7gCfvAHj 8a255Jba/IVSLuuZO/bRmR9knc/ZiyyS
+ JOdskXyqNw+7+OOPxq3mYSgQtA1ycsqrH83pz7HrioZ5 Yo5YZYSWjdd2QTkEHjPvV2CVGigaQ
+ iHzLd3DDrkdOfQf1rSpKXKm0YYmPLG3R9H363Kd40kl0luB 5rIrAbR/rF4JH4VSWCVr+OK2KK
+ fKI+cZ9+aleORbRriWbaBKFGTyCw5/Co5TJDcsWzsiOzKnB3AZ 61rBWVkJq9JxhZeWv6/8Agk
+ tLqIK5xbrkKSy53H0zUohQxmGIlZJGeTy2OSMdFz7c1CgmMqmOQ3B CFQoBbGeSPrzTrbZEVkl
+ Ei8mSJMndx0z7Hn64rR3tuYqNorRc62t19VroSuVjsxM7h53OH2nAIYc EenSpnmigEEkEwbYC
+ HVxuCk9OPfmkuUMaedG8UsLhWKMOwPaq5uHfUd8USH5SChQEE54P4ZqEuZX I9i3T5pLRd3+A2
+ 6+zo8ioZZxLhwVboF9c/jU8zW0ljcS27FZzeY8p/mYll/lSxp5qzJPPBGkaurD HzBvwHY4rPF
+ qJJYXWVERoy5kOcHtkfjVJJ7vY568Iyak29Oiu0/68iK42SSQQRskrSMDEE4YDGCD U8Zht7i4
+ dYpCY5Qq7myNuDuOP1p0Fi/2xZJIzOIQXPl8EMDwM/rUEjxXOJkV4mE4WQZ4cv3Hp0rW 6fu9D
+ n9qr8s/h8tH+hoW8yNZQIixAxzb5cr97j+XqKylkS2kRoVDJjnI3BOTwQR19Kmc2YnIBkiQ Ag
+ nOdxzgEVItudnkNJD5O8sHxznPTNTFRje/UyhGjScoq+u9+3luQQSzTzRTsUZI8/ukGGc9PSod
+ 86XCB1TeBu+5yKsvCrahKWmhgbdljzznkkADtVXyUklt40kMjSDaoxyK1TidTlTo8yUtGtdHb/
+ I7 KZI547YWjkwqMSuD1Zj8vP6VqssEZkbf/pEM8aICezDBB96rGVjbt5NuU3uHRRjlFIJ/HOa
+ v3M8F xePKAoUPuIA+9/tfTH614NSUnZdP+G/4YyVGTgktUZDy3lvEyBoikbrsi25dlB55rUik
+ jCTT3Dww xGQqMLjBPOPqOKcl/G92y5g2RnahKjIVuv1qaeJCdiFCigRxjH3+cg/XrzWc530kr
+ ET1fK4cluv9 aL7jlLm5S7uYYhFNbzSFjIxfgfQf561pWdur2ygyK3l7vLQ9Rg5/+vVtklmgdU
+ SF/MPmsVj5DA4V fx9KuyRoLxpruJg4Zjsj+UISANp9zW1SuuVRWh11Z3UYr7r3f4laOeFGKFo
+ /PJSR3dcgk5zj2Aqv K6vEXjkB/csS+flBBw2BRbxo8Kuq/vQ6DLc9G6Vca1Z5JVCMJHVpI48d
+ FH3gazvGMgoctOreKt5M SGOCCB5oFmuX34wrcnjrVULMQoEDrGQrOz8jr0/GrouEWJhCY7cyg
+ bUc5JOflYe3WoXt1huZpJbp ZgCA0anbkjuPbmlGTTd9zspTVHmc939/+RLHte4QhbgpbP5ZVZ
+ MHJ+YD8qkd3muZmnkiWFrhDFkf e4zx9TxVa2nsYplZ5Tl0aSbLcb/uj9KfLfu11JHB5aiJ9ql
+ lztGOfxFS4S5tEZulJ81S11br/wAE ZPO0VmXWe2WSb94yleVx/CffmpUngkgkhupreLbIGzsw
+ QF6dOxFQu8UsqxMEMbByCR6Y2/map3Uq q15mMTM+C0gHy5A4wPQGrhTUtOp00cPTqxUZNprtb
+ 82NvD9rmYwosaTFpom6AAYG3j35qaxsgbjy bpjJ8xbcDyMDkH1NVILtUjt51uIXlj6psyD13c
+ e/9Kn8u8igF7HIGjbaJABz+H1zW8lJR5b2NJTn y8rdl0et2/X/AIIs80iG1Ft5bM0Ls5xnDA/
+ 4VDEWkjjnZUe6Y4UqoAC5yVI7nHOatXRaVGgkUK0b 4XYuNw7j8arXFvKliHmuYgQp8tQuCyn7
+ x/DilBqyT3MaTb5eeyu/X77FporWZlkM8kaTuHjw5G0e /v71Y811uZUguhNK0mDzkDHbB71QV
+ 7a3sxAtzFlMpIWyfnOMMPRcdqLS2E2ni4hV5JM+XuRuGG77 w+gqXHS72N/rLUZKUrJ6K60/H/
+ M6hZokuZ5GBjbzAjg9M9z7cVkzTTG4bfE00bzqGCfwtjgH3Ip1 iuHZ5SfJeQsrOcjJyqt7ioD
+ BcQIJGuEH2dgrErkEjJJ+uDXNCEYyauYYeb55LnUn/W3/AA5o+RaR MbgmVEyFBd87M8BT7nrW
+ bOBb6nKxbdDHEYyzfxluSRWtJJbixiIuIn8yBiFxn8frXP8AmWsixRMZ ZJJXIR1fhlH17mqoX
+ d27mlGvDmcpN9n/AMC9i1a3cK3ELtdwmMpuYY5JGdoz7is+/cSyF7lHDK6t C2OMY5U+tS3Rs4
+ 7SWRFEOZo2jBHVfT9KT7R5UjSXEBmiJdkQf8sznO1vXiuiEUpcyQoxXtPacvNf rez+RWtRFKy
+ DbJ++yevTaOR9TSSSZ1CzxbEMSXKHG4Dvz+tEaXDndbRkRtC24d1YZ49s1C08Spa2 13L5bm3K
+ tJtJ2nsDjn0re15aGtShGad3p+RJC1xNdD7MhdlyGOAck529R37UguDKVdnCSrDhjjGS D396l
+ ikf7T/elUMrRRjaQ2MZpUS0hJN8rJlVAUHG3jofc9Qad0m9DuwmJ5INJbeWvz20FgvYjEkU hS
+ N9gUOVz17VpSanvtTGhWAqVUKV/M/gapR3EEeoRLPaFoSCxAwGjYdMn24p0NvE8p3zwymZfMYI
+ MMp7jPbNZSUL3aO2lXjGV5wemqe5pRW0kgtgk3mykM5wTjAHJ+n9akspLeJgJ7WWQzIph5GfLz
+ zn 1571DCzxLFGiMo3GMHP+rycgH602YyQ3JuJmyrZxt4AxjIHpXO05XTM4znKMrt67Lr+hqwN
+ ZHUyq szNIXkLhuBt6fQZq2Gu4YYrqcrmTKoFGPkP3vxrLfybkrboCrAFlKnoO4P4c04vdy5Mi
+ P9nCnafQ AYx9SK55U0/+CH7ru0/Pf5O5dMd1O8ciypDGqeXECueD3P0x1oiZIJJTOTGXIKSOM
+ hF6EY96qRSx iHbIk8ULKJYSX7AY5+hpbxvJtt3nJcs0m/avB4ABH454qeVt8pz1qslU5H17b/
+ ft95JK0FvctdLI s9s4+RAeSBwKquwhQoDG8sC53EZ3Y5NV7ET/AG2KBIzHald8fm/NgLk5zU8
+ TyOQ0jxtMclDt4ZM/ NxW3Jyuz1/r+rluEVU/eavvf9CvLdQ3YcztDHayMGj3J8z47g+gI5qGF
+ trSMkiyF2ZnYDAVgPu49 6Y9xJdwLGYY0hd8CQKMIP4R7dPxouFke1eNXTz0ONsa4IBHIPqfet
+ 1FJcux0xUJRUUku/axXnuVk ESKhMrruK4BPrSpCZ7NwGjSWOVVYyjlmJ4P0x2qaJg9xEqWssj
+ jJj56L/Hn1xUweOGRMGOFGVvsz PyCv94+p96qUuXRI86vWqQl7NW022f8AmV5oJFkaSSaPD58
+ xEJGJM8L0709IEYebHHI0k0pkMW/5 htPAqqxLQTfMzNDKkYPbB6k+p96jYXsWqF7csF8tgoIz
+ jPT86qzatcVSM6q3Sf3J/cWb5MRSbZMS +bhuf4c549BVSR7a51aJVyokzI5J4G0fdqeNp3soy
+ 0ZaeBypXH3R2VvUnmqPkSAqpw87/vQiA5AH UdKunGys2PD0Zyp8t7W8/wCtPUlS4ZbpVaEMjp
+ yhAwM9D+FT3FrEqiFGEczOCzNyvTAH49qjsY4L a/cSrLHIM7GdshB3BHer32d5PJjUiVUdR5i
+ nqxBK0VJ8stNDpxVepGom042+4SzUEm3l8s+YwcAD mPYOAfU1ftIzdXjLBHKWUKuc8EfeIx6n
+ FNgWRLyJHjUGFCPNxw4P8X4Go7gSxW8zo5BadTGyErx0 I+tc0nzNruFJybdOT1e39I6BIlJub
+ tQCDL8kJ5wCMdOmRUtzMgmUM6nETngYOeBz+FZCSx/aLcNP 5bksdjE9M4zVzcxtmKOk6LMFDq
+ OoJ5FcLhZ3f9dDmp3Tbldtd9B0U88jLiNFcIyoWXIZepNMaG4Y ySxziNhGq7T/AA56598UrCK
+ e9LDfDJggLu6gHJxU8ySeT58Q8yNiHVR/CvRgfU0XszWUnG/Ilr36 /mZdvZOuoxfb9vlvE7Dj
+ AXb/AI8VoiH91lg7tsbeUOB5mMgflVSZ786nEsSh45EaRl25MfGAtNju Jfs0iwQT+e7jYpbqC
+ OW/LNXNTlZtnJX9p7Vc7V/69BkEjfabXeyBWtyGjcZJJycn6VWNsIz+6Z4A sTKpkbIcdd9WFi
+ e3817WGSdZHBU5zs2jhST65NUDai5kYsLi2JkJIZ84X0raNrt30/roazVOMuZS Vvl+RWllRZP
+ OkjuLkH5nMb4CNtxj696ogyxtCyyrMY1I8vB+b+6/5n9KdsiiDAiQqSXjJfjaOCp9 +nNNWVUn
+ 82AEgoIkHXO/rz7Gu+KSR0XpuDcv+A/68rFzy7i6W33lfMRTE5jG3eW/i/DGKjaOOMGT 5WkaN
+ izMMgMo4WqlwsNo+ySdhKn7r5SfvDp+eagtUje1it0huXKE+YQ4O4nPIz09KFD3b30PPhQc bS
+ T930/zewNfK0tsrpDbyCArHlOCfTGOtNAmNrHvVtm5ZFZRgKBwQfr1qOaAbkIUIxUuC2DjA4H6
+ YqRbe9aw3q5YqygAfwg84Pvn9K6LQS0NsRyxaVl6/wBXGxXckKF2jMtqhyAQDuJOFP0qnPbM08
+ qN unkSYybl6Px6fWti4lmuIJYvLWRjJgbFABI5LD2GKxpwGkLhZPNYA/KegPXNOi7u9rM4IYe
+ lNup9 rtr/AMALW48m+juSpbKFnRRjk8U27lE01tIytEFQg543sT2pk8KJeNHFL+7VtqM2Tkev
+ SogigQ5Y HzBlVzynOOa3UY3Uh1qVJT5ndPvqa1kpTVP9JglEP3QT90D3/GqbzzPPIpeKBTMok
+ O3G1vwp8t3d W0vU8t8rEZUnufoav2yzyyS3EqwlPNAeMIMlwPlNYy918zOLEy9lJyk07rT+rW
+ uYk0YnC7i6OOcZ 5PYk0rW9w9rbkZELuSVXqDnGB6ZqeRnNjFIIJo5GygyB68dB1NTIZ5bm23I
+ YiJAr8cFsHpWrm0gx FWlGmnTd/wAH+enyJLCUPOsAhmGYpHUh+vv9OKrXLNAIEIiETkOp28se
+ uP50LiOWKVRJDtTZknIA yd6n3NQm633yTImzy1ICvyApG38+9Qoe9dLT9TgeFhKo5Rimrde/r
+ f8AzK8zN/aC7kxFtwF287Ty OfX3qL95b3Ufl7jGgwyvyTkfzq55LTJBGJEdURotwGd+Odw9qh
+ AhFo6sxxkGPI5IPfPtW8ZLY7Ka hiXZ9PXQquqGSMsrFFX5RvwSPfikdlR5iwG4t0A6e1NYmYR
+ naoDFV3dOe3SpQzBJ5JCFuCSWTHIP Q1tsbQnC/Lf5/wDDWPQtNDQlYLWKXylYB2kO4tk9R6Ad
+ x3rb33H294hYFkXJOFGcA4J+mKxgbZb2 FJXeJMPvKvjLdVxT4b2W9iNul7EkjOhdyCPn54+hx
+ 0r5irBzblb8zzoqUqsny2XXfT5rcubYRcma KEJAq4hUgFtpHf1NZ++ZgqxywSLCrLhRzkdT+G
+ a1riNra1ZFUCVz5sYIyFC4JU/zqo88F156GLb5 7faCyHG3HBH49amnK6va6/r/AIJ6VGEpWqc
+ l0UJZILazjiO8LLGB9/DbgflxS2cErXU7TSnypDgu zEgtggH+lTy6YssMlymfs/nI6knOBnDC
+ l+yLBb4iEgj83awZskFWyPzBrb2kOWyerN6+LjCm1Cbu /wCvkKCLO0VjazRjMZUOwyoB6n15q
+ S7iM32gxXaRusx8pv8AZ7j8802WK9bVhNdwvJFGxSIKMBcn Jz69qe2pKjzR2yAsvzK7AEZ4rN
+ KV046s5uXmanC0pddV+pUWy1C6uLV0ubYqi7Rtj5UN/wDqpjWl u8CxPL5kxQtgNzIecYH4VDc
+ G8eXz5IZgsk2+NYjtIAGMce9aDSRC38+9gkhEcq74hw4kxwM+ntWz lNW1+6w1iakJxnBL0Vr3
+ 69v0CygP9iWgQRxOsLKzSpu5Lc1bW2CIskUOUVdsink7jwvP86iHnGAF 3RcI/nLjmN+oBx3NU
+ bYXLQxtvMathgWzxt6KfU+9YNOV3cqLlUp80ZW1ZYuvOit7uFogjpMqxEgc 4A4/GoTA10s8Mc
+ sOxTmIbeWXtz35zzTxeTx/vXw5kbeARnZ25qKO9LTRyHyxb28PlNIoxlyev0rW Kmloj0KM5uE
+ Ukrp/1pZjzBJ5EFuq2wkWJfMJiB2ZOSD74qERySySwRyBY8jyvbI3EH344q1DOzNM YADdGaME
+ HngfepbYfZ7mZLd0mkE+5Vxksq5/xx+FLmaucs5TTajFX/J9zNmayiaCRorrMkL7C0nQ DkE/j
+ Tmkku4FgmAiIi2O56bsdvQdK0J4DNBZXGxGtxbuT8o4w2R+fSs+7lLXUd3FC4Z13bD/AA7h jn
+ 1xWsJKVu+pNGfPCL5m2m+vUhtrCK3Yfa5I5WkbCL64U5P51qQzx/2WqxAKrIXmVTgq44X6A88V
+ m232OWbfcSMAjhFfdgBcYx9T61qmN/7OjgjMewhg42/NkH5eaVd3l724YhRVZObb+Vv8/wBCs6
+ Sr aq33wyAxBe69WP4ULeGd5EmxBHu8x2boTjgfjWcv22O+EKAxyLgx7+flHOKt3PmTBJg8cjr
+ KMIgx 17Y/M1TppOzOmeEhF2l/l9//AA5PeQt5ANy6QKyqAcYHOOn0pzzwwIYIoCZokdY5cArg
+ 9eKpedNI k0V66FTmVTt4baeMexoSWeVJEysU81wJUV/4F/uml7N2V+n3Cq0nHlUlon8v+CRl4
+ Lixjtw6QgBU TzBkkA/e/OiK18rzkeQl/MVJPrg81B9kEUJuZgS8c6rJ6Nn09B0pJzLJffuxKt
+ wsmwc9d3St0t1F 6Fx55XUJWj30sSym3+yjyLqPzWUZUoQc56A4qnA4aIJfeUjiYSRlkyVC9Qe
+ O9aCwyrBcI5gLLIDK ioA4IOOtQzwILzDkukAxlemDzVRkmrXO2tCpiaVpzbt979NP67hmS4uJ
+ Ag8t5nMjjGCGHIGfpVq5 hurpbePy/nkVn3kcA44H1qta7JIp3kjlFj5u4SbueOBzVRoZnvJBC
+ 0rqn3WBPODU8t5drHDTpt1P ity7f1c1Y5lijRZ0KlEVVZhnC5+YN6k+taFq9hE91LGytD5yhS
+ edqkcg1myxrMZWuI5IkL+YjE9F /iz684FJm3NkrQspkMgIXscA4zWMoqS9fuOudpRs01f7vvL
+ E1wV1F/IlWOAY2Mwzu4xmnCRbaG3g m3TuylMH+L/a57DvVaWeKWztCWWIhAhDDgZ6/liqkhMU
+ bbMkn7ztz5ZAOV+pqowTSWx0SlGq4x5e Xz/qxsxh2mhJuoZpY0kQtGMKfQ4/SlElwmnOZ3Bif
+ MhbJG2QDCj6e1Z6uY7KVWR5LffGx28MB3Gf rUlw9nM6WtqJsFt+CxI47fiah09fL+mRKnOE1z
+ q6XWy/P/gFiG4VGj89JI5Ik27mORjuMfka0yk1 ywNuoYzQl2kxxuzjgduK5q4tpLYw7pcyL0U
+ 88dyc8d6Yst1FexRp5imNCEQHkoeT0qpUFL3os2lR U17alv8A10f+Zr3LzWiCORxvCMin0BA4
+ +ppbGNomVJd6rErbHc8EY5H49Kjt5XMSNbp+7WE7DMN+ PQH1PXmq0ySXt+Fg37mVfKAbA2dST
+ 9DUKN04vQ4788XGbStu+33afmTz20i+Q8cy20Dw73LgkDIw F+tPsfmt4y6NJLbxnyiD99e+fX
+ iq0Ya5uLnzpFCPIpB7fL2FadonmzS+ZC6sgZTsOAxxzj9KVSXL Gz/r+tjKrVpxj7O7l+X6GXF
+ dtbFJYSPPRCE3Lu/dk8n6+9SlxLqb23mL50T+XE+Pl2kcnH+eabbx SJbbXspkneHCBjn7vapo
+ WZLuFvKEcrRfMxGQ5PQj8eKqXLq1v/X/AARfu+VuH+f3+RVhEMazr9si hMlwPlcE7doIzVu4S
+ WKyuHkYTvK6lTGcbRioRDIrmL7N9rZVIbAH3yOKmaJJYYQ9yiTouyVDn5nb GMegxQ37ydyqUX
+ CanLbraz0+SuvvIALdLJEkS4h2cFi/3t3J+tW7KIygxIphlEZ3O4ySRyPoMdqs XMFuJooIMTS
+ 7GZxn0xgc96rRSlJSzK0MJYu5Y8pg8KazcueN0VUvODcE0/P/AC6ilY0uIYbgxyXM kTYwPv5H
+ 9BSQSRyXSwqkkCkBgzHgqOAaXyJbh7meK0nBkcPHMSMDsQPaksILS2keVpQpBZJfMOcc cEego
+ fLyvuKpD922n926ZJcBQyKS6hx2PRV7Z9ScUk63tzbJFJiEGJn2FeQV5ozaMIEVJpEIBLb+ GG
+ ecfTvT4ikmpXSxXUJiklGwk5JYggY9AaSulfsdOGlKnG838Ouv/Duw2GLzkgnZQjSqzlj0PPb0
+ FbVhKLiOZIU4SVURfXJyT+GKyXVo7aSCSVY5Y/nXjpt/hqGzvIZXd4IZ1Vn4G/7g5JB9T71FSD
+ nB sdXnq0m5v0N4+fBfSm0MboQxAKZLDuR7daswXH2q1+cCCLpF/uHqDjqeM5rPuJUZ5tysu2N
+ mhKNg NHgZ/P1qrC15HcI0Vu7RNjcuQeTxx9BXN7LmjfqZypJ0+a2q79S+IWUyXErFmRkRdvAO
+ 7v8ATpVG 5+1W9uzLG4uEUur4+UADBFXJdR8qcrHPbkKwieMpksW6H8KxLu4dLkW8CzNOA5Ys2
+ QMHpj6VpRhJ y1RhSd6n7yK+f9f5kVvcI2mvcKZDFhQED87wOufQHrVS6uk+3w3CSGSXb5rYYh
+ Wzxj2oN3HGFgO2 IKu5Dt4Hcg+pPaormUyfNF5CIkbIfl6Z5Cn3r0IU/eu1v+R0RhJv3oavzaV
+ umyIFluHmCTL5jfMw 2rjGeB+Ga0IZoY7QSEok6tzuGVwODgfXisqERxQC5lkMUZGI8knecgkU
+ 24cyzTJG28FywAX7o6kH 8a3nTUnboLEUlVkoQXurdp/g+5LudmVmRWmADMXAPzA8/jzVkm3ht
+ 9sTb5Em2ttODg84/ClimvZp raNbJmjJy+EwWY98+nSorSzmXUWh3KLjd5u9h8uB14+hqXbW+l
+ jJrn5rWXKujNJYtMnjucM87glN qtyMfdI9qzpLZbS9S1j8yVmJWRN3L8ZBHsKmit4JrkySLII
+ pHdwUbGVH3f1rOWVRfRpJcpBM3yq7 g+vHOKmmnrZszw1T2kJv2jtHv+WhZmQRyhHt52i4ICth
+ gSM4zWbds7XBJ2xkk5xxu9CPatAGVLto rpzKZyJW2HG3BxVUN9sugYouJ/3sYPITaSCPxxW9N
+ 2d2RCrBTd1bu9l/XyM+OMz3TDcI1MmAWbIG egpGheOY4jkyDgHHtk1NPbBbsxllhUL1IJ5xnt
+ TEWKMDZIXVycY6jiupSvszeUZOqoJ6Punb7+pJ iOS2VmnRmlYs/GQp4qOB54xLItwjHzQWIzg
+ 56mo0RQpViqvuUbj93vS20pkHlsm0s5bKjA+v0qXH Rnm16UYKSvr1029CT7XGZz5qkQBWSMd8
+ Z4b+tRzJAjKgL7h0O7/WN/eX2q1Z3Hm2Hl+RbTMWd8CE E9evTp2qCZ1ubeGW4jcqke0NHgc5y
+ Ov+eKS0la1jOlUmoqXLZdrp/hoVSgQBlYyMflAB6MTyPrUv 2S4juLeaKRUUbyxkGQ6j7uPqeK
+ gR4Ycy25MTCQkrL82/PHHHFajRwz71cTu0TbFUP98+306mrnKS 9DlqxqT+J2X3fLcmtYilmFE
+ 0SXCAqVZehYZAPuTWVe2zxpG8jAttw2BhYyB9wj1NWLkmBZZTFIpk kVgWPDAjqP6VHDPHc3ck
+ jyxrub5S/Krnggj1x0NZwUk+dbGMaNWm3KPw7+b+4I1t28OrcJiN1+Xa ecnsfwNCiK4t41Y4l
+ Q7ZHHb3NV/NNtGIoplaPaUT5AQyk8t07VLaC1tS0aSAKJAGJGS5PvVyTSb+ 4wdOpBXb130f/A
+ O3SYS7kkMcjvKrKir83XB/SnXOnW0MwktXJLuWiYE4yOgPrnnFQJYPaRj7Mr3E xJbg52heo/E
+ VoWotpWLBzGFb9zG5ySjjgn8e9eHKXL70XodSw7ilOL0fl+auNjtLxftF1Mzpifyi jc7c9R+P
+ amy2cKSBFD2uI28wyN905xt/KtOewEOlpHHM8p2qJFLkkvkYNVkMWnT6isgZ4HuDjzDu JLL1y
+ e2axhWctUz0cPiZS1Tu/JW/AgtcW2/96zRsVAJ+7kHnimSTvJq4ZUZY4YjG2ejMT6etPWKO 7g
+ tFwZRCFiOw4yCcn8eOtWPswgaRgJFtpJCyhjksOm4H0zVOUVK73/r9CcRXvJ8+rfyGi982J2EU
+ qsZcgscjjqPqcU6W2jmeQNC7JM5lRo8DZgfcPvWlbWyAKEiaSZtzKvqBwf8AGsgwy2kUKO7TAq
+ 37 xTgEg8YHvWMZRcvd0/r/AIB5PtXKWyVvPX8xEmuI4osFI0kw48xc9PT0qnDPOV3Xmy6dgSx
+ QYC/N 0Of4j61atf7QLgTRpPEJFV0VPm5zgD0pYhHDPIhUhmOTnu/PT29q3ulfRHZTcY3en9fi
+ XJZ7p2jl toMSOrtsKA7SDgFvXvWB5sxkt4TIslzKn7wpwMDPIHarcz3kH2RIoncmMLuB++M8s
+ PakNrN5zOWR cuSzY++o6lfQZwKqkowXQ1w0uTeS8rNX+fclhgmeN47iaGNdwEYZOSAPvfTpUV
+ tGkhfDRS7pFEgA +UkfeYD0rRguopCyi3kJchuSDtGMMKqSRxLbWazRSoiJhirbcgHDZ98Y5qV
+ N3aehcHUTcWt/QuGG 3tpnuhIhQOCBnqB3z+NRyR7buN5HRYyhDOo6Eds+9WltpJtQMVo0f2YA
+ iJXXcdnfPrgipXsJFsmc zxOElRRJt+ViTnOPTtWHtEnqw9ooz5ebXz/4bT8ShIphW3lkjcRxw
+ PG0Stg7mP8ATNZ627QrNErq JIpFgjD87lP3j+tauLhLy6NxIpE7eeCV4XZ979cUtncyXNi8iG
+ F2adJXOwcH0/CtVOUVf+v63G+e m3ZJ/l+RWNnZ2zRpIyPBGHCjHTHUn1rOWST7JEu9ApXcDjn
+ g5K/XpzW20UEVtc3UqSSsbgKGDfKF /u49e1VEht57ny1tJ4N5LqjPzHgcqfcmqhUurvUinOct
+ WtPkVLS1uJrSRVYCZZifmGSo67frirRX zdNhBtZY7cqPm4zyeDn9KmtYEigE0UrK77N0ZJJjY
+ kjafcip57rar2m9GiPIXHOV6HPpUzqNz0CL lOquSS06tbFeKI/ZrdIYQgjiZAZQGO3PP45rPu
+ bV7qS33W8vkAEPIuBhyOp9geMVcZnmado4ZCBK jKqtymBkg1o+bbta3UARyrSFzGG+bpkEH0H
+ enzyg7nb7GVFJ3u+/b8TjlW4ntxDdZzGcFQMc9GJ+ nFSk21tKSgaVgOG3d14P+NakLpe6fKGM
+ cSK6oJSPvlgefpxUU2mQi1DCTyXRWWUsfauv2yvaWnkd tOvFQ5Z7Pok7Ge/mm8WOOYC6Zd0gI
+ 4zj5s/hTn0ib7FC63cd0ybo4xHn5xnvnrxT5YrdWeZZ1WRN qkkk/e4x+Wa0Y9Pia2dcywBJ/L
+ ALHK5IwT+FOVbkSa0+RlicQqNSMrWXmv8AgFFNMmbR5I3lSE7w QjA8flWlZ2FxHfPJDLE8aFE
+ zt/1h67h7VoBoI3eJ23xGXKt3kI4BB9M8VIXAWMrhZEDeaF6GTPHH pjtXFPEzkmu559XF1Jxd
+ J7P7kVX08yPi7ljERgYQsoxlAcs341TklhlQojwW+xgRleqgc/4VqQXH 2tbszJseWRWGT90EY
+ AHoMjpVC5itRbGRbaZJiMEFumeP07/WlCTvaRp7SpdQm9trWsUJo4ktg4lg Ikw5i2/MCOKha4
+ thdRNbRSSzONzKTkAZxtI9cZ5q8F32IhWAu8bCIMOwY5yfyqOZLmS+uVEaQt5g fbs5TjlT/nv
+ XTGS6m0KkXB3k387fgihFcHbJ5kkUayT8K4zuUcAj2qmziI23lxlpMlXkToME/wA6 vxwQtN8s
+ yrLje0TLyAvoen1pzyQzwxRIoTE6hnzwdwyf/rVvzJPYVScoNXu119P1KMwMskCz7wjQ 5MgOA
+ v1/L9adbPbRym483cwUoV7kHqf6VpS2yy3KpyvlMYxnkEt3/Ksi5V45YrZRG6hg6MigblHU 59
+ zVQmprlOjD1VVTipW7pdvuf5lmaBdkjxpMsDSL5ah+ijrTxcBIzNbxNGYN0asxyME8D69aikik
+ uru5itJSys+8nPG7GcD0zUkK3QEayshcy+b5YXpjnB+vak7curKquVve963S/TzJYpI0itJ2kU
+ Oh yy/7Hv7n1q/9uFq8cZSSeSZ/3LIcZTGT+Oe9U47yC4uJDHaySBpzsUEZQY5U+tSNZXKXkct
+ ycxOp AK8YU46elYTjFv39Dj5Kbi5S0t0vv5K2pm3QMmp2WJJlUw7mG45B7CtiGKDb5sRZmjm+
+ QM2SykYG Pxqo8MUGoJmOWaHbuOG7rnaM+9aWmT776WG4iWF5U+0ZPG0pwB7c060m6d10OrE03
+ OlzQTaS110+ etySygvgXNwNgikCYx3YfMT9Ksi4g814Tbqp3rIkhAy4UdPrRb+VPbFi8jM0uL
+ k7/wDloeOPSpXK pbyieJ5I1AijKnBO4evtXDKV5O6MVyTjzaLyWn66kDyWtxfJOkTMkqMylWw
+ VHYfU1BG9vBam7lid 3YBNjHIIzg/j0FAgS3WCOEthEdMHncTzx9T09KUwia1topEe1jYFlMh6
+ Jj5gfcHvV+78gclKacn7 r0a6kMt8qwYCuHQiN0B+65zio4EsjqDRXc6FmQDAyN4wckVKloN0Z
+ 8xE+QI7MM/MTwT6kY/WpJzg zSLFHclB+9eNfuMP4Pqa1TjblidKqQUJUqd/vsx6CwMcsoBSKO
+ RU5boWrPCxxXG2RQ/lkEmPjaec GrJhsxp4W6tbizjkZZA5l4fBx0qzdRWdxqJWOUNN8xCIT8w
+ I+99BUxkk+tjNckLc3M777NffqZv2 cSadJcKWKOyqznnLg4H59aWO1Np50MbJuF4CwYcj0X86
+ vfYrZUlDF3Rm3fK5AJAwMU4IXs9rFY5h taRzyFZeuffFU6t1a+hrVqup7t7r+uhnTRNDqD3s8
+ oUqsgCEnrjhalVt9rb5hu5HaP5NkmMDHIPv 71baOS5SQuoQSEypkDjjgH61TFvLa+XeM/WDH0
+ znimp8y1eparua95pNfiVHCw6PG5t5wQw2sWzv To34iqlxK6ySFGY2m47STluRgAn1PWnvJe/
+ Z7VVDBPKwu4A4z1PSneRLN5qmN7gtKAsUWAzKBw49 hXTFKOsjLkpQXNUafdq70+eifzGQBhDI
+ JHt9+5hIXTJDgZX6DHanokU0dskZSVmgKTlTxvPINOlI W6khSN4bcljI7c7iR1FY8cclsDKZk
+ kYTCNdvZSOv1Aqox5tb2Zlypu0ZNPpZJ/kXDa7LuOKArLsT c0bLncc8MKjR/scbmCSF2LlW+U
+ ZjB55z1qjKVM8rQtKNhARg3UDqac9xIzPnZsDEABeSP7x45xXQ qcnvqb+wqOd07p73Wvz0L8s
+ WLRPJuT+7GNoY5ORlj9BVO3jurbTJJon3rkB2HVQTx+BpVtRFEJWk aTYwJ2HA4PT8afcuJGdo
+ pAUmbPljgqegBpK/wrVCoyV7Lb009P8AhyENc3DLATvVWIVVAGcntx0q nKEjuSLkhz5mFI757
+ j0q0izrceW37twu0joVz2PvVdEmuZI12KZNw25UFiSeK3jZPyNK0eWL5bKP 3W82ORi1tFJbEy
+ SqjIwz0Gev5ZpskczS/u0cqJdse3+EY4P41f8ALMF3PCkEsUjPwD6E88e1RXNt JbLA0Eh2SSk
+ ZIztUDnNTGor+pwqs4S5Oa76aXb+7oQbJZWmA3BiN4B5IA6D602GB54gAoXEvDEcK jfeB9we9
+ SAwi6liEu/L/ACEDHy49x+NSXEix2lulnJ54lVlTbwSFOS386bk9kTjsRFqOr5vnp/Xm jKIxO
+ Yyrud3AHU4p6t5yK7MyBVJfA689R7dqt2iwQXQkQTTMqsDznJYZ/wAaRzaw6cEjl8yVwAM9 Nv
+ X065rR1NbWJWJ5pck20n1X6joDHDcK8ToxbLeUB8xA/p7U2Uo0MZWCQg5yQ3B79Paq6wkwqJWE
+ LKArKwySSeufTFNXayNJFHLJHHMFByNrBep/GlZXuRKvClUU5J39N/yX4Mmfa1uqCEMyJy2OM7
+ uv 5VWRJwkpkmjBEu3KJ/CRwat3F+0stwiW+3c+SVwMen4UrXDvE6LAC6j5iQDyB7inHnS2KhG
+ rUd4p xT7P/gFaK7gjt4YtwEhO5POG4DngYx9agke4VVljhh8k5LmOIBeTg/lSn/WCRooZBGhQ
+ nYMDcMj8 fQ0ExbEgYvEsURVhnv1A+uavlSeiPMqYd05W5eu/UjWYRXcUs4jni37cIMH5eAOnG
+ anjluIdQBji VnVsyIY8/Mc4p8c0ZiaGUxqZ5lmZdv8Aq9vY/h6VEzCG5MqRzRu8oaIs2QAOoP
+ qaN7po51rzqS1f f+l+B2llMxska5nRWZlxH0fhjkZx6VfkurSG8imgmjCKrjDDIznAH6mqVpm
+ 2vtsiDgsmGHVtpO4e wGKlitpZtKhWNobjYUMhVerDkD8a8OoouV3t+B3ynT9o+bVfcvysi9+8
+ mhlEaygs6uATypXt+Hes 2NZLieYNKDJIWYMQcL/skevFMf7VBPNcTHaIpPKVRxuD85rW2Tr5j
+ w+XuRykI29j97Pqan4Fo1r/ AF/kehFzi/dktdv8imlm/wDZ00bM8W6aJmA788kelaQtZElZtz
+ QmNXKeaxYHPXA/L86rSMRaxTWu RLt2lXO7cv3dw/DJrXtYRFfsTcLMY4WhhB5DL3P19/asKtV
+ 2vcynWUkrS+XmVLe2uYzbKrtJhFV2 U42E87fqaTasGp+WY3gzEziSZtyLjtj15oW4aWztkMcu
+ 0xlmCNhnx91gao287SSYkV0UDZK0h3Zb B/LtUqMndv8Ar+vQ8xRk5ObXy6/16otG6uTZAPsWR
+ gHyF7rwahW0lnlwd6MCUyfzP+NTPbXLWP2d mWRSo3FRyR3x+HNSxypAojSZQJv30LvyGC8H86
+ fNZe7uPmUIpwfvGch8m7JEU07qAD83BBUkEe1R /aHazhmmikP7t2jIONy9GI9qutcx6jBdNar
+ 5cjSxu3sp4K/UU0soysZSaIkiIDn5QcHFaJ91qdSo QlNSqL3u1/1EhW2/szMcheQsNuDyF9a0
+ yktxAqPD8uMFvXJ5P8qW1Jd3tkSLaWywwNxPqPRQOoqZ poUtItsy7fKcKxPUMeP8RXLObcjL2
+ jUlFLW/rYhit0/tFCrFjHHJFhSRjdzk/kaGtvtVsRFMPKEi gqDyw7MPbNUo2voTsFtNcyhfma
+ PgH+9+lXtPaOSWWO1Y2t0doiMx3L5XOePz5pzUormvsdk4VEnP m27W09ev4GZeRyNqgUyGXJw
+ Qh4BI5/AjtStBO4jWBH+zCM/OnHLdKu3wzZxxuQiOpcy+oyMVVj06 5uDJLHOYDgZkfO1Mfw49
+ e9axn7qbdjSlV09pOf36r8BdLtg0iRzOwY7S6s3TGc/jU5tyuqxXUayS wupZWU9z3+nAqZ2WI
+ Nl1Z2cFdowVXHIPvjmq1tatbqDbSM8GCIAWJ4/r3qHNtuV9zKVS6lJytfQb FHkxvMrgSFXBHA
+ JGefz4olt/tN1biWSNopEZpIkGHU9Ov4CrTwxfYZCbpFMkqzhsnCBSPl/GnSSw RXjebbytE0h
+ wUIGTwQfoCeRSU3e6ClGVlNKz6dPzH3HlWlkA43NKnmNsGDkMFz9MVmvbW7I8tpIW aN9i/MT5
+ g3Y3D2HSrSpbSXZkkSWQFGM534GfYdh04pGjUERrYXCFXKffH3CMsfwODTg+U6NIpaO/ olf/A
+ DKEr20MVxFKu1TKuwLxgdh9ahkt5vMuoo8gKdspfnL7f6Zq4+mKkhKsbna2WP8AexySPr2r Su
+ CkVmpjgknadzMVUjcAMdfw5rR1krcutx1cV7N/u1dv8PvOQitVDQxRwzTDaFnYNwzE4BH0rq4o
+ 7qOzkjitpGJDEM2Du2HrWbCmL0T2YMX8IWT5twznd9B1rSjjvLvUjsLEhyFK8BxjqPaniKjlu9
+ F3 FjpzqNLmvFau9/8Ahh0Ane5mknWOKBnVk3LypwT/ADqK8RhZWiSxMshJAYcYI9ffmpxAIrc
+ tdSEg Iqtg8ZOQP6VflMk/lwSwPAqRu0hfHLgAcelcbnaSa/rQ82rGTnGS2+77u5gw290JIti4
+ J/iI9ODj 8elRYmiMYlhlMYQqxb+HBxgn1NXltilmwS6BAwXOT8m05/8Ar0gu0lhuXlvLaYm5G
+ 7YuByOP0rbn bba1GnUTbi7p76NW+ZFj7dI6wzRxgEeYoHIfOFH5CrjRTg3SECSZpt6uAMHPH5
+ CorkQwz3TpGSxl VZthwGOARj0xUUEp+0XT28zFzJtQuchRg5H1zUvVXW3/AAxpXhBpOCv69zn
+ r6KSPU5PL2xXIJXkc upXkj2qs8e2VjKGESPsUA43YH/6q17oXZEMsUL+aqiMhhuOSOP061SmU
+ RWSApJJdRBUdc9zyDivT p1HZI9GnXqStZJ306XXz6IpPs88AecJyN8ke/lWFDW8lvvljI3tMr
+ R7+T/u/rUTNJNtiCfMsLJLJ jktnI9/arEMkzyyJ5yRZUuodc7eOe3auhqSRs6U5Sdvne70+Qk
+ 0kkL3CMPIM0sZ29GVe/Iq5LNGj 71uYpUVHWJFB3HJwDn8abZQGc4klVkEPBIyX5xkH6VNLLFH
+ ZxCGITREEblAymD0J9qwk05JExmva KD/Db/MzbhI83CKrAxEDMZxlsYJrRFx58SPDL5CiUcTf
+ NtPQA/WlaCGJZLYyjYzeZI7dWA6kH0NC GxaXzSJCgIRAp+6G4wfVs96JTUlsVXxMeXl5Wvy/E
+ 2r20NxE0DuCylgTGMZxg1BbzW2VkMbvKq7d in5sMfvZ9AKhaZZI7iNJDJNFdxqI14YhRgjP+e
+ lSwXJ3T+TbMsjuphDgHMecGuFRko2f+R50lKKa k9Pu/wCGH3gSW0IidQqEI23jzGJ4cewqmlv
+ DJqMZkM4iVGH3zhCPX3PanG3eC/u13eXEsgEYY5xt HQ+5zWsjzCIAQCR41KthRwDzg+pI6Gm5
+ 8itFmksQ6KUISvft/mytJPFJppUPGRuyQv3lBAzz9RxT 4b3zIzFAMRuR5bv8wxnDD9M1Uku5D
+ AqxeRBHJC8gDplowP4T6n3qlNNG9jbnzEgjlXfuHG09hx68 0KjdWaLVOpGEbrr6/pqad9DcXU
+ yW8GxsRM3mAYEhyMEVnb9Ta5ngRYy5U5AXGTjrUrXlyLx5HzEs cwjLEYAJ6UnmK19CFuVEkQX
+ J67ufmP4VpBSirNI0hKdOD0TtqvX+uhVWGCSC3+0Q3abmQqHl4UDI 2/UmtZdRsbK/INvI7YOV
+ UjKY42k+vNUbrL3f2hWE8MTeW5j6B2Py/lSLEftl24mha5QnfxkMRySB VNRmve2+ZdKFOpFqT
+ vfZXdr/AKEkTxXV/NECYRDIApdsgqByappcmaQKsokTaG+TjqeQfUnpSCeS SWWeaL947oSUwo
+ +bgjFUv7MuIZ99vIiIFZH3clGwflPvWsacFe7sXKGHjNqTs7adfuLSX88mqbxG YGLbiWORGAc
+ YIovpHMzwbmLRLhXB4fHPA96itwohjC7pCoViOrMxHHPpntVq4V5I90jxwztb75lY fxeo9ABV
+ +6prQ6aNSLrptWT07/8ABKzvI9raFkdZHVmUnoVPGfpUTzPbSTNbbnmLNGxH/LJemD9a Y8jwX
+ NpK8q3MKpx5foB7+uaq3DSCFSm6Aw/IMjls8kn1+prWELtdjmjSndqOsW35X8thY1cXqecs vk
+ yL8zk9fRh7VXcSRXJZ4+REwYkcEnoRUiTXb3MKwSRHCO7Lt7d1plyxjsEL5BGIwG/iOM4+tbpP
+ msy5VZqraVlfTf8AIqGCSVbeKFHlmwQ6ge/FWo5HAlt5Io1Eis6uVwVHp+YpYGmbe0OItkgXOO
+ Tn nNNFvDILePbI52Oxw3KlTnn61cpdH/XUyq15QldfLun+ViwJbn+ylQIJMDG8LwT3HvVSxjm
+ uZopi 0YhjYCR8dPr71JbzTw2xuBbyO88jAgdFz0OKnt7gjTJlVQhZzn5eGx1NQ7pNJGdKDqLk
+ TV5X8vzu RXweS5jKlIlDHeSMkjsayZJZGnQqjnOckfwntWlPHELCAo7lpOChYkjHf8qjy8cu6
+ Hy4x1CuueM9 +OtbUnaJ00YzVJ8t9Hb7vkLEIyWSdZ43ZS3mGTPOOTUbK8mn28yszRBQshxkKe
+ w9s0wvJHMjocbQ VAbkc05i0NzJGfly43IegK9M0Wd9BVqMo1PcWi1syFTJvWdiFKMVbK9Djp+
+ NV4IxIAzYj8s4BJ6e 3HrV24jaWzhnWGRRhlkOcjORj8ag3MkBiZAJiWMuR159O2MVpGV1ocUJ
+ e0nzWu+t7FhDD5kZIbck Rf5TjkZP49qekME1qtw0iwsVdSrDp6/zqKAv5EskEWGEyx4b5icg8
+ fWo/LdIFikBRhGwIYn5uefy rNq70ZVaUqtoUrLvtf5+Qy2e1iy5jlkhUNH97JJz6/jSS2bRws
+ 8BdEjk2ybz9854x+FCBUsQ0S8g 4k+XIGen4moAsko3jayxnbNhccnpWttb3OaVKCirv5f09Pu
+ Hs0RikCh423ERqTy2DwatSQ3FxcW4 hjwrgKTnjpk5rNjiENpI87Kk8c4AXHOTz/Srq+UksCW5
+ lmnCNvCE8EDNEtNjF1alkue1+r6evl5l MxsbyzXcsm7JcIMYHfPvgVanmtnnDph1bIUA4KjHG
+ Tjk1FH5jzwLHJGzuhwqjnGOn1okiS3ZNp83 kF4/4l+Xk/mabtzHHNxjUum5SXTa/wDXctQShb
+ HyktgWcqS5ALMp4bB9OBUoSK1ncEnEPzQyPyrg df51m2sP2G2SdnMi5C/7xJ6j2FXEdJtRBu9
+ +1W/ebc4z34/Lis5x1dtjjjzTlJ04XW/d36qx0mJI YoGaeMSIrZkbnDZ5/TFNit5BNnzcKcbl
+ jcghgcqD+HP0qOyjMEK26AZWNmmZ/mG/cMD8q15khfV7 ieQSTAsQqwttyAACfwrzZz5W0fQU8
+ Q6dGUZW18l/X4jF2m8kmAdojkvuORkDIpDL9ujhMEc0oWIE Kr4MmeMj6HrTpJ4ltQsDpHsmSJ
+ S/IJPr60sVisNxFFv86VS77YTg4B6j256e1Y3ja736GUa9JpTc dfnp93+ZTtIEl01oZZGt5I5
+ VJZ2PzAnqPY9MV1SvNb37TxlPJjJjkyucO/QfhXIrczi5UPNBHHFG FLMmc+g+prZa7uotQKKU
+ MLSYPy/eJ5P49qjE05SZpjaEX7tlZ9/00NGS2mLvsUiQcEjgAL149xUU peKWGCG7tFMgdolaP
+ 5iB3J71I01vkZS5QvhnPmfdPv8A1qbT5Z45JIJLUXA3j5gozjknB7A+lcd5 JXfQ4Zzkknbbul
+ /wbk1s9zBbQyzQO6xtGkjADGOf8ap6hdR3MDwQLEUjU7NqgEBSDTbm2vXKokhg GFJDfxc4/wA
+ KiL2kF/JIx+0SwnYY4uCd3b60RhHm5t35Bh8Iub2srN72X+b2EVoy93JeWlwqSTll WIhCwwMk
+ Y7Cqypbx2gjimFsInC4lOTwcn861LkC6WdJswtET8/Reg4/HinpKwtI2je0VppRK5eIH Cgcj8
+ 6r2mn9W/UiUpRkua+r26fqZ1pZzzzCeItLNIoIKHAXJwRWvHCY3kiNxbNF5mCSnfsR7e1Zo eW
+ aUEK1uxfEQU4yv8XTvnvV1pIBb5lvIWtw/mMVGCe2c+lKq5SZ1TxMpSUNL+Sv/AF+A27zD9pkc
+ Sk7gpCNjJPYe5HIqGNrGIRbFltTs3J5j5IHVQT+dTXd/E5ZVhkdGmCtIDwW45/HoPeojfSS6i0
+ bw oI43KOpXlc9j9AKIKXJqvxOxVJey5eVu2rs/01Qtx5H2GCSIPdAx5ljVudwOQR6AelWbG7Y
+ RTBri GMySrIXZflyB0x7iq12zI0k/lEIW3bBgFhjGR6AUs8ccKRoJog23OTzuyRzSsnFJ9TJw
+ j7O8ou3f 9LDbi1nlGzzFMoZWdFGCgU5Ofwq28Vo8M4M+N7CcMHICr0P4YqTzJ1WScqLhwHRTG
+ MZ3YFZNpbTR 6qEeVUQBFk3gnaQOh9jzSi+aOrtYmlUc6LlN25dv6szYntYI3gWNHnt1gcsqNy
+ W/h/DHNUYmQ2lt JHDMp8rJ3tuDHPJH1FajTwRASuQ0ZQ/KOoPTH5VHBb2qTrBBLulSNiIy2dg
+ A4B9+9ZRm1HW5WGT9 nZxbXfW3z7fcUbqZSGGxiVyqRqcErjPPqalWSS5szugnSOQI4Jbrg5b8
+ DV+OG2hsoGkIuBIAQVOC +eMj2qNrMCKWBVm+R1ULu5UYyc/zp+0haxvGtSlBJ7973KzRl9ZaE
+ EtGwfeFODG2MgH8KznkWKS3 mnt7pSVww38bm7flVxLEFFnbz5N2HiZXxuHOfrntRLcQtZwyRT
+ RQtNwiT/MUJb5Qffg1rFq6S1Gn T5+Vaxfrv8v8hph+x3TKkMvnuxdSzZVQq5xj3q3a6jK9lDc
+ +TteWMtIqqBh2GOPQY7VWYmdh5kou 5gvzeUcc5pFWCNJ3iLRSk4G85X2IH51MoqStLcirTp1H
+ yPfv/WpNFJILF0wIgkqoWlG7cU5/TvVv CXdzdebeI7ySsSyEgEEdvTpiqsBkSNFZfPUHcSPXt
+ mnRXCrNI0mn3FiPOBeSQggFeQPy/nWbi7tr f5f8OeevaRcrLXuv+DqUTDGdoil2wkAFCST055
+ p7fZjYtE6CJ1kCJ8uCwbGT+HrV3c91fyXG2ImU NMiIuMheCR7UyVtLk0+RlcpsdFikZ8iQP24
+ 75q+dtq9yo1Jykk38+3yM+6nl+1xW8LpIgjbLAenQ n3ogeSW1hiht2nkj4LR4G75t276VA8qQ
+ ube3ZEniOHdvmBwDnikt7SRrqO4dnjcxERhGICrg5B9T 71vypR1DEQi6bctLbX6v5WMe6kv7b
+ UWulWWK1Mv71pOQx6jHpUMaNc3MckrEbrZnXDYIKk4B9TUk 8M/2K0CzLPHcASbSCcEHA6+9Ub
+ m3kju4wXYXCMXlT+8wIyB6DHavSp2asnqdtGrCcLxfvbdfyZLI bIWoYmRZnVMpv+7jgj8+agc
+ R+fcxKkkhD7Bg8qR/jS+UJr2Qh1hhdycuc/KD1q0dPu7NIZNpkGHy VH38dG/WtOaMdL6s9GnU
+ jBqCqczl3enpctIxSO0uACoERzg8HdxkewNVY5723sMFAVhbycFOpJya tjbPbZ3iKRYyqs33S
+ oAyMfWs1pgiXWYJA00wcAvwFxg/jWUI811b+v6uVQp80mlDm7+X4luW8vhN bS+XGrCBsZQEbT
+ 1J+varcNxZLpyGJtsyKqENznLZz+FZ8TQ2jPm1m81W/d+ZJwVxxnikzeyabZGY 28ysrZZIgu3
+ nIBpzpp20sv6fmXWoLminDlT7WX6M07qWEIsYKNsBLSR8ZyeCT3JFVTf3iyW0cGwk rnbt5AXk
+ /pSBUgsbeVLyFZJfmfcpbkHjseOoqNrW9vLgxrEWQzDa6gDaD94VEYQS128zhdOmoOUt I/3v0
+ 0X4M0lvLoC3lleKWFk4VV5yTwSe9W11SFY5JZbuG4KKVPloV/n1J7elcxIJFvAVimhkUgkO ch
+ QeDx7VPMkLRK+1ZkDsIfLG3egHBPrnsfaplhoNq/4GU8JHmSavfayS/H/gpmrJqFg93HEEYxcc
+ nsAemf51RubssLNjbhjP++Cj/lmd2AuMY7Z+hpLVHisprgRqqEkB5ACAeMD8aDqFwyuCkUUp2n
+ DR j5SowR/WtacYQulG/q3o++htLDQi709fV7fcJefa5LqNxMirFceTJleDu5yfpWjBiGZDJAx
+ fa4Zi OAOmPqTUCR2H2eKNoLy7ZNrkxy4z1IP40xHQf6Q9rdLCylipk5JHf6CsZaq1v6+85Kan
+ zcq0Xpa/ 43+8049OS18kwM7jbiWLdkh/U1jLdTwyiXaI5HObgMMliTzj0xxTp5LloI5ITJbxy
+ WxaZ3OQW6Ae 2f61nQzgWke2VRN5ibo35J54H+NXTpSablqdcqUoxu9blppoZLeaS33yFSqZXo
+ CD/PFWEAjvwrPK SY2bhvvAkY/E9KWWGVbmfzWigRpG34XAVl5B/HOKq/ZUEkbXF4kb+WVZDnO
+ 7GcVScWtyZVoVIct9 ej1+75E3leY0k8EcsbIdpg3fNGT0U+p71YkW+RkN1F5khhcbcDPUbs/h
+ xWcst4IFdUJdUDTcdW6c ++OafZpK1/se+QSpkRFskMvf9KJRaV3bQVRVFC7Sdvm/vW33Er/Zz
+ pXnW0DxxRNgCU7gw6j9aGKT rELn95ggt5fy4b0NRyTSS23lKCLO5w8TY+6B/CT68VVMrrM5Y4
+ YOgZfU+v41UYO3mP2EfY3bSktd 29yyLdDcR27q8IWErJJnqd2f/rVTt/7PmmRJFljdpCWkd8o
+ GB4bHp2qWS6nn1JLeN49rI+Pl5OOp qqyQ8Qput/kG6RznnqoH1rSMXbV/cYVI88rzbvbp089d
+ /wARC9xbPMDGGiMh8zHHPYZ7U+3e6eXa 0RVW5PGOp6/Sq5t7qdGRElJR13J1Jc//AFqha/ZXJ
+ jY5V/lP93HUH1rfk5lZWubzpp0nHRy9UW3l ZJiY3WZQpB2jjHTNQx2iuin7TGuwFyCTwMfSq2
+ WmkaV1G8tuOOMjPI4p896jo6wwtsJCIVP+rHXk 96vkktEdPNZJ2t935Dw+5U3OoTYwXI6dqml
+ Ahht3ZPNi5A2Hkntk47URCc20E0LwtKWWNY9mTjqT zVOWRXupd8wH7wmPB4Zs54/lSWsrI46u
+ NVSTjTaS6/8ADdCeGa1ICybS0bsOv39vIP41LFsuJdvl MjyZbnk5J6VQ3SpMSIQBv3FSBkZ7U
+ z/VuFG/zFYjBP5VfswhGpBOUrrs+hYkeJd8Rm8gFgAjdSQO vFSAI+9ndGM6E8DlW3CqBVVkia
+ RopOucKTwO9StIZAI/MjjCHKMBgNj/ABzTcexyzrVU2lJvza/4 BbaVob9oYATKZc9M846Y9aq
+ AfJPJJJvZCEAI6ls5qwsTlP38gWRm3JHn5yR05qRLqZmkiEcaNLvI VkGRkfz9PSoTtsZUK7pu
+ 1k+7/wAylGpEsSNA3mJEwBBxu5yMjv7ULDJMziCIorANOc8f7P65q2YI 18mZdSi85YwCNhOB3
+ zxyapyzvDelrcq28MuMdBnIzVpuT939TJwTm3Tjb1T3+6wWot7mGETHyiqM rk/xMSdpqCJFVI
+ w6SRtn52Z+p6enAqKC6la5QxxbHXJORkNjkGopZQIozE2fm+bqehz6VsoO7MqL hGc5Nt2Wiff
+ 0t+ZIXcruaN4m8xdgHGFH3qr+arTqQ29GkzuJ4C/wk+3rT0EbDDSn5Ry27gHtmqwR zDHuPlfI
+ VII+9zWqSFifaTqXg/Xb/hjaEsks0UciF0CucoMKfVh7VmebF9njdJ0bCMMEcsM/zo8y eW0jj
+ dGLxJsiKHGVPUn+VVYYTJuU7EZpQFVhjrnjpUwpqN7nnSlOF3K0Uv6uejyfZJJIIpmYo6MI 9p
+ xj2PqxPT6Vct2NrBI13wVcqzHjLY7e2Kz3vLR0hlMRmdmyfLOPmzgY9OKuvHLcRXQMUodrrPmM
+ cpgYycV4Ek7JPRHqRvzLndovpdf0vuIWLtfqY/LC9GDAHDgAj8cZpbGS5urzzvvTmUbWUYCxHO
+ f0 5qNbe3ublVknOxVkZo1YhiR0OalhnU6arAqqnl5FOArEcL+lOVuWyWo685JNxuv6+9mpeWh
+ eVWgj VAU5JUENjG2mrJHJMWuJI2Zn80qoxsI4I/EViNeTyXJVplCyDzMAYGewH1xUkdwlwjef
+ PFGzAvKQ uNjY5U/Ws/YTUUpDnQm0lUlv1SuzpZfJuZZGRhErO3LHhRgZP4YpAZZZNyxSxpFk+
+ YDw2ev5Vn2U tsyQSwndMIBEATkbWPJx6irEAuAY4mcnzM4IPBABFcrhy6du5moxpt8zbXS/T7
+ v1Lkkm284aQJuG 5mbhHHRfxpGjiN0UYpkShy6jGVHX8amSz/0LyDIr+UViyf4wxzu+vvReS2i
+ zsJJE2cRIVOMqTz+P vWXMm7I2nONRNpP5Fie5iivBIkR8vyZNxcZCE/dB9T3rCRXtUjkQNPtI
+ jkPbI+Y9auhp7e4JjQsg uVDbxnB6DNTT3jXH2mMrHHH5nyybflzgDH1p0046LYqhTjLRbdWVG
+ d3gd0tZo2mcyFyeEzj8uOfx qzbXMLRRySxKVVWjQYHzB+Fb6Usst0kSxBoCVBibCdAT83446e
+ lUIZY4bySIgyWUWBAw6vj7pz6V XLzRehHs+aGnutee5oT2aiO2i2v50RLMFPDOnH5Hpj1piR/
+ 6MwkkWQlgwVRhiCO57kU1r61aWVjI UdF27y2AQwyRj1J71nvcXUjpECkm6RG2IMPtAznPt3pw
+ pza1O2nhpxhzS6fL9bGsZDJA2AGjlILs P+Wew/dP1pl1bImoteSyKY4o3RohwcnHP4UyP7XJq
+ WWQeW7MyqgHy49fXNP+z3kzT2qrucsCHIyD x8xqV7r3sCUef36mnXW2n3Dx9o/eNIwhtkQ5J/
+ iOBluvTpUsECDSRtulupTIrsqfeJ9c+maxWiug yqzMVDAGM85Rj839Oa6ATIkMqJE4hwS7LgE
+ MD8vPp7VNVWSsa1qMHFKK/L/hx7SQz6bKDEYi7M7u 33S38IHoOMYq1bQyujSSSwb/ACyCiphu
+ Rk80NYh4ZzAHebz1CEn5SMA4x+dWI2uFkczTW7qzZASP ByOn5/0rklNcvus5nVdOl7jt5df1t
+ 95DD5pie2ZVYeYhYKPunBIwew46VG0VuodpLoxebKjZYn5g RyadKzreTbSFeSQNHkdPlxg++a
+ p3dxM9kC8kMEhIVldc7d3X8qcItvtc0pX57LRPcjnMAEkSzPLb hiUWNsNgjIAP1qO1vLZyslx
+ beTuQuY2xu3Abcj+dEkMu1TbFXlWEnAH+sA44/CqcM13EIyEiEZcq xdAcfL8tdKinHc0lGKpu
+ z/HX/gfcXfOki2Ss0T26NtaONMOx7HPoD1p6X8TuivGiqEGxyBhic5xV ARsJomu33yxwYkiQY
+ PT5j+HBpkTTC4LxXVo4hXy0zFkHd3/HtTdKLRz1IRcHbfvdr8ja+0/ar25g QCQF1wiDDYC/zp
+ ILazR0ncziQpuxJJlW54JH0rNF3II332kiSmRfLZQBgfdJP0PWnOziZ4JJFkRW dJAByz4DAj0
+ A9Kz9m1onZGcnLrK0Xvr/AEie7+1LYvMRzuwxQbQrA5C/T1Hesze0cskjqsTmfId1 +QfL0xRH
+ crJefv5SYXAWQZ4EhBwfxNWnuEhuIES/sFWMMjRyJubdjOD75rZRcfdsFWPKuSKTa1vZ 7fcyB
+ N8bxreSQTQuE2+WmGcnoc+gPX1qlc3kS3Hlos8FyIyCzt8uCeTikuJkuYdOdEf7YlvuZQeu Dx
+ xULDzhNcmIq0xBZDydwHOPQD0rohBXvI3jB815Oza22/DsZFpd3yW8EsBSUq33GXJzu9PQjNNl
+ 2SanMJ2dS83mIQ3Rf7v1PWtKVnjtIooQkryQliETkn+8PbGarQQOiG72NbrHlA0w3AA4wenvXc
+ pL WVrXPQoUNHVStJ6aeXe3T5CQ3dtHa22YJSEXEjE/x5JA/KrjI0UJnn87fOm+Nd2Ain74x69
+ KnEyr oqoIlnJk4KqPmGcZ/CoIoLiW9uYoyymLgCQ7jz1rK6d3sXhaPNBt2i+uv9WGTGAWjrBO
+ qKzhmDZJ Vm7Z9B3qF7WeK7/fxuiKjIHPQnHFTBElgMinziHVJEQYyRzSvb28dvBvmkkldwSik
+ AqCcgdOpqoy UdLnZTqUqV4Qk7el/wBVb1ZVUMJYZTIjJEgZo3GS3GPyzTnEcYkllRlGAihWx1
+ H+FRgC1uRI1vMG 7BjkDrgH8ancSmFoDBK7yuG3Y4OBjj9a06m9Kynq7J9bojR1IiWJEMSSCEL
+ IudueQD79asFb+NXj gDxwvNuHru7c1atrhkKR3scccKjcvygFmHfPrV1J7VbdJPOVYiu4gnOx
+ uoU+5rnqVGn8N/xOGs3z 6U1JfejGkl1CDzDdIsuJv3pMYHzN/D/9aiVZ4B5FzEoiR8DauMkHo
+ MfhVy93/ZEklkRJWbZGhXqP 4mPrgd6rTRmTThCkcrB5so7vkMBjafxqoSTs7IwjUp80JOCSbt
+ 1t/n9wk96r2cYEZSI5eM9Q3/6j Tra+ubhts9ks6glm8tQp3Hrz/SpHkuEsLj7Q0Fs4dk2vHnG
+ cE44OBVIQSjzZ7hwsAuCWdDjJIx2o UYOL0F7KjKN0km27av8AD/hxZJvOiUbHhTap354BHf6e
+ 1SOZ3MsVuDdKhWLCjqxGc/jjpTJbqKGV TMDHIynygRxt245H1qCGFVtgrTtFPsBZuSAQDkYH8
+ XatEtL2NqNWT+z6aX1/P8SaYhLQSMsyyEDY rNkMDwTj0FVZLUz3AhjngkIUvHtXBbaPpV6M/Z
+ NLDAw3O9AyLIu7aAcAc+5pqWVzcC2ZbWXKxYwg A4YkE/rRGfLrcFJOF5JaPdr9P+CVUihnNxJ
+ NKyszKck9C3Y+tXIoB+4M13DNFhi0ajDnBx1NOazM EIjkKQSo6LGX/i56n6Vbit3k1N5LmIqS
+ cB14XeOQMe47VFSqrXTIxUlyOXNb+vT8jPjna1ljGxrd cBQ0vzAqTzn3qcajbokiwRo8oz5Tj
+ HT+79TVi4hsZpLRrmRk35V/mwFc5OfYCqJBjiiaG3J8tWDY 58wEffHoBUrknq1qcMHSqtcid/
+ w+8j81JmkeZWtkVCqA9z2xTNQCsVaO6hcunmEKvJAAx/WrErwT 6Zbopw4X5M/xAkAmsqVTHqg
+ j80SqqvGu0fwj/wCucVvSSbvtboddKpCO6tJdOhDM0nmxu8LnzBhN q/dB45q4xS1gYO8Z8wAb
+ tudu04IP50x0eDTkSJHaRic7jkocfMDTG82cMiSeZlecJnfjnI49ua33 S7FU/rFWjfmio6/P5
+ sgn+3pO8UZcsjkEhCN2cD86rvE1tHKjJyjgMGj+4wPANWhc3KXhmyWcxEBi vr1P1psb3N3HHb
+ yyROpBKOQBkepOOee9aJyS1tYxqUqlKVpRjyv1uvO1jOEk6kyeWcs+FG3r6ipT GjJcNvFuVO1
+ FZfmIGDn8qkMM8Cxx3Mi7hJvV1U4wOuPeniWVYzI1utwZCx2bc8E8GrlLseXVxddx Sadulv8A
+ MzoXdQBAX3MzNweQP4R9etKSq24Blhdh8oOPf+dX3tJtjIqx28LPuRto5x3GO1Z32Znj leWaB
+ nEuGKAgA5z6VpGcWysPWgpOKSbt3bHLc7lMaKJY93oNwI7E4zTkSKe9DeaLcRkMu9uWwelL Fa
+ qkErySpLI0xKiPgYPQ9O/NMaMecEMErMCACvc9qfu62NYxc6EmtL99P0ZYQmS8kbIU/MqLt/TF
+ RMhEm7EixgfddefpnHWmGF0uoVWQu046AYPLdfzqyWuHtpEOGCykDj7wPBb6cUr2ehtCpV51Ra
+ vb tfT5oa1wUu4JW2rNj5MJ3qz5swkWZvLWZZN2Cv3OxUiqXkNOwfZJjGFbrViK2H2+JXuEhyp
+ ZzICQ p9xUTUbGePgqsZTdml836P8A4YrynMxKxkAADA/hBPf6VE63O2N2gbBVuR147/StKaSG
+ SwV4ZUl7 ZVcHOcc1lF3N2+9cBWwwI61dNtoxo1Jzpqzsuy/pNEkc0sEUiKUULIAm5eSp/iz6D
+ +tVpJY47h9x S4TfxtHbp3FTQqbi/tzIFmTkSRIMMwA4GfWmQBPMMUdvJkgklz9zGc5q0kmzj5
+ KMpyXNZd3/AJLX 7ypgeXIY1cLuYEkVJ5RSFBt3ZUsDn17U8m1eCJ5QUkYf6scZwfXHtUCwtLI
+ zGTIV87cfeHXFarUh Jyn7sb2+V/k3+Y+a2kKKYX3KuN+0HpkcVUkjeS4llfchMvmDb2rYciWG
+ WW2Cshk2hUHKk84PrSyJ EYTEbaRZliIdw/3mz1xjgVMarXQmcY15OSp38lrb+vI6d7SHyo7TD
+ QqVaSNyfvYPanXl1bGysBIl 0rlSxRZMYy3zZ9cAVopduW2vash3ncWwfn7Y9FA5I70RWwv3TM
+ 0MrxwtyF7d68L2lrOfTzE1FWdR WUddHcjtkt9zeS26N3bAzyRnHX6VWY28jSWcbbYxs8pgfvo
+ OrfhVmHzYjDHFBiKMYDEZ+g+tZslz AZLi4gUJOspiQMPuqR0/PNEE3JmeHrTvJNN9nfb1NdGg
+ +0XEolgl2zbIzjhoyByPpT4rNbS4uT5k RhEnIZeR3APuaxwn2e2kmmjdmFwoVFOC0Y+8RW3DL
+ bSa0pfdNESwi+bgqRkZ9TkcGs6kXFaO6LjK avOL5k/QtLNHLFE1pHGxRdgRQMrnk5+lR2y6gi
+ Ncy42wMEK7e7cZqaxh8+SaKONoWldZd2e2Ocex xirssZiuo2ihuFhaPcdzZCk8BD6t3zXJKai
+ 3E6sPZXi0n1d7DnEcdmCgfYsTbBnkg9CT3PWs37Je SSW8fk7EZA+9xnIHBxWxN5l1axWyBo5t
+ jqxIyGwB849FGOlQyJLHa25WUPujyoH8Kk5P+NZU6jS8 y6fM6ajDdt79CF5LhbKVUMagziTLD
+ O3HrTEuT9uBQKfl3jK5DA9xVRzdF3McqyqZhIgUcSKBjcPa taIuJDLKq2bI+N0q/LgdBx/nmr
+ klFG8YU4wtJJt+n/Dlbzrh5SEmgV3lU7imQcDrRdyyBF226TbI 9kbIAA0fBLfnxUjG8luXeGS
+ 3kVZVBjWPBTuwPvirYSaO6iuI3iuI1R1RFXkD057nPFRzJNPQy5Ve z+S1/PoZ9wy+TdGPy2Pm
+ lom28AYHBHc9cVRSe5Nx9sS0O853HAxgjnj6dKddW3nwxlnMEsiK75PA Zc0kRwIo4nN5NgyMs
+ fGG7Kf89q6YqKj3PWiqXs1bV9d0RxBjGqrM0EZYFVkyWAB9fep7+SWO8naO 43RrcRuETIbkdM
+ +3f1qO3ZppVkmjeIhB8x6M2SRgelXrmQhbi7URyvK6gbE4TgcH3obtPVf1oaqN p2klrte1vvE
+ txOtzG026XClJSO8h6Y9OO1X7OO43zJ5f2iNAE2qOR8p5PrVK2iaZWil3gSSB4W6f KvJ/H3rS
+ WZIQJwzrK0W488NzjP4A1zVm22kcs6kpScUlf7whup4ESJIpZcRjZtPMvHLD6VYiurh7 ou0kK
+ Z3YLJw3HUfyrPE48kq2YtihWJ6ls/Jg9gfSpAgnnneeVYTJJ5hHQRYGNmPWspQWraJVJJNy SX
+ 9eo6dg8MMkrfvYkZJVXghm5/QVDDO7hI4Wh+V1L713EHsPxq/bXNpsaCORGaSMyOG5YN1UZ/Cq
+ DS77dLown7VIELqvAAPXj19PSmuzRM4pq0ovy0/rQtSC9innaW5tZDHcCIrHHgjcM4rNuLq4Mi
+ QR RDlWwSucAHk/X3q3vgneQhJIiZlU7myWPY/gKyLuC98yVFdFjw259v3QDkj8qujFN2f9fcc
+ lOEHP ldr/AHfkZ83kyS284mk8t4H80lzwM8fjV03P2W0E7IArH5iR17ZH0FVXgeWGHLI0Usys
+ MLjKDqR7 dqnuDK13PIAttDubJmXcoZugx/nrXa0nZMIwhPeV7f1p1BbxFO/7QssaZT5f+Wn+2
+ PYfzpEnSVD9 pmQ4kR1C8MFxzk989Kql7h2uFhgUOR8yhRw/YD0759arGS5+zRo9sdyxKJDgA5
+ BJ/SrVFM66WGi1 yvWT81dG5MqRXF00RyGlVkXHRgOB+IrLB3zubsJamaQ4Mg+6wGMH371VbVS
+ u1FjL75hJKxHH1HtT bq7u5lZCglgMzksiemMNn0qqdCa0ZcI4ty5ZRS83v/kTxQzLE8oxLJlS
+ rJ3H8RHtWi87NaSGOWFX UhQdvAz0P41nrMv2WNJw0blHLyA4UsCOAOwPHFSfZvNt7mUSAxyzI
+ Qo4IbGMfn2pSV3eRo5VKslU qO3S7XYcun3P2ppPOjJiby2Kg8secj2xTjp5JnzcKkDIXwe+0c
+ f41FcpPZ26qUuHXd+8YtwGGOtX p4FZzbHcjSTtLhj02gHFDnJWfNv+h6ClXp8slPR6aJFWyhS
+ VTunV5y6yjaCFCD73Hv1okjf+0DLH e25jD4UKCCw6g+9MgvLgXkjwiCKSXlFMY6dx9Kcy3G2K
+ eV4gpXcE24LHHOP51UoyUtX/AF9xpUw7 jUak7X9L/kW7ZFF+u5M+ejSOq8c9AR7d6ou0tjaQr
+ 9nlVhhQ7gNjnPNS20aNPbubgAG13uRnjnit BkV7me3Eocq29QfUjpUN8stdTWdBxnd2aS6r+k
+ Z0qbVmBkimnnl3bsfLhe4HpUqs0FnFJFiW4jY+ Vx1TrnHvzVoQRNbwyXPzSJExJU478n6Ypsc
+ 0csEiCaIlySkm04CH735Yqea62MlSlV5U1pfXtb03 M2RElTfMHDMf9Hx3TGc/0ppeNLW1RYjb
+ mVC8zy/MpOcAgdsCn+Xd/unaI3VurAKsXoBwc+lV7iDI Vmy+CJCg67F7/SuhWbs2bOz91v4dr
+ flb/NE9xbNEI5rrfLAqMigNjpjHPvVOK0WVVE100crksgLH CAdc1Ok4urtsxTyxGTzUi3fe68
+ /SqQMshlgwI3I3lWGScDt+FXBTtZuzMFGrUnaUrNden3bFqA87 1PnKSN+85ByCB170yUKWidW
+ WcxwiAoP+WjHkMP8APakniWHTrCSK4jEgiI293JORU9nEJraBhhZZ jvZz93cCen4DH40m0lzH
+ FOUYx55yur2/r/gEQQW908+Y51yy/OMkn1GegqOK3jfzhcNLFNGpDHP3 8n5WH4GtWWS3lngSB
+ 02yRtKQRnYc4Gfp3qvFbPbLJvjeV9g2yBuGAyTUqr7uujNac3zc2ql9xYEU cUEsVvKkciuBKs
+ vzcjnirqXDSbHS3n8wNmNlOFVemCO55rMhmRmU8FWIKKPvKP8AaPc571ektWUz SrLtnWRfMPR
+ cA88Vzzir+8TUoc8rS/Hq/kZ9w8wjQ7HVo4yCZOcjP9DVy1uZri+dpLSaOUk/Ox+Q DHPHr6H3
+ qyrh7gwzzQMGYuWC+/8AKq1xLN9te6kjZkMZA2fKpB4/Q0X5ly21Eo+3vSsk1/XkZKzx rcx+a
+ CRIQVJbiMkEAn6d/Wr0L3L6itvbR+YsTbJ5Qvyl8cAegPTFWIms5dFUtA/nQMICM8kn+tZs tl
+ Khe3t/OEzSDZ8331HJP9M1reMrpq3qclSEailzaPbXXb5oi+0+TfszW7RLtbcrYO3jCgVbZfPh
+ iVkEzn93P5YwUOAP/wBdVZEtCZJCXWTzdpSRs4BGcH3ppmjmvnyJbYNzIhPIfHB47DFauKeqRN
+ aE J01Kle662/r8yMmC3leIxuJ4wUbe2RJn+ID6VVSFzeRRorSxyQlgyenc81YjgMtteebNEhW
+ aIq56 learyLY29w2VnkiUOqsr8fMOnTtmt4vdLc6FOdOLjF697f5/oTx/ub2ONo9qqDlnAyyn
+ qQfaq32p RdAxWxlzGEAU9Ewcj6571CkFxFbo00UuyMlWOcD/ACeKvwpbSWpkKSCJ0OQDypXqM
+ 02orV6nLyp+ 85N303/4O5WMM1vEbgKscRjZAJl3enHSoXnuXcweV5sMbFMImCARnGce1aEUEF
+ zdopnJjERY5PCq BnJqhNBcIlvcvFKF2byduATng59OnFVCScrPc0aVWTUtZR20t92t/wAwtIF
+ lZIvNcrjKtu4UkUGF fteLhTKYh+7CDbvGPmPvg1Iu1pPMLfZpwW3uT8jEjjA7Clhed3sY5XUR
+ SMCxZeWK5yoP5UNvV3PL rx5FKUtu3Vej6/eRFriC3EoWN1kQMxVeAM9qYIo7maJpX8hUT98c9
+ WPTHp2q1ckwtFIssXnoDCyb crzyePYGhrUPZRyJFLL5kRyUYcso4/nQpq19r9TSlWgqS520pa
+ XMv7NcxiNpYmTjHJ5OSeR7VItv LA6wNNHKsrMEAXczlecg4zj2pryO/l7kWInr8vOPX6U62ga
+ e0ZmUiQSkDH3l2jJP8hW8m0rtnTXi qU1epf5NX+5ldIo45l88svTcQMHnn0qz9oimjieSCSaR
+ VfZGBgqBwNx71F88l1byMARMNxyOG5wM VYW0NybgF7cRQO2XRApfAxwfTPalNreTOPHKlKa12
+ 6X0/C5meT5LbAGIVG6fxd8/1/Co5pxHcK5M W9n43LkOcD+ea0o5LZDI00ckiEDABwckc8+nbF
+ VSscrAgKqEqEU8sB659q1U9dUViW1C1rfiQxPI kgSC1kkuTI6h1bhOOmMdcVVdQiWYMxilkyZ
+ Cc4Htx61Zni8iVUeJ0Bz8z9x6023k813liSPdG4ZR ImduOo5H8qtbXRwywzmrKV29kv1fYI7V
+ IUKxxzuWJwCclW7Dp25oeR0kVUaFlEyIzbcjHr06Vaju HkeNZ0JN0jyKy8BcAjn8qht7tksYE
+ PkyKExuVB8vPfI5PvSvJ7oVKc6kHTi7fp+f4Iu3NusF28iQ uxjuTsWI7Qigc7vWiEIL1wkQJz
+ x5w3eYOuRntViFx9kObK7kvuszE/LgHOMfStG0vtOvpnikKwRS MW9CvbAP4VxzqSUdr2PPrS5
+ IStd269fuTJ20/wA+KeSMTxgToIwz5x/ez6nvVjyEiuS1is0oQneF f7x7Eegpbszx6a8TSLl5
+ MqyDG0jBP6YqvatcHUg7zxso+VgoxlWPP41x3k43b0Pbmp1KKm5K3bo/ n1LF2LuWFGYMgVWEs
+ YGDvP3RWAkVxJdslwoiRgN4xg7sYX9a2ob6KHU5o2indMjcS2clR8h/Kqtz qNzJayGPyTHs25
+ MfJJ6mro88fdSKoOvCXso2Sev3/eTyxKmmOyrLKN6RFlbo38Q/E1pWV1KqxrFH CrR7UcMuSpJ
+ 4NVpJAmlnzEJjWVUYDgqxAxn1NWW8kW8CHcZEUiQo2N+T/hWE3zRs11JiqkqSjP3r N7frc1mu
+ DatI0sLFA21mQ7RyvBHtzSSWxa3ha2nfMcJCB2LYIIwD6k9qzxJpiy3UX2iQAMBGsshO 9O/41
+ pRpZyBZ7aR5FmYlsPwmDjafc9q45Lks7P7i4KOHs1e/mrp+VtSC2trh7xGuRJbupwct2I+b 9c
+ Va81WmjUQTTytEVKo2MkenpxT2uf3UipaXEqyzeai7ucLjP4U2W7hMSyR7djjfIy9ju4APbIrN
+ uUndo6qdec58yX9fmStBsRirJbRmRfKEi5/dnjj8arCZ01Z4J5AYY45Izle55GfpV+WQTrLPIP
+ LP 2gPEn9xc8qfXPUUySdFnmuPKA3K23f3OcbvoOlRGTe6IpVXKUkkn29fxZBbRrDaefI+CWCy
+ N2zjl vwqq0l9bm0W3YyKvAkA4ZTyf06VoKIHtImZxLtUCQKcAMAePxNRyT3H2EC3lhEhG/wCZ
+ M+WccqRV Rk29vvFTlUk0uX79v8xJC7adIyTRNG1xvUleSB3B9B/SqbanapKiBo3kZQymIbd3P
+ P5mponme0Y3 AUxxNsUouAAwxt+vepDbW9vbwwuI47oKCrOuf3ajDfzqlyrSWvoOnXcFyyXXpt
+ 8+pBI0bA3KrJvZ wxGeDngkewp8MtxgRPbuYSjMCV9DhR9apmWFbuI/62NcIqqcbtx6/lVp7+P
+ dMlmrsvnjgtnJHce3 bFXKLta1zonUk0oxV/PoMLXSywuY2TeuenAwcfgKngjuGinN9BIio+2N
+ AfuLnlT7niq13dXjXrtI 0UNurHcCnVgM4HpVjcGW3lmvI4oJ4iy7s/N83B/MUmmorbXsNwcIr
+ mas+qv+HmTBJTEVQAhpFYBu u3JLH8KssYPmMUm6AYIyclgenNYJUhGmaZ3y2x0UkY+npVmG6g
+ VwFxIuCSA3TBwv50pU29imnKK5 btfcarG1P2mQwyF5HDbUOChx0pjWcq2R+1SqpAzlRjcR1I9
+ uaha5LzRAhQ7IUkwPvNnJx6cVfuby ONViWGV4JJyXLNnacDCfU1zvnTSRy1qzjaMdb767GRbx
+ Q/apgJt7u+8c/eGPvD09KazfbZlTeYl8 plCk5LHr+dXfsokmmkVTbyqdqo3XCnP9arXVyrzyI
+ 1nIyySeYRFhWBPHB+tbqXNLQ7It1GvZ6+b6 f16lcwgWmWtLnfHLkAPjI9BSPpzyXdribZCS7O
+ kmSQR0zUUwmgWKSRZYyrKz7zwSM5P0Gal23J0u RzFOGFwoikJ4HcA+ua1vJK6ZzVoWio3Su9/
+ 6uir58pZWR4y+zMhC9Sc4I/AGo4p5g0QHlvHMu9iV BMYA6H3q41tbTN5k0m2RH5CkjJzx+B5G
+ KpfYIbczCOK4TZK6sXkznI6fritE6bViJqjFcqWvf+rD obeGOS28wxGQRHYCox17+vNVGsrae
+ wEkry28u05+bjjq2PSlkKNPaW7tgFi0ZU4MaJ2Y9yTUn2uy 3RII5C85M0gZujA/dH4dRWq507
+ q9/wDhzOnKaqLlb/r1GRRQK0kC3UTIVwHbkMSOMfU1ckV10YQS SxyCPB2qMMSMFufaqjTBY4p
+ ljRAsLZ44LMeo9hxWjA159iX5I2lDqwJXIfAwWHtipqN6Nnqc81yu VrX0vpr62K9xcw3ssCsc
+ Kys2A3RCe/qRjrVQEW+o/LJJPIjkZJJHsamXEksUVttMqxuoO3ONxzTE ZooTaPEJLoRlVOBk9
+ 8/lWkUkrL7v1PSwtFtS5tF5vRee2w2V7YeW8cEgmBKyuD8qZ9vpSxNbtqEY 3v5aj52ZuN3t6C
+ nW4lcxSgIZtwwNowQOScfSr9y0BaCVoC8fzNGUwMKByp9T70pSSfLv8x1HTpz9 lBuV+qav+N/
+ zBNQiexfEItnk+b5wMcdMe3tV0XRigt3keCKWORY3kZeGBGc/hWdMI3t13jzYyF8k LxkD+H6m
+ nXMSy6WgKEzISzx5+ZeR/KsHTg7K24pUKdVQppWTe99vX+rDrpJplGyaOUFvnSNcEZ42 fUjms
+ hkMaqsSM4zt3A8Dnp+tW5wy+Z54YwiTflDgseOR7UyNZRfNLGVcIpZhjAYnoR7V00/diejF ul
+ Tave3b/P8A4BYlldknR1aErMFUrwFQevv71nRXNs+2KUmMIpVZCffgH61HFPLcRFpSoKnYWHR8
+ HP8A9anRR2huna5lAD5dgvBVjwBVqCimn+BwVOV+7q+unQjheOS9cM4ijLEvjjHsKgkEKNEY98
+ i4 JQ7sHHTrVtbp44CIYoisbKFdkHfg59SaaGLCeG3VGjLHy3ZeNo+91HarTad7aHPGo/ae+rR
+ Xn/wN y7PPZtYRKE2RFsBm5wx7f1pPsot0jlKSi3gBjdC2CHPfNZawBLRJBlkYI5IPC84596uX
+ OotMfJiR zGjYVs9s/eY96z9m1ZRenUiOFlFpQlo3rpsvJ9yylkY7aKR3E2+MqNgxuGcZHtU3n
+ pD5FqkckZfc FeVtwVD6/wAqoPOHW6PmsrGQIJM/KFPJIHbOK04UErTSxIQ8bmJd43bdw4z9Ky
+ mnvIxqQl9uV332 X9fIdN8gaVFSMwusakjgA8Ekd8VNZ2TpMZDexM8YKcqSoAPRh3JzTYWsprF
+ rf5p5mYeZsfHzAe9V x5hurmNw6AuquF4wxNZauLV7HZRalSdp2a30Wq+ZbmZWJaSF4XWRYpG7
+ euP8+tVZ3sriGeTz2kTz QibDjaD2PqeKs3Jiud0ckv2cRNtLseGYnr+H9agnGLSTyZLeXzyp2
+ Rrj5s4/PFFPS3QmnJRa55Ne m33hKtpHZXDW0qguWkTJJ3Dpmq8Nve/YUWM75HK7GxnpycfhVp
+ ZJhmCX7OwhfayGPBA6Af1rOLzL CD5oL+WV3A4XceBj6irinZo5KkajpuCVne+qvdf16i3VosS
+ tcorRF2O3zGyCuRzU4uIobwOLOS6W eRpMpjlV44qCG1dpopi5ePyismTxyMZHpUgh0+wsVi8q
+ 6uFDEFll5QYyT9DVSatyvV/15nnYh6Kn K7b9V+v6lDZDPKfJPkyGMsfMORkN1+mKrTLCDapeI
+ 8gt0JOzjfk8H3xUksTS+SLWTyT28w8kHrz6 CqUsc4d2kJkWBvJLjo2eRXXBJvf/ADOmDjWSg3
+ a33r5/8OaD2zGOSSecLbhhtbJwc/1PWnGOFtPV IGEMbFgN7Z3E44/HFVAssis94ssUEkg3OCB
+ 8w46VNeQEKypGxRgfNXuHUcfSptqk3/kQqicopVE2 nppp/wAOQxW6+ckUj7ZGQExr97jqPwFI
+ sFzOVti8nl28ZAIb7wOfm+g4pZEjZU8y4ETOpd5Sp+Zi PlIxyAa0obbyoGhKyXC5O90ONvy52
+ /U1U6lle4sRibtSk9Vt0t5roZptI0trEvLmJod7PnIJU9Px 6VDHJpqPIxt7pdzEqjScrkcVPH
+ qTq9mPISTJUKm3O0E9D6k1LMbf+2Z5r0qY4nMaxxjBORyfw4p3 ne0r/JnC5TlL95e/ZP8AQyU
+ KTwJFJgzhVjjPqeevvRBbpcrAju8PykrubhT3yPoM1cWOedA6w/ec EsP4V6Z+uf51VYFIgY1O
+ RJuifPCqOx9Tmt+a90jSpJykowWvS61v+IxoHLPtljkJnTy2A6oQaka0 uo9NmkmkSWdZlVtg4
+ 2sOB9asRpLLaSTDEfmOWjGMZUEcj2qpLBsvJIGuGiXcGdmBIJHOcUKbbtfY pTqS9+Mr8u+5ei
+ aOK1jXymd1DpGncDOR/jVaRrdUidQ9wTGfM8tyASep6fh9a1khOx7hpoYopZPM Vip6D0rJ4tg
+ xkuIZFOGARcZGc+lY02m20edRjOabir791fy/pox3tDFYRMjOA5PDnJHqfpTgYvLl NsWlIlGG
+ A6Keg6dfWrtoxt4rm3WFw8rEuZedozk052aAXci2f2aPIHIHGR/n867HUd7Gnve7Buz/ AA9DF
+ IfGzeHYEgMRwQPT9at28ai1t/OubeBZFyDs5AyRzjqanhMFvcpFIiyMFOeenrxioGZHuS3l +V
+ FvK+W3JUVbk3oVWpyu43aa1utfu1f4kZuIJ9UtApKQRK0bNz8p5NXYRb3c9k0UkNyI90bQwxlP
+ MYgnIGO1NWJonUqkQYHKRPGDuUckn1NXw2kO7y5KSGUBvKGxcHqwGOlZTmraJ/1/Xc8tvm6t/e
+ 9f zX3lqxMSXFuSGtnW0KM0rZEhwcN/SnW1ktzoUMUM1usYGclPmPrz7VZSOyWSTdcpAJGKIXO
+ d+e6+ wFNjkBeKOWNsRv5YCHGM8lfc9K4XNt3X9feYymlUVSnuv66mtFEP7NdlImj8xPqBu659
+ +n4Vm39t eSSEJGJACSEjGGGTyCafEGTQBEsjHzJVZZB0XB4H51PcKyyN5sc5lkyzhHxt2nkfU
+ iueN4zuenT5 FG8tf0/JL5DYI1huhb+WSHYlZTyMDJAPfmqqRI6CWaKQ/LtkC4ADNkgVHPbm2u
+ Z1ijuCA4VAWyT6 H8M1AJY7h5LRWeBWJkG5snco4z7DBreMW9Uzr9i3PmjL+vk7k8MN1E+5raa
+ QFgGjByZGXnePatSz uY7uVhLEYysnylu67Tmo7QSg4huo3llkVjkZ2gj5j9KZme0vYt0LOQpV
+ FUD68+uKyqPnbXUmVR1p uKav0eq/MtTMnl2xtI43X7MWQsuT14Ge/erfmtHAySMiuyiRlUY4X
+ uP60sc7+cTIq7mGI/lwpB6s PQU+SS2lEiSssksUioqrw3TOPzrlb6NHdFctoy1a3t+pXnXy7d
+ ZYZZDJIhZzvJA56D05q5BbI1zc yvBMTk5Ct8uR1OPYVJFLJCt280tsQs6CVdnfGRj0Ge1W1uZ
+ Jb4s7xDefMZFXGeOR+P8ASsp1JJWR NfGTpwdr8vfuQMi7j9n3tFMyyYB544yPY0k1pIb6eRm8
+ qMy+ZEr84UDG0/jVea5k3oxeNV3CNFC4 JUjJx9e3pVeTU3kDxxwTBWyYtzZwgIHP40owqPY5v
+ bOony9tf6/4Bf8As7BlAilZF5kCtjaw5AP8 6RlRJDO0MyQMgLOW4YscAj2FZllcLI8qxJcsRO
+ cAyZJHp+eav+YRFbpOSIiCzZP3eeAaJQlF2Zk5 Si/eV/K5Dexf6KEVZGdZU37WxuK8n9KgnEi
+ yXZF7AirMwjdxnCsNxXNX7m4Riw8+J2dGdNq4/wA5 7VmTeW2kh3+duQFB54xyfp0NaUr2VzWj
+ OUeW63f9biqYmhmWJC8wKsIgeVU8n8utXBLaN9q8xGdW uFVWi4zkZH51jLOwvDIhUF4+FHVxj
+ HFXvtVstqEit5TKkmWG7+7jH861qQd9jpxNSTmko3Xr+ZLc aj5spgSzmkYksy5B2k+tMedo1i
+ geEhdg8hXGflHf6ZqQPpxvXOJSwASRlbG0k5Gfesl55odTEk2W i6ZI+8RnAX05xxRTgmrJWNI
+ 4mM4unGNrd3+Wp0Fiv/LScxqRGcIf4euc+561nPArmNtwaOBEibZw ck5IPue1MkdjF/xMNyKy
+ AyFDtKP3B/SoN1gyXCxXW1XuVl5Y8gUQhJNv8loRh6tVNrX5amwypp8i ThXZfmBRjktzgMD+N
+ ObUo7C3ijSB3ZGbZv5ySMgnPXFWDIJtUuwAGi83ace68Eeg71Nb2CyRKuEm WKMRqCMk5GS3Pp
+ XJzQ3qFUVRc/36bt+KKEeqXJsre5up0YL8pUJg4J+YH3IqWI2ss00q3MSpx5eT zuB+VfqalWw
+ tG+zrIS4dQyYPHTmpZZbVIJUhiVptyeVtwAV6v+PvTcoXtFHVGpS5uSMdX91uhXFw El/0oK4M
+ RMiY5BPBP0Hemxq9vp3lR3KTRcSEHJJYdSD9KsyXCPfzeWqMjLI6naOoxj8O1ZxglujI OI5tu
+ fK6EY6n6dqcVffQqph41FdaFYPM175gMcls08ZVlXqOauNEsl0yAPJEZW3MpPyDqFb1YnvT BJ
+ bCSRUuYflJZVA7AcEVkOb6K4MqeaFypUn7uMcg+/PWt4wc3poP6pUqScnaL/P5D5LKOXUmhhkE
+ Nw0e5/M52Duv1NPMv2ey88rFPAZVYqqjcMcdfQmtHNw/mPbIiMqMAzjJxjr+HNZHl3Bs9skkdw
+ fl 3+WuNoPQfX/CtIy5/iZz1E6sl7X4eu3520+8ry3P2l5GlUhBnMSjBBP9KtrIqJaNb3IjwCw
+ 3nOV4 yv1p52R3bCe1ChG8tJccMpB5PrzSTWMccixGZXk8psKowRgZ5rTmg7LY7F9VkuSTcbbb
+ Nfht8yRd qX9zcrnyFzhwehPO0+9RNdxvBJMtu0U42tuf64qujy3Uc7rcwNvwWjCkE8ckfQVfi
+ ljs4YpYl/d5 KAuN4AznB96TVt1dm7lF2Vm326MrXAnlkeaOMZXcJ0Xgp6D8qglWSVLaNZBhY2
+ 3nJ5yM8fhUzwwr M0yNLLFcO2Nr8AHGAfU1TiufsrlbYbnUmMhuev3h9T0FaQu1obQxcpxUVun
+ 8/Q05lgXTwWWQHyyY WzkEcc/jTIr+CyuXuEPm5UKQ/O7I6j8arm4laSziWIYjhZTG/OFzzn1x
+ 1qGCJFspo5IWaNJE78vj OSD6dKSguW0v61D21qLhNd9L7/16jzPKb1ASPO8hlbPQHnIx9KbFJ
+ Hc3QkkZhEYdpCnHIXgfiaqO rQ6hHNKd0bkPkHG49OPxNRRlrd5zEDGsUiqyvzlhmt+RNaDliK
+ krwpu2nf8AUfLHJHbwowaJEALM e7dcVYN1HHHDJbRL5w8xZFkXOAemf1qBWeRvOdWl2xjcM9e
+ aqossT9dyiUZcjjHb88/pV8l9zlq0 Vo6i08tn67kqvcMkk7cmOSPbtX5cDPBHrV2N3RXkI/co
+ 5jRRwSXOc0C5EcojZQgRiJkK8sazraQS 2rT3mIyS2yLOC2P/ANdTbmTujKTppPmVr7bu/kX3d
+ 50SN7aUhJQFKHAPv9KYDJYXjlkATftcMOrD nH41H5qoYnimJ2ybio/iO3j8B3p0SyXhWSd+do
+ BA6+5oSstdjKlC1+ePuv1/zGRvs0+feNs7SAMj fw47/rWhDewwh5JEkNw7Kw2tgHjb09ealDW
+ cLw7rmDAUq4aMkliMAmqBjS6vYVKmGQJjeDxxxkis 7qd7rQ6aNSNSnJSg+VdX289vwNK0t7dc
+ TwS/vYZChJOQoGQSfU81YbUbq2kh2wLIuQGbb99v/rUk hjjsWihhfIYq7g8Mxxk0kNw7xeVPt
+ RmJbzCML8pwcfhXM/e1auZ0o05q8ld/p0sTeUhuGcoFjKEs pOTnr+dVmCMYQ0qJC2HZFyD7kH
+ tg1PE3laxczFgI1Yj5uQuR0I96alz81rmDcEjO7HYvxj8KWp01 IuS5Vt3Wn5kk+oRQ3IItjIj
+ 4dmGMt2Bz7U2OG1ighFy6hZk3Akkb8N2qEQfZvs8T289wAGXKng84 P86tF4L23isjZ3MXlkx7
+ nYE5Az/SpfKkrbdWcFSrSi1ON0utmR3L+XbpNDl7eQCVwvXcpwAD2B4o cmbTkYbWYxu0qY53b
+ h/L0qhPJcWI2grycmNhk5bByPbtWebkkho5gJG3BlAI2ZOOfwzW0KDaTX3k zwqlZv79QZ8osi
+ RyBpCDEM9s81auppUgYxRYRZf3pdcjfVqeSzW4iiimjZEcbWB6gcgj8KzZ7mIt GQHSKeB5CXO
+ QWz8v8q1j7zXumFPkqVUqkNPO4Ti7lKO21YmRn2sOqnrj8amsbwR20Fv5TSyqNif7 QOfm/A1F
+ DJa29pDvZpLlvlJLZCBhzkVJO6GKJbcxstsuNyjBbn19x/KqkrrltoTVXtkqcaeib6NW /wA/v
+ LEEVlsnQt9qKg+YU6A4+XHoBUCyz20dvcWkgLtBmVWGd56BvoO9NLz+VKwCW5UfL8oyytzu /D
+ +tN2Ty6MS0L+bGxRCOAA3LA+tTy93oZum4uKqStF6NN/oE0rCaK4mkhOSGQImAyqOv4npTLgCe
+ yile5gVZmLFtuMMM4/Oo4rO5PlxFPMETbAmMkgckj2GasLbWEsmFkwypnlsq4PO4e1W3GLVnt2
+ MH JKakparsk9CqqqVhRfMk2xlHkjbAZ2OePYdKt2/2lbG5S7twUkuIxKdg/dE8fh24qVI7fa5
+ QgtuU RgHG9eu8ewrOEsyWsmZBmdt2WzjeDz+lK/Pp/XcxqVvaOyaXrv8AInS8SC9uFcgxQ5hZ
+ Qv3u2R6c 1Sku1ujH9oXbtXACjBPH9akltbZ7V5beR2LSMwQnJYAjLfQc5q/PPbwaXNPC1vPKz
+ EgBfugkYH5Z NXeKaaWr0CdSnzx5U3J6dV+X5mUlxPLLJC6OsciHykPO09BWfJt+1qJIZNsJCH
+ J6EdjWje3UjmRB GkAJOH24Z+nzA+gH86zfKcxOqiRzJJ9oXJyQRwQfWuqnte1jatKVrRjZev6
+ 6E1zPFJLKodnmLk7l PHQZGPoMVG8TSwW1wkihPLYgE54B4J+lQuGZFkmVVMkZIAXBJ6A0WUcs
+ V9bTsDFD5bsA4yGHTp71 ajaOj2OWpdQXJq107+X9IuxJNcaTl2ijkO5i5TllB4YH0xWbM8TR2
+ qqNrhHyeMsO2ferZmJt43Id Sp/dgHA2n7yn1qi0QjhEKrnOGB7r360U1rqcmGoVXa8LO/f/AC
+ /Uv2awCVVd3jgCtvZm7kcfhnHF XEjtn+z+VtOUK3Hy8I56D8ax7ZjHOrPIi7RjDISMHnmnzTM
+ 1wHjDRMwO9ccEf4980TpuUtGddaDn UbgmrLorK/mdPcxRRwx/aoXJiUoCDjA43fiOOadKtt5M
+ rBJd7tkYk7ADmksr6O5tEieeKQRRFSGG S5PJbntxVmLUIrS0vH/dyTb0c5UEcDoPrmvMfOna2
+ v8AwTx41qqlyRT5+urtb7i7ZRBoTBHJH5DH JJ5JIyVI9OaqzXM8Ytp2/eTFFYvj5Wf0/KpZYo
+ 2DkrKBHKRuR9oHy5FQ3E1obaPaxSTH7lGbJCsO p+hrGGsr73PTw9V2va8PMoz3krS2txcP5i7
+ d4CDBIU4B/XP4VCbi5M8SW8SlpJciXaMD2/GmTwSS zW6Rt5swty0hTowB4IHYVEzs0aedbywj
+ a/mnOAGP3f0x+dd0YRstP69DrnQpcqlbX06ely8zSxpN HvSFtwk9Dz2/PFPmWRoEnd3V0VhKC
+ f48gAD061HbXGm3NpFLPHKobAIL9wMY/SrP2GUWgkhJDuQz Rtyc5/p1NZuSi0nodeCr0YOKWi
+ v1W50Nus1vbODA11MisSq87Rgcc+9QwlNyugCXMh6MM7Tjv71U Nw9oywz3CSCSNpHdSeo+7+B
+ pYHk/s4eXcwkD5ZAyZZXboufXFcDpuzfcxSUbycd3uXdQMk1l5bTQ KLn98zBfvbR1Ht2rOhju
+ 0iuD5ckvluoGDgrkdD+dThmCeUVwyp5aI3PH94ew709lkguYYTKoGxvL bJwyqO/qT604Nxjyl
+ 08RKlT5W+byaLVjcLbzSSXTRSKVGeOrEEDHpUXl2Uott0U4ZIwku2TGw5OQ femRW8XlBp2zGw
+ LKAcFSPu5/Gqf2bfq7QC+jhUBRITnlz0/PrUqMXJtOxPNGo5VFLla9TV27Z1+z BYAswXzDznj
+ Az+NQzQ3rTJJIoe3izE8YGGDnkZNVbotBLJgmZNjeYUJ4fjj2PerdhPDPFdNNcFC8 qyKpPTav
+ Q+9S4uMedanPWo1Ir2kHzX8l/SMi1t5Li+DyLLbyL0VjwVHXH41uzWhN/CyRPcRFZJ5V jOD8o
+ wBTZILe9szNb3KxyOyquSf4u341Pb2VzFZyW7u4aGb7OTz8yscmlVrX1vbyMcZjYS5W5a9t Uz
+ F+0FDE4WMkR7sFB064/HrVdobZ7mI7pZ5XV5MRPgEAdfp/hWpcGOKeWNF3M43lcdQBjj2FZuLh
+ oYdpiESIYy4XlApyQT710QldXWhvGrSUk3dP1HxG6hhRp7WRoTGA7AAfMRgVnSDybmeKeQpsmS
+ Mu /IC4yfx96lee4k+0JCWfcN+M8YBB3fTFPghSXzXWCVw0scoLNnCdwfUit0uW7Z0z5qadS92
+ +3/BG pDmwkEjSkDI5fJyM5H16Vo2dhb/Y4Y1w0uV3g8k4PI/rVaWd7Z32Wsgh+bypG5DqeM/n
+ SW4vJJjL cKyJFJg7fl6L/U4rOfO43vYiNOs6PNL5K+/3G/dxNv2BXSGN9pdTjdznr9Knhv7Y3
+ ds1sszYR9yh s9fX6VjW85HkQsxtmmPztM24MR1I9B2qxaEfbZUMfmKswMZj4MYUHIb17Vxype
+ 60+h2U4XptSVmv UvyO6x24jjfYFU7s5zt649OtOnkd4dtmI2lLMSNuSg6AH8M00XLl5ljCvk+
+ Zux8qEL938etTw3Jt rSBvI/1sS4bAwM9B9cms2mrO2p3fVpwjGX9fiPjjKw4SSKQiRQ6qPmOO
+ hHoPX1prQqNQmWRiQjFW wTl2Azke3tSsbpkeKRo41jOHbZjcVHGPqTTRJcLA53xRzrMmSyZ2M
+ F6H61HvdyalOo4tp7/12KVy Y/It7lYGjMcfzPxtUnnBH0pFW8aaWVGj/e7QkTx5wRUF/DP9g3
+ PFI0CyjcFONoAztPqeetTXFoJk S4S88mMxOQCTwcDFdC5VFa7/AD8ypezhRj3lp3T8inLczoS
+ Xy7OGbbHx8o+Xj+dQLGsttFBPcoeM fJ8pJzkE+2KIC8Ee62kSeLaMoRlgSM4z+pqjDLGl1bTP
+ l41/czFegZs4b6V1Rho7f1/kS6cVCXdf P81oLdFo98Ak3bcgHPXBB/OrgmhuJmW4SUOZWdSGx
+ yB0qwkMSynZLG8aHyWB5Oeuc+prDuEhWJwj mYqxbYMkgY5qoWnoZU3DESjTbtbW+paiZreTyx
+ NDArK6KrjLEsM4qKWa5jtYw4XYgQoGQENx15rL WUhxKsifu5AFB53ZGMitOK5vIxMTGWljyrn
+ aGVScYGDxmt5U3F30O2M6lKfI1Fp/L+vuZHIlxHcR hI2KtExRh0IHUioZpQsqExsVVAuBgdR1
+ zjrTXuFGnwkCQSOcyEtxjPCj0zzVqXc4kLRbEjZs5H3W IGAcVabT1R2UMZKcveVr6X/ytr+Qv
+ n2xsgYmKygqMk9hwAf1qQRCOwhmklJaKMhUyeRn/wCvUEdr KY4o0VCZAJWGMlWUdKRz5qONks
+ Yd1yWOcORwKhpXsmcHPSc+Xm0W76/jqWYJjLHFK0BMUb71J5AP Py/U1Uu45VltXT5nZP3qehP
+ c/hUiRTiyaFj8kL4nPq+Dge2OhqB0dfJ8+YSgfM0aEhvzxRFJSujl p0UqjcVa/wCP6fiiwFVL
+ YxRxy5kLBeeWAHQVBGbSWxzJMd0qAgAk7McYPvTIm8qSKRSTsOWU87jn HH509ImjupgqIfL+U
+ 7kyoq+Vrr/X9WOmGFxCfLKVktVrb8NSoGJWXzJkaQtuYhetWoWEljLCQrYu IykgHA4xj6Vcnj
+ ki0yA+XBL5n3HRABg8HNN8m2huHQLIhjkMe4njHbPvSdVSj/XQqU/b0kkrtP1M uS1ZVlXEhli
+ nCDHQAg8H3zUsQnUpsDCKC42tnrz1ye+KvzyXeUjJjXOOduN3v9aW2ZVeYQRuXMgY h+e/X86b
+ qScdQnQbp2aTt/X9aEb2Rd7jkMBIN6jqrdqV4DLIdriWRWCtsGMk9x7U6BGgJjYtI0r+ YVB5K
+ pkn9asbA5edQxEqgqqcEZ7/AIHis+dp7nXQq1Ixeune36Ec7XEQBlw5DYyo4JPIFaTWsAgM Zm
+ VbqM7NhPJ3DLfnUsziw8gGHz2l5U9QmOMtnrzU9tZ3Mm9J4wud0nmbeQ46ZNck6uiexyzrQnOL
+ ulbrt+n6mQk0sMokiZVwArRSDcSADg/hWskk8mlQIhglEqpIzKnRh0/D2qF7RxpxWcq028AMox
+ we tTsst5A0EEiQlTuChMEAf0pVJRlZm1b2dSmmrNp6t9hsc9zcJcpHG0VxLIZg7j5QF6gemao
+ TzTbk dHQqUICAYZQ3TJ7nrzT2jleAmebKOSyyJkBVLYIqN2CagySMq+TmJAOwcZyfXFOMUnoj
+ jknLWCSS +f4/oZ8sCzBI4bgONrGPk5wOAT+tE6LFpM0C/ZxiY5fbyTxjn3qIK0M5kZTNFEApM
+ Zxu545qxIft dzcNJG0UDNuZ+yt2Fdeqa10RpiaUozTTuo6303IY4YhcCIqXVSGDA/cPYH15p0
+ oiVDayQFpI/lIX ggk5yPYGojIz237wEuU3yBePnzwP61CxhliLrP5cvmiNg5O47uapRbd2ee8
+ N7/tJpq+ztpf1LbGB pZXtmiaU4GSPlbcOw/OqN0JY4m8sAGErGGA4YEdfc1bispk1EpEyqY3G
+ 4Hkgj/6xNW5raGSVFjDA tvkV2bI2g4GfpS9pGMlrcdavGlU5ea/9f13IY5kMc8e8IyS4DtyBH
+ jkfhTbeGK2u5J5DJLGG2RkE AAY7+p5qKFZVlGCnyk4BTO5e5P0qzJdTRG3gkltvsnmEZ2cgD5
+ tpPqexqZJp2XUwxkZUpcri0nvq 9V9w0MtsZZDHL5ynDNu+VvcD6cU+a3Atrie1QyASBFA6gEA
+ Y/OqcV3cG/ZgY2EiM7AjPQYJ+lV7p ZFslkklI3IBEF43L3B9SPWmqcuZXOapRnCqoOyf33X9e
+ ZetpmZreSCSJpLfKTfLw574/lVAGS81K NEkigLtnLLwuQcr9apCO5tUSNZY8PG/Qcrzk596fH
+ O8srJcRmXch2tEdpBxgGt1Stdo54YdTUpJa 9+33ktyluJvs9rDceaR8vz/6vjODUcM0DReakq
+ ZdG8zcMjJOBgfhU0V5J5CM0Qh2KB5hTODjGD9R StEdsUUJguABtxGmDnr+lUnZWZU4y05nZd7
+ /AK7fIrQ37TWbys8S7XAQOmcAdPzqQXsf2gPMjElN qhRjHOTT0tjDpyy+Syt5Q37gMZLHBxUw
+ WVkiWWS1kYjcieWOGJ4B4+tKThrZE1I04Qumtd3s/wAm ZMitcz+bK+yN32owGFB6gVfXT34mb
+ f5DgKxz39vQCqIguXuJoIY5M+fu2f3MZqxcs5smQrITGoG7 OF59q0k27KLF7Kc1pP7t16bGZc
+ 281oIVZ1Eb5PzdSAeo9Kz5vP8APiCzDYxz908itWSWJkJjjYbc qpc7hg9OKryPbkp8mOBjcet
+ ddOTtqjVUZypPmla3qm/u0Ic/daaWMfLg4+Xn8qsecUubeRURI49y OzDPJ4wT61Uz8yywlJDv
+ 3AFM7fz4p2x5JGW2UyLyxBG7k03FdTln78bP4e7b/LRFyygViiksksW5 ZADjIPf6VrWkcUdsv
+ 2dXnDyLzuz8uDk1hRrlkEciuTEc7euAauG9JtraBQUMULKhHG7JyDXPWhKT 0Zy18NVTUYz372
+ aS9f0Okvb8ShZYwwt3jdSgONpJAGT+FZqSXFvpUYyhdZCnlFMuvfrW2tvBYxXJ tHiuplIG3G7
+ Izzwf0+lZ17Z6ozFYIWigEnzlwCdgHAz65zz3yK4aUoaRWi8z0VUUFBW0T30/9JbT X6g89wuP
+ N8os7mPcgxlAM7h7ZxVUS3dwpWWaEM7jgrgMvdxx0GKuMto2lI9zMFeSVEgXPOGGD+oq hIjW9
+ 0UM8c/2VigZFxu7H8O1a07PZam+GqXbVN3f5GjbwQ+TJCrLKi3AKhRzjqD9KszmG2niks7w M8
+ sLt5ZYnbz/APrrMi2NdNG+6ORX2pg4Geuw+56VcnuFuDbR21vuuWDArs5iGeB+hzWcovnXY7Kt
+ KFOpfmunvsvvNCLZPcSyOvneXMAhX+IEcfgTmlubaGSLq/zuWkVf4cevvjNUivlRvcNdRK6N+7
+ AB AKk8Ej37VIZbq1DzR3Nu00b4ZGTqOvT8aw5XzJxZzyo1PaqdKXpuv0tbzLcE8EEPmxhwJji
+ ISHJ2 g4BHtzzTL1Un1m2tzcCEWiMskhPDnqCPY9Kkljm2LKkW+6dS8S8YjUEZXHqc0/zCby5k
+ AjaWMhXU r1BX730xWStfmX9dBPlk25tv87lszl4LhIFBkllVgP8AnkOmD/Oo7uSEpKkmyNXBL
+ SgdCuMD1ye1 UzvTyt1tMlqUKRgH5gGH3ie/NUY7Bmw0kwaKBfL9iSpP86UKUU7tmlKlCKvKVu
+ 3/AA//AADW85vs 8z+bGYpSJRhfu4OOar3d08ojMBjhaUkPlOFJ4x9e9Ogjhh06CCRZVmFoVkD
+ NnDZBNOWMtesw2g4y UYZ2E+v4c0LlUrmMqkVPmta3f/gEVtGE06NUc/u2Csp6gjP8qum9SaOS
+ C5uhE0k24tkjhOv59qyZ UjsGWWBjcIxyCG/1ijgEfUmpNqvKJIruC2nj3mVZoycgdcVcqak+Y
+ HRdT3ld+aW35/eWrlJpXijL iPzVYrn7ygclCfXHNJFbQQu3mSsodWK7m4LHgVfhmknuojFGJM
+ li6gcxkjhT796o3W9o2Vp0hWMh k3DlwB1X2BrKMpP3djkoOpKbgtGv62K0YCtEl4PIieMhpOg
+ yBjFK0VmNkTLcNGcIHSTAJPAHNILZ rmSCOdxJvy8pBxtccZ+mO1RvBPbX8sjgLFH94kZCuFwP
+ 0Nbqze+p1UKbvLln7/ZaL/hyxL5ccYtX WWIJGTukbIJGCcfyp9rPO939olkimjILEovyjPXj2
+ plrEshJguorm7IyAVJGPTHvVwZS3EjQG1yX K7+ijgbTjuetZyaSsdUq1OLtvff+tySC3tFjCM
+ j3LybfKZW6gnoPerQtHmlkgkjdZoW8tFRsF16u frWQikPNAqvE6vj5jnGBnj8OafczTrPb3cj
+ O7NvwEYgnoP5Vm4Sb0ZvBSfvRevTV/wCZs29l5MpP zLAgKqrHkhucn6CkmilN1bQ2ys0UduVi
+ Oc5H3ifwNU7DUJSXV1Mwb5jjHY8L9a2A0ovY2SBgEVlJ z98YySPQZrnn7SEveOx1a1J3mtfl/
+ mNuVNxYeZbJJulO7G7O4NgEj6Yqo8Ust3H9qguI7cRNI8e/ DOBwDmlS+uYdLEwaKN2dXTcmcq
+ ODj6VbW5AneUHzWU5dycqvbb+INSlOGyFGUlD4bp7WetzMN3FK sEcUm1JVOfMJPGOPxNJHNHc
+ yxqXUxrCElAOMEngfWopVWV0W4iMboR5Kr8pZAPvcVYktolt7yWOR I4TIGRz3xyP61u+VKw5V
+ ZWtG93t5P/MhRre0nQjDsy/6tf4SPlAPuRUKRpDFcpAqRtFIqfvRu3EZ OfwHWh9NWW1uZrecP
+ 5kgcnJ4HWssSQNdtBK0qROCyMW+97/jWsIKV2n6nHGFKo5VFK7XxafpsUUv pGs5JGcC5kcYZe
+ AVPXj14rUitdqM/mRxJLnmQZLkHhh7Y7Uu6C31GZTbiZRc4PyjCfJkVTleS6nh VMwRbBwx+6D
+ wR9TXS3zbaIbrOV+TRLdsjSzaSaSeW3biQMGXAQjvx/KrE87S6dEfs8sMTFpJSSM7 iwGPypFl
+ maSK1jdYcKyv5vIT0B96WELKsHmXMUrOmDGoII3HFOTd7y6ETi3UVSotvX7+xUdpZL2F IEUxl
+ cxIV5I3fz61OL2VtXcw2T3cTlkwBkEDjceO3rUMsEaahJGjOiRnYCT8yn0z+tMd5I42cNtw 4D
+ ovBYkcEe3qK15YyWx6FWDs+RaND4vtJuQiCQIMMrHuq54/Gp5orqG3juoJUAcZdCuTkHkfWodz
+ JPcfvlgZJFjw3O44Iz9Kv2yOtlDCY2mWWPdJIDwCOorOpKzTOTERcZXsrPfT/MymuViuZVkYyI
+ z8 Ecb8HipWR5Bb7Co25jlyOV9zVyCCW6aZSmGBMiqFGSNuOP51BGlytr5SqMBgzHb2HU/QVXO
+ ntudq qqok4u0l+C/ryAx7LMQrAdpypLY3bieBmoY2YWkqzEhWkAYd9wBz+XGauh45J2ltpDkl
+ 2R2OVx0H Xv15qSG12WNzHMVhjdlcO4zg45pc9lqddBT7K34/qVEV8W8brIVBJky3G329Oasww
+ 2812HQS3GDn APXA60sz25gikihnQsuBKzZVselWYIGlikiWMo0S9hg/T86icvdvsaVKS5HKL3
+ 03/wAhslsyRmUR m5BT5lQ4OQM7h7DvVIIhh3NbTtuOZJEbhj3IyOBjmtFmaEETRTsGbfuVsAt
+ 3H09qfGfPs3RsIC67 UA5Xjqfas1OUVdmK9rCKlLXzTf6fqZkUFvI7gz7o43PlFeGcMOfw4q3Z
+ oWnt4XUxmKM+axPHXJ+n FPMa3FiVSFllSZV2rwWXHzEfhUhZEhuBDBOoklDKzHOAByKJTbViY
+ zi7qKavtr177/oTvObi+VEl RLcqGy65OCRgZpv2vyZpEa5E/mb3UpkAc9KLiPmCdXSFnIXbtx
+ kfwgU42luJXuJoJYn+YtGWGVyM fkOtYr2elxxhh3JPl07abjVYzyxfaJEETKXJ5HzdAtLcHbe
+ wQhmQxHbchTguSMbh6AdxVmyjlJgW 4h8hYY1QxOBuBz1P6fnVRpn826LwmJxMqNMw+VMclT78
+ Clf3rLoVU5Ze4nttbb7u4q2NlLGlvDfK h6eYWJUY6DHvzUN1bWSokkxlkUEttV8Eeise571Pc
+ rHBcvcRW0hlY78Z4XHt7E81WhuZbu6tY5EW NTC20MnXPGT+NOLn8SlocVVT91yqPkW+quvuK0
+ UUCTypGssNoVBkeVt2GB/+vTyZp3jtYwjxPC33 V5HOMH3pkEsFjqEQ8uSXEZ81G5AwOevvzU1
+ 0JglognjzKdoVBhpCOTj0raV+YjGWhJO9n0b/AKuZ 8xdYYrcW4WURmQM4zkKORVBIzMUEiLCu
+ C4YjAz1BOO3pWg9vLLIPNV0hlO9pv4VYZGPYH0qBGW2v LVJ0aR5IsKgONuD3rphKy03MvbRjS
+ vDV+WuvqwB/cRIJDJKG271ON2eefU5oljaV1Y3KhiMbEz8m e39amLSS6pIYo1ZncSAIvAIGMj
+ 0FRnfJcLJco0ZAALAYx7/n+lJXuFTC1oys997X/T/IWdbaPz1+ 0JI4l2qUGNyNgGoYXtIMiWC
+ YiDzFILctngP9K0YkVoYPMntZHRWErCP7xJyo/OlTV1ghJNuhnLDc rqDksKjnk1ypN/OxwV1V
+ qacrfzsjJQ3LXVokCKxjQtgLzIAOQKgvW+06dZtEksYVirFmyE3dAfpj NakkjvcwLKVDquNqD
+ aQSf61TffFKiRSxy/eZTjgY45Hrk1rCXvJ21/4c4K6caqaVmvN/1+BDaWTS WxWY+bcSSfumj4
+ GwZ6+5psZ+zNbmZ40Z49yx4+bgkAZojkuZ4ijzJBMWHloVwQoyG6VWa3lazjn8 xZltXEG9Rwy
+ sTz+taattSZhUqzc+Wc21/Xf9BywRKSJ7gOi4xtyeD1H15qSeSYOBFIkkcb7Q6Jgj Pc8d6Rba
+ eC+jVrWWQRoUlUrnLHOMehqKCQxXNtFImxRGY2OMZOc5Nab67nouVNrmT5rdNPyV2W7i ytGtg
+ ftMqvHPtCsxOF4yCKmF+2y4RYVdTOroQMEKO1UpriRrqZjGrGRwTxwB3pLaYNbSjHlnzv3c WP
+ mZe/NR7NuPvaieHpODcru/fb+vVjb5ZbnU2cs0R3MdwHVsdOKhVjPGipbmUScKI8ZXnkH3qW4a
+ G5uVaAOoJwFLZPTn8qqzwvGINknlhv3jZ/jA7j0raGyWwVHVpRcYaXWyX/BKsoKS+XuULvG4YP
+ AP +FOAE8YBxcgPtOxSBzyB068VJMrOHmkXyGWQY3dAOwPHWq7gM0qYUEys7benTgY9ufzrdO6
+ Odyqu XIne+9/6/UPs4s5BF5ckYYFhvJz79qQW+IzJ83UEDJ47c8VVmtZ4ju84DrkMOxqKS6m8
+ mOMqclvL Pzcgk5B+gPFaqLezIdRUItVo25dkl/w7/EvqyK7heCyHAA56dar4e41C1/eoFHUnj
+ juKaYlmBcRS xmI4f5uhz0/StBPOuRMkcaRqz7x8oyOMY/Gpk+XUyxLU1dpcvc6kzxrcWz28qG
+ WJZEkfqGJHynH4 /pWpG3mQQmW6EkcahNqkjecjJ/PiuY2W4uGihbfIW2qT0bvuFT2nnz+Qt4r
+ CSQnyyoADoT1x6A/z ryKlBWTv/maYjDw5UnLlXV2V/wCvQtX0ofVJsGNIWuNy5TOwKBwPxpIC
+ ks0kk08JBIDDbyS3A/Gp r0RItvard2rBo3D5HKsPu546mn2N1HC9tbmENcrEfOUoPlK85+uKV
+ 7U7pf1/SLUlCgqkL3Xy077C L+5nYSmJDFIowV5XPBz6n3q9b6hdNPvhhhwFZCgiG4N0GT/nrW
+ ebiOeG8OzALh4mYZ3KOT9ajjjm nlEgWZ0lQsrJwCAaiVNNXkdE6VGS9pLd97W/QsvA6zSNemN
+ FBAcFfusF4Wli1GOE5V4cnDMXjz8x /pTY1trK9bzZzL5kgePechsjgn15p1u/2zZC9j5mMNcB
+ QFYvgjAPbpQ0mveV19wOlTUX7SDkvu/B /wCZcN/LdWpllChlZj5qDaoY8Y/Gqs63U7Ov2qKXy
+ yqOI1wZCVOMfyqoEvnWPMTwJKPkUrwOv54F OtnYxRhI3JEZeOQHgqp5z6ketCpqCvG39f10N6
+ UY0oJwS/P/AIP3FnLpZKAJYW2qxaUkgMODx6DP SpltktxKnm+ZEHMiyAnB2jkn9aSPUHlumEf
+ leVJtEe9c4DHkn8RUNzFeDTbuJmQwGThtv3s9MH0P TFRZ3s9CasnKqoysk/P8rotR7Lqzd47h
+ fKMgbJzlsc4FURdXdxJE88iZnfd8q4JwfmH4DFEV9Hbx sksDJFISzBRjYRxs+uK1dOv4nRYka
+ 2ljhZ0jfYPlj6n8T60pqUE3y3OLEOrBqMY389zKZInupJYy VMPESkkgj1+nNallZLdLIPMh/d
+ p5bqF+c992fyqnLZ6emnM0UzbkXbneSORkZ/lTbCRlihk8wTxl R5qx8NjJBGfX/CnNuULxex6
+ 1So/YKdK6e2qSZoqJba5w8ciuxDGXOFzjG3681Qe7WS4SKWaKOJVK NuXkNnp+lPu7dEglWP7R
+ vJPll3z1I/pTIo7e5nlWGEzskhKqG5IwRj6ipgo25mYwownLmW738vm2 aMtwlp5dzgOTHIrr/
+ tY4Htwc1Qs3uprMBFEssMibwwyGz6+pq4bT5o8SBJRbsWR8nb8vU1ammnfT YJY7yxhcKGm2xd
+ 8ZA+pHSsVKKVluzCVKlCKT3b31Ms6beJNHIJFbehACLtKjP3T781Q8ue3lQPHN GIj5e2Rid5L
+ dB71ckJuILSRROxMY2RqxBPzcH8O9Thpv7VLjExkLOEIyRt+tdCqSS11/A7KVVRjq 1Jeegrah
+ N50kKQq0pbc6EfN6EfXHFRx3DW4L+TIHiPlqJDkAE5APvRNGkNvFeOkiIQZAucMB0AJ/ HNTmz
+ tDYsHn865Rw7Krnop/ng1F6aW2hs8RF2apvle9no/XU1ofJuMXJkjlmB2qkI28Nycj2NN8m RJ
+ WSK5MMucCRzlZPl5I9BzVW3kl/tOTZGV86f5owozyuMD0zVS5Bkt4AEmSSJQqqzffUHlvpXNGD
+ 5rX0M6FV+35ItJPvqaVw8U+mSKsyRy2x8sow6hwMfljmoI2nijmb7VbCF5VkPyH5x0Zh7CpI1u
+ Jw 8pjSQSZlwigEsBgc/WmzRXEMZtoVzeTkO6kZEQx84I/XiiNl7t/66h7ifI3Zd9H+mn3FZ5n
+ GpNC/ +kIFPlugx5g6DHtiqK31vJbrEokMe0ADdnBzjB/CrSytYqktxGZwFAjdejL04/MVntFb
+ QxGU200Z gmREy/Xkkg+p5rqhGL6F1ZRnPSGnR3/4KJ4rk+cyRMzbnBKDuR0I/wBnHaqVxeYvJ
+ HBiBEWxMp1B HQVduordWtNgaVXDiF4zjCDnJ9frSTiOWZllkgtodmI2dckflVwcL3saUKlO7l
+ KNvVf5bjjLJb6T KH2lFdvkK/NIMAZz7ZqlHFFdyRpAkoQfu1lJyHHXPTtipoTdHSrcqYrsvgK
+ iryAD0/H+lU2FtdSQ +WJIZoC3lENhQT1B96qEbX/r8Dmw0qii40klq7vXRen/AACYrE5Bjt5R
+ MHJiO774IyAfU9a2LZok jiR4o4lki3s20AoRnCk+tYAsnRbeFJGd3XzFKsf3Z6EGrIW9e+4tp
+ 1Cy+UUPVdwxz/OirBSVuYur Dmi1KasvW/4kFmYra+zG/knG5GuPmD9Se34VYBSXWFmkXCMuXI
+ 6AgcjHt0+tWf7PlWGC2lnhIC7Q 23liT8pB9KtRW26/S2urZxIwKySKSFLA9MY4z9aU60NWbwr
+ 05U5N7Ja2aTt313/EzpoxPZwmFN7F clRy2eufwFQW0a3EEUbu8XHOW4CA8j6mrMkkdrqfnxQS
+ iFEaNI93UnufpUaRJcqJJpoo23gOo4LZ GCw9qpSfL5BOo5U7JtR6SV2/nYdeWhglaSBmSJ5So
+ Jb7ikc5pli91sESQSyoyjaMdh0/OnvIsUC2 tvMLhUQlJApwB1wcjk1es4C9tBK0U+1kBba2Aw
+ znI9PSlKbjT946IRnSw37zW+2lv8vzIGt7VIJ4 445IZkfZ5bPzt6//AK6beO0SrAhBUph1IyR
+ jFXLyKdlE9soG87XLDJJbrj0xTreBrfUsT27XDoCo C/x8c1nGorcz18jqoVYOldO7XS/6sSEW
+ 66XcLLG7AXCiJd2cLkZpgW5aSYxRuZHlaNsfxtnOR6AV phbaW2Xd8iyxCYKTyhB6Gs12vVjla
+ 4haRWYAGL5c55JH6VlCV2/1LpxcXeP3X/4a/wB5K8cy6ewR WeVTwx5AB69fypGkvI7RJY/KxE
+ +zOwfKP4c+pNVpp/LaRSJDsx5YB+8uOTTbd4pmjtNs9tFO29Wd s4x/F9K05Ha7RpWpuMbzSSW
+ r6/gacswiR2tLiFpzPtVduc9Mj9TSRtbpZyrOWO2cEbTg4HGPrzn6 UxjErqIpY5pgU8xlXgnP
+ GPTNWVuJWvZGNk5iYliuBken5HrXO9F/SZwVJw5ubdLXsytfXcUjCGCN ss6AMeig8VI8JIm+0
+ 3kaNHIURtpw+eCfcA0yKMREXUg2jz0JJ6MO9aM/myRTRLHFIs8m62YgfdHc /XpQ5KNktjr9pF
+ SjFO0O779ejMuG5eR5Wn3TgFclTjGOSK1nure4QrsEZKtkNz1G7J+vY1lOGECR CBm3JkheCXb
+ jFNurNEvI4FuQsm0KAepULyfwpyhCT7GeIVGfvN2a7fn/AEiSLy7rSjJBNnzFXezH IDZ6fl1q
+ nbuzzvPdSrnft3JwBk9Me+K2UMVvpj+W0LQtKMkLjJOMY+lZshimu7mGOPyysvyMfukA ZJx60
+ Qne6tocEcXUmnBrQqBW/tLmPgoWLYGQOScn0z1qrb3E7PbCS2kuH8sSYUcggnBHpV2FWMWL Nt
+ 8Ux8+QuM7e5T8ccVqwRv5nmeXi3YhlKgAqoH3c+3etZ1VFaoWJrR5fehr2ujLiuYnla4WRVYMS
+ 0JP3yehA6ACssRBruNY7iKVwmd5XOOPm/wAK3J7R2vGEaLl2DiQLwBjB/wAaoXCzC8eNTCzsPv
+ xx 44z/ACq6U430MsNKLctenW2n4f8ABKLSCOUlLhQxQMQo5BHAWlso8kJMXJdd5ZmJDnd94ew
+ 9KvQ2 6BZTPCJJY32DGPmwMhvpVUz/AL6znELxIiPGpY8AHqT7Vrz8yaRGJrQqRaite5OLOJbh
+ zbs2USRM k5Vm6j9M1kok93I22PftdWYAc7sHHPpxV1IIoLkJcXiSW4DKux8EnHB6VmxSSiQC3
+ lXPy+YwPBIz z7CtKadnrfzM6cHyySfNpvrb5vf7jRmv2YPlUlZpFl+Ucj5eF/rVBEE6wTy285
+ aRNqIhxuGeWqK7 uX862VWSRXjLL5S4LEHr07U/7SkmntIWId3Mke04wB6e1WqbjFWW559WUoR
+ 5YKzejs/+GZlS7H8p YQ21W27mOSzE9fpWj5scek3UMzssq3SlVTjtyfwqoCzR3EoVQfOR41AA
+ 9Tn6Uryqtys8jQ3DHcJE C9T2b6V0yV7L+u4V1pHmWnTvf7ixdXcguwiSswD/AHgfvnjBqGO5a
+ DUFlaEN5UhRm25CseckVTa3 gm1GNTMqgLkSHIXjrT98ZeUkLH+82hWHJHb8apU4KNjeEI2dGL
+ 5U9+t15vYdJdGRycbd4xICOjdB 9Kngc2kz5AmkjPllfXPcfrVXa20eXypdS4xzkcAe1IrwLAk
+ bI0bFiWZmyWPRe1NxTVkVNtrlXp8i NRJyQuF3Hn2xSRSMzxzRoU8sFf3gyBnt+IqeMSrMVbyy
+ 2MFduc+oqNTtZ9kZIwCPbirua+xvaD26 33/DUgkj/doXGWJ4XnKj3OMGo5plggjii2Om5juxk
+ g45yT61JKJiyF2Csqc7uj/SqSzGO3XzmR5n GFwuQo7g8da0jG/meRjVGLTaej0ul+mxDGxZAw
+ lZmxgBzkVOGK3EQgj3s5wvGevHf+dPdovNVAUI ckBkGAB+VJakf2Y0pYk+aVUKcEHHB+laOWl
+ 7HC66cbLd6eRatYpRp8scUgOZAxQ/eycjOcVNazT2 rIYkDOYmBQjJGDmlMTSXcUbSJFN5Z3A8
+ Hf3Bx+dILJ1+zmaYW3y9G549eK5nKLvzdQbhGnyyV0+i X/A/I1ILjzTLcTKrIWAYxAL14IHpx
+ WzBJZpvlYmBCPLj8x8kDPb+dYETm2W4gbbIHlRmAGM7c9Pw NWJYIbm8kSOXbCrEQF24w3IU+p
+ J6VyVaabtsv+GPXnQkr2bXez0+W5p2bW6XOwlLto42ZnA+++eM Z9KAlukiXE0MsaPJlmL4Ib3
+ Ppk9KswC4trYJdRRvMSVMcahTyPm/SsC5Xy4rOC0aT7ORukV23HPr ntWEFzydmc0IyqVPcWnV
+ 7misksOoTeUgnXG0pt4PBGR7CrMAW0tw8Ql2qgHmM3yjJwePxrKQ3HmK U3PtjKlh2B9at20pW
+ 5jgumDBYjgg4AO8YyKupB2PTrU3FNvWy6XEV7Ga8InlGxGEcZ3fdPofpWva TW1vqRcafduGcl
+ sTHOegHTHfPWsW5soxIQq8OS8be6nGD6k5xU0aXBE8TwziFpSR83KFuxPrnmoq RjOO+nqRKdO
+ vB22atu1b5Jq5rXXnXMYtLWKSIxoEBdsn73JqhdDZdLvtZ/lUqGjbCkE8jH0q3bKm 5lluf3kY
+ xO4JHPU/oMVnm5sprsywx3LEyAwoZM4Q9c+prKmmnZLRfqRGtaXLFO0dev53JrtIbXUH uWzEn
+ zKsQ67QAOPcE1nwb2gzH59xErYlbcSpxzx6etblywnaSEoGQSdSv8IAU/nmsOS3NjqTFJ0j SP
+ dsU5O7AxyPfNa0Jc0bPcdCq6vMqkdX6/oXRL9rhPmwNE73CBWONoQgk598d6f5dvHDMiLkM6FW
+ Q4ynQn/GpFtobiwsgEk2om2UB+euAfwqEC0jiUBzcGNirBGxuGc5HtUJrZfcKjO14rm9P+CW0i
+ lR 2lhjx5ZYx7+Qy9Rx3qFJhL9nlnTbGUIRIxtyvPX6njNWZftFxGWhAj85twJ6Y4GB/OqccUa
+ MkM6s 3kB4wynAweVH1J6VMHda7/idtCrzK6WvVrf+vQsLd7olHWWNlTYw5Q98+uKmhv7g3O57
+ dJCilGMK hSxY4DfSqdpFZtYrDPDO1zHkqivg4/iB9T3pF1GzhmzHbTfNtYHfnPBwfpmiVNO6U
+ blKjFxlFw97 5foy5vnXU4pLu5jWO3BikOMByOf14FaFwM6fMyvHctJIDIsa42nI4/Cqdoz+Ra
+ BHiMphdWEi7hIx P3vp2qFp7uCFXlaFsuR5YXDMw4zwP8msHBykrdDkadWrGUZJNaWv2+X6ot2
+ 9y0Sz+ZJCgZiVm2fK hHBGPftVZJ4zEhS9t43iiwqkclh16etaXlwvpcESERkt5ZRvmJ9W/A9a
+ r3FrDKlnbugjIt3ZHAxu z/8AqqYyg3qrEwq0XLWNnfW39aleKR5YVS4kTyooWjMjD5SG5FXIV
+ 228KI6XMsSSBVTqExyG9W75 rJspoIbTyEXzBNIr7nORH/sn3yKtoGlkRmguJERShhiba6An7p
+ Pc55z6VdWGr6FSjTlqk1b7v0/M mNgzTwLMk8kfl5DRvtJIHX8jVUPdRNFOkscoCZ8sqSSAe38
+ 6tSp5ltbhfNilVmWPc2QU/iOPamS3 9tLAn76FAE2sAOu48D8uaUHJ9LjoYhyTly3js1YZNPLO
+ 6mSdENwrSJ5eVGc8KPTmo1vdRt71pDdQ CcJh9yZ+o+tUrgRTw4+0RxRW9wYY5MHBU4yfwP8AO
+ p3uJIblg6JcIWKzKBzvzwM84xxWyguW1jvV b2kOTlV+3/BenyC4W+l01ZHZZjHhDtXGTkdPpx
+ Vq7iuRNJB5sUku5yVC8lhjn6YqjLI8sk4mWS3l mYsMnCrjhhj1PalTzknt4re5S7ZlJKoPnz6
+ Z/SnZ6PTQ5HOEUm2rp9tP1/QXyb14n3KZw6bj5ePk wRwPTNLJLNfRKmIond+Qy5Kn0/SmXwRY
+ YpA8tuGGzBcnOCBnjtmr6tNGm1XhnMZcSSKvDt0Dj2Bq XLRS/r/IKtfnUZJ3d9LKyX6FFIZbM
+ QTW8yzEKQ6qp78Aj070+zigjmNvI6WrjMjPKudxA4/ADr61 VMs39nzKWDneoZ04CEdAfrVyJ4
+ rrUZXLJmUNhT1U4wq/UmtJp8rv/X9andKlTdHmqX9V+qelvkSR W1w8UDPIkuFKh0G0EN1NVGJ
+ s7qMzmVwIyGAbB3D7uf51eWae20hYZ03yJgTBePLb+6fc9qz3kjjl mnJ3MXEZicZKkjv9Kinz
+ Nu+xGEhupaJ7ef8AkOt9QuttvKWRyLdspt5GM81MLuZ5EV7oWzOglZ3B OCO341BPOsqiFlj/A
+ HZCxmMbdy9z9KijuHilD26R+azEssqhtoUe/bFa+zTV+Wz/AK/rY7XhIqD5 o+8/T/J/kTW6vd
+ TuYY3ljcrs55OM4GfXjmr95E1rpAleJoiD5QLY+UNyc/54qW0gkMMN0sTOHjY+ XEduMHIHHc1
+ SSyYMs8t6hjLYaNiTtcg8H3rBzTlvojgqV5Tqp8ytHbTd9rX/AEH2lqlxdpmL5IlK gD+Pb61c
+ kuGO5VgmV1G8KrYEY7KQPp+tYSQxwPDbrclopF3Pz0bPI/KiW6eS5ke0LQws4Zg53FwO nOK1d
+ Fzle56sKEaj953fbZGz9pkkMb4E7SOJXSPjac/Mv6UlzHJcT3txLFcRA3QdDu7HsKpQtHai Ga
+ VvNSWL7iNhlYnp+VWbNkt7+RmZYwkZCh+fYqR3PpWbjy3aRpSp06bk1Hb1s/Ly9bGhLLA42I4J
+ b52QHoAvzCqLuLTWIJHt7hICCVSST72FwD9OaZDDBEY0luUmWJGwqEhgCOpPfrTNzHy4ZLiNFR
+ N6 NLzjHQfjUwilpuiKDTTunbqtbfkatrZ2ly8Uqv5RijVCrnPX1qpLMpYj7M8UeTG0mRgE8gf
+ kKUS3 S/Pgojq6ncBgHHfiqj3Ey6eiSwNlduG7Ke2fXipjTlzXvf5lfV3Gq5wmn6Pb5lq1CXF6
+ sUJJSM7E GPmZD1b3x61O1hb2trdJ507SM2YoxIQ2P61SVdU+1w3Vqse2MAqoQfMucYqvPd7J3
+ AtbkzNKMsX4 wB2/CnyScvdehx1K8o1G9LeqevntY1keFCZkZlE0TYWQ5A9/w6fjUsN7I1laRy
+ xGacEBFiOCRnOf pWS75RZVkQRqwVYerIOmCf1NXwYotRgu4xJvSBxvY/KTjpis5wVtTLE14VK
+ d56vp/l3JHjmhinIu ol8yXfFleSAeo9smnwi6k1GC5cJORHk4XkEHBH1zVY3TJEkkhjKYUnK/
+ c7bfyNSJHEjZtJX3JIFj BYncvr+Z5qWny6kVJJUfZyitfx8r6Fa7tEmSbbeQh/NDFATg8csB2
+ Aq5bXAnACouxyG3EYJwDzn0 4xUV4tusM7SxPvEmAVbHJGB+XekYQIbSRC63C24V0DYALf8A1q
+ pvmgkzSTjPDK9/LZ2+4jur9pYE jYRR74lZ9i7djZ9u3FTRzq8RBLiSaeN/KzyM8Efj1qN4He6
+ SzSEQwratl3AJOOQc1EM3Ailms57c ufMjOcbuMcfl+tPlhy6f1/VicXOn7NKCs/l/wPzLGoMI
+ JdsbPCYm2F3bhm6hh7YyMVRgkto90pu1 dwpAbaeRyP606a1lkx5MUxEhOZZDlVOOn1zUUkOYR
+ HLFHmRhueNAoHHK/oPzq6ajypNnLT9iopc1 79rL/P8AMjfUYzpZU/vCWVgF4bA4YUfaPN1mRo
+ mSb905CKvB45I9v8KintFtZLfyNmdrE5GdpHXN Z6SJs8x5o0RbgINq7Sc9G/3euRXTCnBxvEl
+ zpQi7LR/12JLlGNvDLHbTJtXIdiCNq9fxz3qnE10J 3xGJHbLsqrg7T1/CtW5Vbi5ubVJS4ViU
+ xnHHUfjUVpJL9rnhiWOSc/cIXOUI+Y/pWsZ2hqjOo+Wh LnjqvNozXSQpG6FdrKyxbUzhW7fnm
+ s+b7VLd21uiBQgIKgY4IFXkmminhZWRYzIpZCO4yFHtkVGk 8kOqztAP424YbiPb8K6I3XQ4ZO
+ U4WsrrZ3aIGa9ZI4js+QMANg5XvT40aOJ1fazSLvUqOUHUj+VQ IJd0YZZQjKd24c568ZoSHfc
+ pHuaMlTww5+7nP0rVpJG7nSoWqRTsuutvw/4BGsgZvMV4wcgDCHkH r+VPZHM9zuKkiRixxwee
+ MVHYSxQCIXhEkIJ4ThjmrRULEfvbCzN5Tk7h7k4/rTk7SIjKdSak07/h +H6jE8poN+5E3DIHU
+ tg8/wBabsRZWMYRlR8x7gCxH5U7f5IUrbqMgFtx4wRg444zUkzvHBtjG0kd W6daFe520IuqpS
+ qr4fIjaRPJJmkBkZ+Tjj2PSoQWIcqrJ1JLdtvUfWophlH+YFchskdgOlSgyZMg TaNp2hvSr5b
+ IyvUcrJ6emtiBimAZCyqzAwlj0Hp05qDEBL4VyEkCq4bjPXninmaT7OZBGTHH8qgq DjcMZqJp
+ CD5UZAIbawx0I/ritIpnjJpycHa3q7siSU8eWyM7MzN8vT2/KrUbKY5W2rjcMEDAB7A8 deKZ5
+ EaXivEhkKksFAzhff1471IDEFdA4GZP3RP90evvTk10HScqUuSS0XzLlulxPdvC20XM2WUn 26
+ qfc02YFYUSRJPLhYEZblVI5Q++e9BkaJmkJEkqsANvHGOefrQtxgzI+JYM9ckEH0JxXPaTd0ZT
+ 9tKstPdW19Hf10LFjD5ksQeVAskLMQfvDB4H41fEq/Y2WNUDyhWZSvQ5xj8qyYXH2BlQ7FTDbm
+ HI wfu5981rWLvLNLdq8cZDbIo3X7wPAP4VlWVrtnoxcaN5Tlr07fl+hZ+0XAmfdJtkgm2ByMj
+ kY596 seVk2/lzwl4YShyOozy34Zq22be2WScpIFWTcoXBLDHP4UttfJBcpF5AmEeVmwBkk/MT
+ 9K4HNtXi v6/r8zOGI9xzhC7/AD+9FOZb6LbJDFiOPKs4UYbpz9KozpHJO7SB/M4wynA5GQfoK
+ 6Rrpv3xiwhe RXIcZxuGMVgO8waWO2QOFKljt3Zxxjn0NXQnJ9LHThazt71NX73t9+hMXmSwV4
+ kM4bMfmDkMW7j9 alt4Wi1h0ll+VIGBPZiBwR+dRbfKiV7iCeOZnC7AcDr8xA9hVi2aSa2lC3d
+ uSg2jenOc8fmKUn7r sZ/WWk409U93tb87lNRZSxbYJJATGvzluE55Dep60t0kEVyTEd8cO4Yj
+ GM+n4VO+nziZt09u8ImH miNcEEj6U+VbecRS20se9FZGGPfHPqaaqK6s7o2eJg3abbflt+KuJ
+ IQ8skTiSFs7lUn72FBJ+lTS SxbYDbRG4nAK7D83B5BoX7RMJ4whRTOuHbnjGDj6ikSUR36LAh
+ SRY2SEkAgrjv6nrzWf6GanC9ub Va2vp8xlwUbT32ziGaCREVdxGc8mo1kPnuFh82CdS8ip94M
+ PQ9gB2ptxfSDS4wEgKyc+Zs64P8zT JJJxdTXLELuZtqquMgrzitIxdtTpw8mqWi1d+v5bWLs9
+ z51nAI5o/KPzMw4xtAA/CmxfacmFpYJX 83dOQvOf4R+Ap1vaW5sEmiG7KYWHPLjozD2FSRKrS
+ MsjBJopEUEcZX1Pqcd6ycopNLockZxhzOCv b8GXI0txEFZXnm+ZlaM4wmcf/WqBo7Z7VLmC0d
+ JNu1Ucg7gGwCPahMPfPFBeRbZG8zO0/KEHT6Zp GihlngcXIPnJ5ihSQAB2/PmsldPf8zVJqSk
+ m9NWnf+rE0EdsHnRre5gnBBVHk5Ht+NMF4Y55DOFj m+0bgjjOz1U+9VUFzc2UsrXCgPKjEY+Z
+ +xYH0FWpPLSzlFrieXzVMRPzZTnnnqc55puKvZ6m8/Z8 /Lvf+tW/8i1GkZCJc3kW4xMV2ccDO
+ aoSzwx2qGPzWSSRSjl87VHT8O59akt7x2nhhVUaRoWZztB2 kHJFNmu5PLeIpDEXdQW2DGWHAH
+ pxUxhJSsznp0pqoo307X0+5LUnnK26NALX7QzuGRo+BtXqf1ow jXUrXUxEJYCURnaVI+6f8aj
+ Zrh7Zjb3UDMXVI1wcopOOfrVq4LyrID5bYYh0UYPy+/rxU7af10Om M40/3fPZvte/4ozw0siO
+ jTpwjqrY7sf/AK1Q28V3AJoZVilRdxEoQY5H+FWbx7h9OVtibXj3gquN xzkmoJA4t1ntpQJZV
+ Y4c5DKOcj+Vbx1Vu5tKMXTSvu9P+HWxBay+YwkaMRwJgMWGQSTwarz3dxdt cllEXnEyAlcDch
+ 5Gfwq2k15HYTqkK+UzD92U+YMcbapXAAntY5VZbkbyX/gJB4GPetoJc17f1ucv PGFRt2u9O9v
+ PyHjy5kmuZLjeguMB1PChjnB96kuNh1KeNJo4NzqzScjp1PHpVe3uYhFaxXCbGVGD rjG1s8Fv
+ XFPjhVLwOG5VTGHcZV+OcfXtTas3cyp1/Zxm3o+nb8V/n6FQCW5yrrLJIT5kZB4BPGPx xWlFG
+ 0LzTy+Y4hYoyI2CSRwfp/hVS4WOYW7W8oUrErhFz8oB5BPcjrW9DcIJrsBont5JfnPUk4+U j2
+ oqzajojv8AbzjD4bJ9Nn6/8MZsNxGPPCFVdpMDfyu3HLY/l6VZuTDCkc0KFlkBjCpwUX39ST3p
+ rWbw24JELs7qVUDDZHTHtSSXVzZ3LLMYjMuRjZ0J71npKXu6m1Kftbezlfvr/wAC5Zmth9kRYr
+ W7 G9SDIzg57/n71NbJCssn2x4ordjuIcZbJ9/bpVBtSvTZrDlX2Abzt5B9KrhkmTz7gtuK4VV
+ ONw/v fQGo9lNxtI6KeGrKHLUdr7W1a+8sSafbw2SyTbmMM6wyBW5GfX86fcfYY5CEjeONg4iJ
+ Iy3Ytn0q vDdG1htw7xzZlBkVhnzDnjH4VuXFsizSGSB50kfdGo6gL/B9T7UTnKMlzO5zqcqdR
+ Nu/bVK//BMm zliS4W1SWUr5ZUNu4Bxk/rVoC4S0QXOxbcRbXBAyr5xz75rOjlRL9XuIWjRjvd
+ BwQRkgCn3EcbwB mndR5oyrEkoCMkH1NVOPvevzHiWlU93Vel7Py8wtbdUFz58hZYpRFlcZIOf
+ UUkkdvHCTAXZygPzD 7gHGDxgmq+5vs8gglVYUkAAfktuOd34VZiLpLcmUpLtkEcgAwCTypHt6
+ 1o73u2dNCvLl/eyd+i6D 2hWR7cJG8sUaESOp4DAcD8aginBJCNvGBtJ5wR2PqcmrMoJ0xoow2
+ 4SBtwP3yvUj2qcTndJKIInj c71WNQCCetZqTsFLFOMHKyfkMtrlJbxJQI0kZBu3pkcZBGP1qC
+ UGW6MkTJJ5ThFwvBB6VE1s8iZS WPaMqrD+LnJx9elFtcXK292GKRsJApiZMNycjn2xVKK+KI3
+ U5ptqSldf1/WhZupHjkWO5jlJ2uAU faOvWqyzQm8iWMmE7SMzHcAfU0rwsq+ZcszFiERQfmI7
+ nmq8MipOgYRQptdCzqCfQ/4VUYx5dA9t S9lLkl739bLUs3d4HQNCzo8cvloFb+Hj5TjqT2qnN
+ qUt5JvSP7PtmbzVIySemB6cUyW4iW2RLWJl VGZn3/MeeMH3qpbovmK0Nu8jtIcI2DtIAG4+x6
+ YrSnSio3aPLcYqMZ1F72t15dNVp+BbVY3uJt4c AMzRgHBwBxn61qaWIIZoGlYxpNgh5D8vQhl
+ +tUTazknzjHGU3KQBjdnqfoKlEMMcsMy3EchjP7sD 7uQOWx6VFRqUbXNqtanOm4pu78m/xSt+
+ Ben05AytG5khwvybuWYf0piCZr0JPKLYBHDSlTtPcY9M 1FKzXjWKLeRbowwZFBBfHORWnbqiP
+ Gcbo0t2CmT5gcc/pmudylGPvasqhiWo+/rLbs/x/wAiulsr RWaq0su+IsfmyWbOQw9hViB/sP
+ 2lpraSSScecrMegHUdPeqv9omSCFoY1V0iESDaOecnH86fa3CG 4Xyo3UlcoZm3fKQRj+ZpSjO
+ z5lodjUqcJOS5k+l3+n+Qksm6II19ChKjZnOUXrg+pNSz3CvbxTWm ZRGwQAH/AFeTnaffrQ0E
+ V7MYFAHzqGcD74A5YewFR+baw2Qmt7qKOSVpNgcEjZjGcAdfQ1GjtZan l1a8bxhFu99rfnZBc
+ yywSARN5cBT5CxLbxn7w+mefWoLhg9lFcx7p42cNJsbgFSAP8arRQW02k27 W6TzSqG3IXzuPt
+ 7e1Pgtr1Lq3uYgnnsm94yo2jJ6EfrWyjFddV+P9dyKtWnUg5J2a6d/XqvX8B66 hIb65ad4o5f
+ m+Vkz1HPHpVK7tWXT7YJAzl4CJGH3Q+eB9QP51Tv2mkuoCI9kUhLKpGTtBwcn9ald PIEodZ1e
+ ab92+/5W9xW8aajZowlJRcWn8v6t+QyK3n/dxtuAAJMnTDHoCfWpZLtkWJbiCQXCxtEW Q7T8x
+ 6cDrUM81zb3EyRk7FLJhgG9OaheS8jeK4mt5eSGiJHUDgn34rXlctXYdec+TWzXTVrX1IGk t2
+ l8pt8YIGd3O0r26UlzNEI0eMofNyYyvYDgg+p96givWnvBPsiL+ZtUEZyM49KhmSRrolYVQlm4
+ K8j2+tdMafvK4vbTnOLT9O/53JbadI2YzRO2MjBPHI61WIWS7iKySIUYEKxJL+1SMItjeYjh8E
+ gA 9B+X41FEF+yEjzJI0bBOCeT0FapJamFSmuf33o+70+64scFqJV3sEycqSTyT/wDXqzJIMRy
+ BhNgF ZAqkbccAn15qgJURvmid95x2+XtThKy3SxmEkFtp9vr9abi27l+1jF8ylZLol/wNSx9o
+ R4U+YM4H 7ojoQDzxSODNbyyrIJVSYBSp6++MdKpJMHVTtRWVxjA7/wCeo70Ga5jKhI0Ul8MNv
+ Ue1P2epz+2W 769lf8Cz5jQ3ckWCQcgFhkAdc9O9VJ3kYxeTIWKZEiMD0GORx0qWNXnlIMckbg
+ c89ewxSJbTxlpk VtqPhi65ww4IP9RVKyfmcuKrVKkLRk/S6X6XKqRiWXJ8x9nKANgfjVeOJt8
+ AiKA5PmEnnPXP0rR8 uaNPljYnG0ADnrx/OqkUN3MQ/lnYpJPyHnHJraM9HqcuJpRpOLlfn7aK
+ /lrsTNdll2Sja5wq8djy aHx533ldEYBiF4NKsLKqlQZAVynHb1zUaoWARJUK7gZBg5PoOnXFT
+ 7vQ6pzqO0pu9/T8+v3lnzld ZXkRn3NncBjdxz26U23knW5R7Et8zhcMoOARwTkY4phgAI2jIV
+ Sck579KSR1gjtWjcywSStgISCu MEZNSkun/AOHGQtT5Zu172X6dvxL0QdY4nRSltO3VjnIB5P
+ 4VuwK0SWXlobqaRJCuzptJ5H1FVLe 2jlWKLyJYEVsFXblic4I9B6+tKjzRX0UbMI5S6DJ/hJb
+ t6VwVHz6HoRm6ivdNx1fb8DqZLiBNOgD W0/3N4QtlmHGT+dULgSXeryKlrKoaQ72BHHy5q3PM
+ zJevHNBJKLkALs524AOPQVcijiBZT5gjtmM Bkz97eMj8e2a8yM+RXtqXRrOhBzjo/V6fIzYre
+ SKK5K7pJFkjJIPDDHb8OazxJENWQRRysCDhlbg A9z9a0TA1pMsltchZRbner8gkDg/lxWXEqy
+ WisWDfKuHTgR84w3qc10wd7s9Jy505R2a31/Ine2m jZJJGaNGO9WkYtuC8n9P51oWSWc10YJL
+ iNJpnypwQF44U+p7UyZTBpgtpnZ4TJ8zn+HAwcE9iaqN uDqUSFJX8xsheVwOV+vcfWod5x3OF
+ /7XFq+uy5bW+7qW7q6uLOS8gVRGJZ0cOy5AwASKZb6legSG OO2ZPOY5EQ+ZWFSJNNLbRXEtlJ
+ GVjxC7kEbMfNkdz71Wmmne1txB5Z3orLtXHy1MYpqzirkKFOpZ cqvt9332LSXsMcNrHcSKksk
+ uQBwemPyPaqKTKXYSI4ztHXBQ54H88+tSW9xLc3hM0cYjkYSKSgyF wQSD6DirSzQTyC3t7ORQ
+ JFYlmBbAGQM/WnyqDen4k0aCpSlGMb97NafkQRXNm1uiuVOHKlffPDfQ UyRAwuAZFdoblGLDO
+ 0tj5uP7vtWtbwQ3FkbmMRCSZ94wByc4wB7/AM6jXSjPE0LZjdxvI7oQDtDe tZqtBN9CcPXpKX
+ vJr1s/wsUoo2WcSeTN5fzKpRuF55qHybmeRBb286OU2sHbluoz+FW7eO5srP7Q xOZlMgVhnfx
+ gsPYGmv5loPtNzdJu8oqO25uMEe1aKb5nb9T06GLvUlyyS00639UWLXy/IMbyxyJb qIkZOCVc
+ YOT35omDw3KQrJE2bZjkL1KnHHoPak8qO3jj+1uqxKWWZV4LNwQPw4q3FZLc6xKFSWRF CsoDY
+ OQMkfjWMpJNyvp/X/DlyqR5ZtbW62t8iC2gWztZvLlXDSDG7kIM/dPuc1IkD2m37UrRKuUX HH
+ llzwp9TzmrLyTSI8kVsLcv87JKAeW/wAqrdPJLFbMpKxiMhQ/PQg5J7moUpSevU56L9rZuS+/X
+ 8Cf7Ozz7YLWTzI28oyqfl+7yD7mqkRuo4438yCRVfy1Qx8jP3c+46Vqgzi3muTIj+ZcRCEJkD5
+ vv GqMscxcwW0ihg7mUMOcr0b6YqYSvo/67jptynaMk4+jf37laeLyQyI4WdnVdwOAo6c/rUA0
+ y4i85 WaQbztVi2dw6k/iKvQE3ulNmCSVVKMrqcE8/MaiS7mGozrOTNDIjSEIMEkdMegx2rZTm
+ rpbo7ozq Rva116O/oUPs4SwJa7+Zf9WpY/dbGPy71bK7J4Ty6hWLMvTrzgemOahnWcCye3tnf
+ NswDkZUqOS1 U542ihMjrO6OC0bK3BXHJ+nNbL37XZMqilDl5rX+/wDr5sjks5pXZopPtKMzOr
+ gnDYHXH8qdHI9x BE9xDIwYKY0HDFQCvX6jNakcEuz5EYuAxZU4w6DqPbB6VizvNPALiRkK5+V
+ QMfe44HpWkJ8+hzrF KpL2bkkl1t+hLdrE0KTCQSo8m3CDB3jgjNV0g3Swkz/ZoIn+TzTn3I46
+ 1O1iPs0UUbG4Kk/u4j1Y d+fqaLi0uHtT5YMaxseDzn2+vWqjNWtcwniIygqTavfe35X/AMiuL
+ hXuAtsY2LuVBA/vf4U2C6lI j8sgAsIxIB8qgDGCO5680ItrHC8QBtwASsjHOM4Kr9evNWrGyj
+ lbcZTtD8p6MDwfpVycFF3NfaNQ 5ai28v1X5DgJFjztnkYErFznCdDn396tpLJBA8UMLvMW3gz
+ DcQoHQ5qK7uHjupzg5QADacAZxkfU 1YH2x3zEgUSKWBZQSRngg+lYPVJtHp048y/eR9P+CTW0
+ EM16I5CVjZMh84z6H86sTWtpbtFIJBvc Mixn19RU9pmCzcyJ5qtKpDKMDnpj2zU9pbia5ka4j
+ eSdZWjQrwCG7j8a451WpN30RksdPn3tFeZk zRpMyhgrMgCqFGCSe31NXZL6S28mMK0jKmGbGf
+ nJwAPxrRm02Y7txjd02tIVXGHXv9AKqm2hudWM UW5pnTzPMB+VgP4gPQVKqwktdUjrlWpVI81
+ rpev47GHJHLFdRRGJjPuQOvXJB5xVk2c02p3ASFt7 KxQnG1VHAB/2vepp7aC0lWaSVrgbGUFW
+ Odx561RS4aTTokjMkchAA+bnqSB9Sa6VJyV4jdWd1Ui/ d762+epbRUs57VrieDdsTdEEz8vfP
+ GCcmnTQi41ZxERIFDLtTghgPlJqtOjTSkwxMZD80pc5wcdB 6Ac09LmS0YbZo92xhG2zIKEcE+
+ p9DU8r3T1MKlNuo3Tl7z7pfp/w5UY3K6mGwnnN8gAHD5HJA+lM jliil2eVKu1AG3Nna3UH6Yq
+ x51vdTxp5UodTuUbuSMZx+GKkmhs3tMxuRcdSO20jn8a15ukkOU5y k1Ui1fTRbeen+RDc3iJZ
+ GKGJ+CAHJ4Y5BBH61XV/tF+ZpnEbSTq5XpuIHb25rQZbmeyt7cW+1oNm G2g7jnr9AKkkt2zcy
+ xR+WwugVLDIIyDkD0qVOMVbqcjrRgnSirX6tpsx21CeW5DSf6yJiqggYB71 dSe2/dyyyxqyw7
+ XQpkk5zmqxgAvJzc8tvdnRMA885+grNG+S5hXckbBfmLDIJA4rf2cJLTQ6bUql J+7ZrdpL+vw
+ L8TrPG0Drl2OVA4JA5Jo2xRi2nZ8jawk2HGGbpQbq9itIJnktg5AaMiP7w6Ht26U0 XX7udftF
+ uIjKMKU5Y9iOOgotLpt/Xkct51ZabLzf+RUhkZ7MEB5WyUGD09Sa1I7MJZlrtm2jIV0O AVHp9
+ ayVuGxIpZWywyUUAY7/AJ1YUf6JiIuxchvLJyVA9a0qRl3sdjhUkoqU7Ly/z/zNK3hdM3lv NC
+ ZMHam3JB9PxFTW5mgjt2ifM7DzdrcjaB6fic0yIwQwyzSMZY1cIAhwSSOopJopblLVkhkjkfIV
+ G/hHQ5/KuSTu7PY5/rMNFVel7cz6+T7/AHISRJG1aNrcLOUh/gTAI9f6VZtPPMcwA8mMPnEi5K
+ sT 8o+lRx2csEqtuI2KY5FHUknIqUlRqIkWZdskbloyeUkI+UGlOSasjqrYihO/IunUn8yS2Fs
+ yAp5p ZjuGdozyv5c1WLESsIreGWJUzCSgO6MdxUMBEdujt5kxdWaQ7uBnsPTPQVnyTQTsdknl
+ RHIRSclT 12mnCldnDh6bi25W16/5dTQjnjS4FzBhY5QPLtx944OAfp61HPLPa6vKZWZlBOEU4
+ yFB5HsDUEf7 /TlkEDkruNuV4BXPzH8Kvubf7T81ldSQ5Oz58kjGetNpKW1ylzNuqo8yenToZk
+ UsTWcc9xA7KVGw bsZQkhsH61dk8w3RNqgM2/bGW+ZSmPmIz0xUxgmeximghA3qC6FchAT0Hpn
+ FUp3jmjurgJJaywS7 FRm6Z6jiqupPQJSjNJX1/BeqILgo1iXk/eStKu0A+oxn8ayt8q3MeJAk
+ gUlQy5zjtirZgjjVNkoX zAGG9uFAOMdPWmG1zPJErRuUb75PTHUE/jXVT5YrUUYqnCUZNL5Mz
+ 1MIYdWjcMfLXqp7Akj1pzl4 ZA/kOwlLID2wBy35mmvCFViZELpLtG3ocDn8jTVd0aFVheT5jG
+ zZ+VN/c/rXRvqjhqVJpKcZOy/L yJUQ/Yo8FW3qSCRnft7j2qq8saxwqoUOWGWLZBHcYx1z3q9
+ NZ7JlC/NalSY3B68/41SCPLKCBGSG wTjH3qcHF63NbOajbWz0av8A0x6/O0qugc4ypjGMD3+l
+ DxvGIjMhQtH8rFcbu+fyqZVtzZxjLJNu IJJ9OQPxp6Ry3d38zGQHhlA55HUdhjApc3XoTJTfv
+ 2vr13/P8Rm+CaUptjcMDtIOADQLFwltIYpj bv8AOzL/ABKBjC5HXNacOnI2jwSlUS4MZCgLgv
+ zjP51cnsljs7iGOQmXzAJEz3HXA7D2rnliYp2R nXxftmlJ2/B/hb9TFuFs57dGjhmh/egRl2z
+ lO7H1HHFOhsd90ZXSVLZZNwJbjHXn1rQa5RHiLGJ0 KFgu3O0d1+tI8dv9reQXqfO5KoScBSKX
+ tJJW/wCCebWhKEeVrfrvf08zDW5EN48kCrKgOYxt9uvT pVJGuJGUpbyM5UgFF+9z1+ldBLZkP
+ Ir4uZGbgQqFAwuRj+tZkQkWBJchZVRQcd92cY/KumE4tNoq MqfMuV2b7oozrKh8uNjt6MMdPp
+ 6VDGlyp/1eSTlsIBgVceS6nkWOIM+xdhxHyO/Jx1zTSLhYla4i kjITauR2Oc5Hc1upWVnYqUq
+ LqK7s+lnp8hjpiIGbzfMB4Yfdx9MVnRlkTaxWQGTPyjG0Y5H16VMV
+ ZpLZVl3hEJOfXng+tTh59q/6tgyA/d6ZrRaIz5FUlfVee5//2Q==
+X-ABShowAs:COMPANY
+UID:F0A6918D-8E09-43FA-9684-226810B8A96F
+END:VCARD
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/FCBA0FA3-00B2-4C95-B4EC-4CCC4843F8B1.vcf
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/FCBA0FA3-00B2-4C95-B4EC-4CCC4843F8B1.vcf 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/FCBA0FA3-00B2-4C95-B4EC-4CCC4843F8B1.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,12 +0,0 @@
-BEGIN:VCARD
-VERSION:3.0
-N:Kawado;Saeko;;;
-FN:Snow Leopard
-ORG:Snow Leopard;
-EMAIL;type=INTERNET;type=WORK;type=pref:snowleopard at example.com
-TEL;type=WORK;type=pref:777-777-7777
-item1.ADR;type=WORK;type=pref:;;1 Fidel Ave. Suite 100;Mountain Top;CA;99999;USA
-item1.X-ABADR:us
-X-ABShowAs:COMPANY
-UID:FCBA0FA3-00B2-4C95-B4EC-4CCC4843F8B1
-END:VCARD
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/FCBA0FA3-00B2-4C95-B4EC-4CCC4843F8B1.vcf (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/FCBA0FA3-00B2-4C95-B4EC-4CCC4843F8B1.vcf)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/FCBA0FA3-00B2-4C95-B4EC-4CCC4843F8B1.vcf (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/addressbook_store/ho/me/home1/addressbook_2/FCBA0FA3-00B2-4C95-B4EC-4CCC4843F8B1.vcf 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,12 @@
+BEGIN:VCARD
+VERSION:3.0
+N:Kawado;Saeko;;;
+FN:Snow Leopard
+ORG:Snow Leopard;
+EMAIL;type=INTERNET;type=WORK;type=pref:snowleopard at example.com
+TEL;type=WORK;type=pref:777-777-7777
+item1.ADR;type=WORK;type=pref:;;1 Fidel Ave. Suite 100;Mountain Top;CA;99999;USA
+item1.X-ABADR:us
+X-ABShowAs:COMPANY
+UID:FCBA0FA3-00B2-4C95-B4EC-4CCC4843F8B1
+END:VCARD
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/common.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/common.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/common.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,897 +0,0 @@
-# -*- test-case-name: txdav.carddav.datastore,txdav.carddav.datastore.test.test_sql.AddressBookSQLStorageTests -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-"""
-Tests for common addressbook store API functions.
-"""
-
-from zope.interface.verify import verifyObject
-from zope.interface.exceptions import (
- BrokenMethodImplementation, DoesNotImplement
-)
-
-from txdav.idav import IPropertyStore, IDataStore
-from txdav.base.propertystore.base import PropertyName
-
-from txdav.common.icommondatastore import (
- HomeChildNameAlreadyExistsError, ICommonTransaction
-)
-from txdav.common.icommondatastore import InvalidObjectResourceError
-from txdav.common.icommondatastore import NoSuchHomeChildError
-from txdav.common.icommondatastore import NoSuchObjectResourceError
-from txdav.common.icommondatastore import ObjectResourceNameAlreadyExistsError
-
-from txdav.carddav.iaddressbookstore import (
- IAddressBookObject, IAddressBookHome,
- IAddressBook, IAddressBookTransaction
-)
-from twistedcaldav.vcard import Component as VComponent
-from twistedcaldav.notify import Notifier
-
-from twext.python.filepath import CachingFilePath as FilePath
-from twext.web2.dav import davxml
-from twext.web2.dav.element.base import WebDAVUnknownElement
-
-
-storePath = FilePath(__file__).parent().child("addressbook_store")
-
-homeRoot = storePath.child("ho").child("me").child("home1")
-
-adbk1Root = homeRoot.child("addressbook_1")
-
-addressbook1_objectNames = [
- "1.vcf",
- "2.vcf",
- "3.vcf",
-]
-
-
-home1_addressbookNames = [
- "addressbook_1",
- "addressbook_2",
- "addressbook_empty",
-]
-
-
-vcard4_text = (
- """BEGIN:VCARD
-VERSION:3.0
-N:Thompson;Default;;;
-FN:Default Thompson
-EMAIL;type=INTERNET;type=WORK;type=pref:lthompson at example.com
-TEL;type=WORK;type=pref:1-555-555-5555
-TEL;type=CELL:1-444-444-4444
-item1.ADR;type=WORK;type=pref:;;1245 Test;Sesame Street;California;11111;USA
-item1.X-ABADR:us
-UID:uid4
-END:VCARD
-""".replace("\n", "\r\n")
-)
-
-
-
-vcard4notCardDAV_text = ( # Missing UID, N and FN
-"""BEGIN:VCARD
-VERSION:3.0
-EMAIL;type=INTERNET;type=WORK;type=pref:lthompson at example.com
-TEL;type=WORK;type=pref:1-555-555-5555
-TEL;type=CELL:1-444-444-4444
-item1.ADR;type=WORK;type=pref:;;1245 Test;Sesame Street;California;11111;USA
-item1.X-ABADR:us
-END:VCARD
-""".replace("\n", "\r\n")
-)
-
-
-
-vcard1modified_text = vcard4_text.replace(
- "\r\nUID:uid4\r\n",
- "\r\nUID:uid1\r\n"
-)
-
-
-def assertProvides(testCase, interface, provider):
- """
- Verify that C{provider} properly provides C{interface}
-
- @type interface: L{zope.interface.Interface}
- @type provider: C{provider}
- """
- try:
- verifyObject(interface, provider)
- except BrokenMethodImplementation, e:
- testCase.fail(e)
- except DoesNotImplement, e:
- testCase.fail("%r does not provide %s.%s" %
- (provider, interface.__module__, interface.getName()))
-
-
-
-class CommonTests(object):
- """
- Tests for common functionality of interfaces defined in
- L{txdav.carddav.iaddressbookstore}.
- """
-
- requirements = {
- "home1": {
- "addressbook_1": {
- "1.vcf": adbk1Root.child("1.vcf").getContent(),
- "2.vcf": adbk1Root.child("2.vcf").getContent(),
- "3.vcf": adbk1Root.child("3.vcf").getContent()
- },
- "addressbook_2": {},
- "addressbook_empty": {},
- "not_a_addressbook": None
- },
- "not_a_home": None
- }
-
- def storeUnderTest(self):
- """
- Subclasses must override this to return an L{IAddressBookStore}
- provider which adheres to the structure detailed by
- L{CommonTests.requirements}. This attribute is a dict of dict of dicts;
- the outermost layer representing UIDs mapping to addressbook homes,
- then addressbook names mapping to addressbook collections, and finally
- addressbook object names mapping to addressbook object text.
- """
- raise NotImplementedError()
-
-
- lastTransaction = None
- savedStore = None
-
- def transactionUnderTest(self):
- """
- Create a transaction from C{storeUnderTest} and save it as
- C[lastTransaction}. Also makes sure to use the same store, saving the
- value from C{storeUnderTest}.
- """
- if self.lastTransaction is not None:
- return self.lastTransaction
- if self.savedStore is None:
- self.savedStore = self.storeUnderTest()
- txn = self.lastTransaction = self.savedStore.newTransaction(self.id())
- return txn
-
-
- def commit(self):
- """
- Commit the last transaction created from C{transactionUnderTest}, and
- clear it.
- """
- self.lastTransaction.commit()
- self.lastTransaction = None
-
-
- def abort(self):
- """
- Abort the last transaction created from C[transactionUnderTest}, and
- clear it.
- """
- self.lastTransaction.abort()
- self.lastTransaction = None
-
- def setUp(self):
- self.notifierFactory = StubNotifierFactory()
-
- def tearDown(self):
- if self.lastTransaction is not None:
- self.commit()
-
-
-
- def homeUnderTest(self):
- """
- Get the addressbook home detailed by C{requirements['home1']}.
- """
- return self.transactionUnderTest().addressbookHomeWithUID("home1")
-
-
- def addressbookUnderTest(self):
- """
- Get the addressbook detailed by C{requirements['home1']['addressbook_1']}.
- """
- return self.homeUnderTest().addressbookWithName("addressbook_1")
-
-
- def addressbookObjectUnderTest(self):
- """
- Get the addressbook detailed by
- C{requirements['home1']['addressbook_1']['1.vcf']}.
- """
- return self.addressbookUnderTest().addressbookObjectWithName("1.vcf")
-
-
- assertProvides = assertProvides
-
-
- def test_addressbookStoreProvides(self):
- """
- The addressbook store provides L{IAddressBookStore} and its required
- attributes.
- """
- addressbookStore = self.storeUnderTest()
- self.assertProvides(IDataStore, addressbookStore)
-
-
- def test_transactionProvides(self):
- """
- The transactions generated by the addressbook store provide
- L{IAddressBookStoreTransaction} and its required attributes.
- """
- txn = self.transactionUnderTest()
- self.assertProvides(ICommonTransaction, txn)
- self.assertProvides(IAddressBookTransaction, txn)
-
-
- def test_homeProvides(self):
- """
- The addressbook homes generated by the addressbook store provide
- L{IAddressBookHome} and its required attributes.
- """
- self.assertProvides(IAddressBookHome, self.homeUnderTest())
-
-
- def test_addressbookProvides(self):
- """
- The addressbooks generated by the addressbook store provide L{IAddressBook} and
- its required attributes.
- """
- self.assertProvides(IAddressBook, self.addressbookUnderTest())
-
-
- def test_addressbookObjectProvides(self):
- """
- The addressbook objects generated by the addressbook store provide
- L{IAddressBookObject} and its required attributes.
- """
- self.assertProvides(IAddressBookObject, self.addressbookObjectUnderTest())
-
- def test_notifierID(self):
- home = self.homeUnderTest()
- self.assertEquals(home.notifierID(), "home1")
- addressbook = home.addressbookWithName("addressbook_1")
- self.assertEquals(addressbook.notifierID(), "home1")
- self.assertEquals(addressbook.notifierID(label="collection"), "home1/addressbook_1")
-
-
- def test_addressbookHomeWithUID_exists(self):
- """
- Finding an existing addressbook home by UID results in an object that
- provides L{IAddressBookHome} and has a C{uid()} method that returns the
- same value that was passed in.
- """
- addressbookHome = (self.transactionUnderTest()
- .addressbookHomeWithUID("home1"))
- self.assertEquals(addressbookHome.uid(), "home1")
- self.assertProvides(IAddressBookHome, addressbookHome)
-
-
- def test_addressbookHomeWithUID_absent(self):
- """
- L{IAddressBookStoreTransaction.addressbookHomeWithUID} should return C{None}
- when asked for a non-existent addressbook home.
- """
- txn = self.transactionUnderTest()
- self.assertEquals(txn.addressbookHomeWithUID("xyzzy"), None)
-
-
- def test_addressbookWithName_exists(self):
- """
- L{IAddressBookHome.addressbookWithName} returns an L{IAddressBook} provider,
- whose name matches the one passed in.
- """
- home = self.homeUnderTest()
- for name in home1_addressbookNames:
- addressbook = home.addressbookWithName(name)
- if addressbook is None:
- self.fail("addressbook %r didn't exist" % (name,))
- self.assertProvides(IAddressBook, addressbook)
- self.assertEquals(addressbook.name(), name)
-
-
- def test_addressbookRename(self):
- """
- L{IAddressBook.rename} changes the name of the L{IAddressBook}.
- """
- home = self.homeUnderTest()
- addressbook = home.addressbookWithName("addressbook_1")
- addressbook.rename("some_other_name")
- def positiveAssertions():
- self.assertEquals(addressbook.name(), "some_other_name")
- self.assertEquals(addressbook, home.addressbookWithName("some_other_name"))
- self.assertEquals(None, home.addressbookWithName("addressbook_1"))
- positiveAssertions()
- self.commit()
- home = self.homeUnderTest()
- addressbook = home.addressbookWithName("some_other_name")
- positiveAssertions()
- # FIXME: revert
- # FIXME: test for multiple renames
- # FIXME: test for conflicting renames (a->b, c->a in the same txn)
-
-
- def test_addressbookWithName_absent(self):
- """
- L{IAddressBookHome.addressbookWithName} returns C{None} for addressbooks which
- do not exist.
- """
- self.assertEquals(self.homeUnderTest().addressbookWithName("xyzzy"),
- None)
-
-
- def test_createAddressBookWithName_absent(self):
- """
- L{IAddressBookHome.createAddressBookWithName} creates a new L{IAddressBook} that
- can be retrieved with L{IAddressBookHome.addressbookWithName}.
- """
- home = self.homeUnderTest()
- name = "new"
- self.assertIdentical(home.addressbookWithName(name), None)
- home.createAddressBookWithName(name)
- self.assertNotIdentical(home.addressbookWithName(name), None)
- def checkProperties():
- addressbookProperties = home.addressbookWithName(name).properties()
- self.assertEquals(
- addressbookProperties[
- PropertyName.fromString(davxml.ResourceType.sname())
- ],
- davxml.ResourceType.addressbook
- ) #@UndefinedVariable
- checkProperties()
- self.commit()
-
- # Make sure notification fired after commit
- self.assertEquals(self.notifierFactory.history, [("update", "home1")])
-
- # Make sure it's available in a new transaction; i.e. test the commit.
- home = self.homeUnderTest()
- self.assertNotIdentical(home.addressbookWithName(name), None)
-
- # FIXME: These two lines aren't in the calendar common tests:
- # home = self.addressbookStore.newTransaction().addressbookHomeWithUID(
- # "home1")
-
- # Sanity check: are the properties actually persisted?
- # FIXME: no independent testing of this right now
- checkProperties()
-
-
- def test_createAddressBookWithName_exists(self):
- """
- L{IAddressBookHome.createAddressBookWithName} raises
- L{AddressBookAlreadyExistsError} when the name conflicts with an already-
- existing address book.
- """
- for name in home1_addressbookNames:
- self.assertRaises(
- HomeChildNameAlreadyExistsError,
- self.homeUnderTest().createAddressBookWithName, name
- )
-
-
- def test_removeAddressBookWithName_exists(self):
- """
- L{IAddressBookHome.removeAddressBookWithName} removes a addressbook that already
- exists.
- """
- home = self.homeUnderTest()
- # FIXME: test transactions
- for name in home1_addressbookNames:
- self.assertNotIdentical(home.addressbookWithName(name), None)
- home.removeAddressBookWithName(name)
- self.assertEquals(home.addressbookWithName(name), None)
-
- self.commit()
-
- # Make sure notification fired after commit
- self.assertEquals(
- self.notifierFactory.history,
- [("update", "home1"), ("update", "home1"), ("update", "home1")]
- )
-
-
- def test_removeAddressBookWithName_absent(self):
- """
- Attempt to remove an non-existing addressbook should raise.
- """
- home = self.homeUnderTest()
- self.assertRaises(NoSuchHomeChildError,
- home.removeAddressBookWithName, "xyzzy")
-
-
- def test_addressbookObjects(self):
- """
- L{IAddressBook.addressbookObjects} will enumerate the addressbook objects present
- in the filesystem, in name order, but skip those with hidden names.
- """
- addressbook1 = self.addressbookUnderTest()
- addressbookObjects = list(addressbook1.addressbookObjects())
-
- for addressbookObject in addressbookObjects:
- self.assertProvides(IAddressBookObject, addressbookObject)
- self.assertEquals(
- addressbook1.addressbookObjectWithName(addressbookObject.name()),
- addressbookObject
- )
-
- self.assertEquals(
- set(o.name() for o in addressbookObjects),
- set(addressbook1_objectNames)
- )
-
-
- def test_addressbookObjectsWithRemovedObject(self):
- """
- L{IAddressBook.addressbookObjects} skips those objects which have been
- removed by L{AddressBook.removeAddressBookObjectWithName} in the same
- transaction, even if it has not yet been committed.
- """
- addressbook1 = self.addressbookUnderTest()
- addressbook1.removeAddressBookObjectWithName("2.vcf")
- addressbookObjects = list(addressbook1.addressbookObjects())
- self.assertEquals(set(o.name() for o in addressbookObjects),
- set(addressbook1_objectNames) - set(["2.vcf"]))
-
-
- def test_ownerAddressBookHome(self):
- """
- L{IAddressBook.ownerAddressBookHome} should match the home UID.
- """
- self.assertEquals(
- self.addressbookUnderTest().ownerAddressBookHome().uid(),
- self.homeUnderTest().uid()
- )
-
-
- def test_addressbookObjectWithName_exists(self):
- """
- L{IAddressBook.addressbookObjectWithName} returns an L{IAddressBookObject}
- provider for addressbooks which already exist.
- """
- addressbook1 = self.addressbookUnderTest()
- for name in addressbook1_objectNames:
- addressbookObject = addressbook1.addressbookObjectWithName(name)
- self.assertProvides(IAddressBookObject, addressbookObject)
- self.assertEquals(addressbookObject.name(), name)
- # FIXME: add more tests based on CommonTests.requirements
-
-
- def test_addressbookObjectWithName_absent(self):
- """
- L{IAddressBook.addressbookObjectWithName} returns C{None} for addressbooks which
- don't exist.
- """
- addressbook1 = self.addressbookUnderTest()
- self.assertEquals(addressbook1.addressbookObjectWithName("xyzzy"), None)
-
-
- def test_removeAddressBookObjectWithUID_exists(self):
- """
- Remove an existing addressbook object.
- """
- addressbook = self.addressbookUnderTest()
- for name in addressbook1_objectNames:
- uid = (u'uid' + name.rstrip(".vcf"))
- self.assertNotIdentical(addressbook.addressbookObjectWithUID(uid),
- None)
- addressbook.removeAddressBookObjectWithUID(uid)
- self.assertEquals(
- addressbook.addressbookObjectWithUID(uid),
- None
- )
- self.assertEquals(
- addressbook.addressbookObjectWithName(name),
- None
- )
-
-
- def test_removeAddressBookObjectWithName_exists(self):
- """
- Remove an existing addressbook object.
- """
- addressbook = self.addressbookUnderTest()
- for name in addressbook1_objectNames:
- self.assertNotIdentical(
- addressbook.addressbookObjectWithName(name), None
- )
- addressbook.removeAddressBookObjectWithName(name)
- self.assertIdentical(
- addressbook.addressbookObjectWithName(name), None
- )
-
- # Make sure notifications are fired after commit
- self.commit()
- self.assertEquals(
- self.notifierFactory.history,
- [
- ("update", "home1"),
- ("update", "home1/addressbook_1"),
- ("update", "home1"),
- ("update", "home1/addressbook_1"),
- ("update", "home1"),
- ("update", "home1/addressbook_1"),
- ]
- )
-
-
- def test_removeAddressBookObjectWithName_absent(self):
- """
- Attempt to remove an non-existing addressbook object should raise.
- """
- addressbook = self.addressbookUnderTest()
- self.assertRaises(
- NoSuchObjectResourceError,
- addressbook.removeAddressBookObjectWithName, "xyzzy"
- )
-
-
- def test_addressbookName(self):
- """
- L{AddressBook.name} reflects the name of the addressbook.
- """
- self.assertEquals(self.addressbookUnderTest().name(), "addressbook_1")
-
-
- def test_addressbookObjectName(self):
- """
- L{IAddressBookObject.name} reflects the name of the addressbook object.
- """
- self.assertEquals(self.addressbookObjectUnderTest().name(), "1.vcf")
-
-
- def test_component(self):
- """
- L{IAddressBookObject.component} returns a L{VComponent} describing the
- addressbook data underlying that addressbook object.
- """
- component = self.addressbookObjectUnderTest().component()
-
- self.failUnless(
- isinstance(component, VComponent),
- component
- )
-
- self.assertEquals(component.name(), "VCARD")
- self.assertEquals(component.resourceUID(), "uid1")
-
-
- def test_iAddressBookText(self):
- """
- L{IAddressBookObject.iAddressBookText} returns a C{str} describing the same
- data provided by L{IAddressBookObject.component}.
- """
- text = self.addressbookObjectUnderTest().vCardText()
- self.assertIsInstance(text, str)
- self.failUnless(text.startswith("BEGIN:VCARD\r\n"))
- self.assertIn("\r\nUID:uid1\r\n", text)
- self.failUnless(text.endswith("\r\nEND:VCARD\r\n"))
-
-
- def test_addressbookObjectUID(self):
- """
- L{IAddressBookObject.uid} returns a C{str} describing the C{UID} property
- of the addressbook object's component.
- """
- self.assertEquals(self.addressbookObjectUnderTest().uid(), "uid1")
-
-
- def test_addressbookObjectWithUID_absent(self):
- """
- L{IAddressBook.addressbookObjectWithUID} returns C{None} for addressbooks which
- don't exist.
- """
- addressbook1 = self.addressbookUnderTest()
- self.assertEquals(addressbook1.addressbookObjectWithUID("xyzzy"), None)
-
-
- def test_addressbooks(self):
- """
- L{IAddressBookHome.addressbooks} returns an iterable of L{IAddressBook}
- providers, which are consistent with the results from
- L{IAddressBook.addressbookWithName}.
- """
- # Add a dot directory to make sure we don't find it
- # self.home1._path.child(".foo").createDirectory()
- home = self.homeUnderTest()
- addressbooks = list(home.addressbooks())
-
- for addressbook in addressbooks:
- self.assertProvides(IAddressBook, addressbook)
- self.assertEquals(addressbook,
- home.addressbookWithName(addressbook.name()))
-
- self.assertEquals(
- set(c.name() for c in addressbooks),
- set(home1_addressbookNames)
- )
-
-
- def test_addressbooksAfterAddAddressBook(self):
- """
- L{IAddressBookHome.addressbooks} includes addressbooks recently added with
- L{IAddressBookHome.createAddressBookWithName}.
- """
- home = self.homeUnderTest()
- before = set(x.name() for x in home.addressbooks())
- home.createAddressBookWithName("new-name")
- after = set(x.name() for x in home.addressbooks())
- self.assertEquals(before | set(['new-name']), after)
-
-
- def test_createAddressBookObjectWithName_absent(self):
- """
- L{IAddressBook.createAddressBookObjectWithName} creates a new
- L{IAddressBookObject}.
- """
- addressbook1 = self.addressbookUnderTest()
- name = "4.vcf"
- self.assertIdentical(addressbook1.addressbookObjectWithName(name), None)
- component = VComponent.fromString(vcard4_text)
- addressbook1.createAddressBookObjectWithName(name, component)
-
- addressbookObject = addressbook1.addressbookObjectWithName(name)
- self.assertEquals(addressbookObject.component(), component)
-
- self.commit()
-
- # Make sure notifications fire after commit
- self.assertEquals(
- self.notifierFactory.history,
- [
- ("update", "home1"),
- ("update", "home1/addressbook_1"),
- ]
- )
-
-
- def test_createAddressBookObjectWithName_exists(self):
- """
- L{IAddressBook.createAddressBookObjectWithName} raises
- L{AddressBookObjectNameAlreadyExistsError} if a addressbook object with the
- given name already exists in that addressbook.
- """
- self.assertRaises(
- ObjectResourceNameAlreadyExistsError,
- self.addressbookUnderTest().createAddressBookObjectWithName,
- "1.vcf", VComponent.fromString(vcard4_text)
- )
-
-
- def test_createAddressBookObjectWithName_invalid(self):
- """
- L{IAddressBook.createAddressBookObjectWithName} raises
- L{InvalidAddressBookComponentError} if presented with invalid iAddressBook
- text.
- """
- self.assertRaises(
- InvalidObjectResourceError,
- self.addressbookUnderTest().createAddressBookObjectWithName,
- "new", VComponent.fromString(vcard4notCardDAV_text)
- )
-
-
- def test_setComponent_invalid(self):
- """
- L{IAddressBookObject.setComponent} raises L{InvalidIAddressBookDataError} if
- presented with invalid iAddressBook text.
- """
- addressbookObject = self.addressbookObjectUnderTest()
- self.assertRaises(
- InvalidObjectResourceError,
- addressbookObject.setComponent,
- VComponent.fromString(vcard4notCardDAV_text)
- )
-
-
- def test_setComponent_uidchanged(self):
- """
- L{IAddressBookObject.setComponent} raises L{InvalidAddressBookComponentError}
- when given a L{VComponent} whose UID does not match its existing UID.
- """
- addressbook1 = self.addressbookUnderTest()
- component = VComponent.fromString(vcard4_text)
- addressbookObject = addressbook1.addressbookObjectWithName("1.vcf")
- self.assertRaises(
- InvalidObjectResourceError,
- addressbookObject.setComponent, component
- )
-
-
- def test_addressbookHomeWithUID_create(self):
- """
- L{IAddressBookStoreTransaction.addressbookHomeWithUID} with C{create=True}
- will create a addressbook home that doesn't exist yet.
- """
- txn = self.transactionUnderTest()
- noHomeUID = "xyzzy"
- addressbookHome = txn.addressbookHomeWithUID(
- noHomeUID,
- create=True
- )
- def readOtherTxn():
- otherTxn = self.savedStore.newTransaction()
- self.addCleanup(otherTxn.commit)
- return otherTxn.addressbookHomeWithUID(noHomeUID)
- self.assertProvides(IAddressBookHome, addressbookHome)
- # A concurrent transaction shouldn't be able to read it yet:
- self.assertIdentical(readOtherTxn(), None)
- self.commit()
- # But once it's committed, other transactions should see it.
- self.assertProvides(IAddressBookHome, readOtherTxn())
-
-
- def test_setComponent(self):
- """
- L{AddressBookObject.setComponent} changes the result of
- L{AddressBookObject.component} within the same transaction.
- """
- component = VComponent.fromString(vcard1modified_text)
-
- addressbook1 = self.addressbookUnderTest()
- addressbookObject = addressbook1.addressbookObjectWithName("1.vcf")
- oldComponent = addressbookObject.component()
- self.assertNotEqual(component, oldComponent)
- addressbookObject.setComponent(component)
- self.assertEquals(addressbookObject.component(), component)
-
- # Also check a new instance
- addressbookObject = addressbook1.addressbookObjectWithName("1.vcf")
- self.assertEquals(addressbookObject.component(), component)
-
- self.commit()
-
- # Make sure notification fired after commit
- self.assertEquals(
- self.notifierFactory.history,
- [
- ("update", "home1"),
- ("update", "home1/addressbook_1"),
- ]
- )
-
- def checkPropertiesMethod(self, thunk):
- """
- Verify that the given object has a properties method that returns an
- L{IPropertyStore}.
- """
- properties = thunk.properties()
- self.assertProvides(IPropertyStore, properties)
-
-
- def test_homeProperties(self):
- """
- L{IAddressBookHome.properties} returns a property store.
- """
- self.checkPropertiesMethod(self.homeUnderTest())
-
-
- def test_addressbookProperties(self):
- """
- L{IAddressBook.properties} returns a property store.
- """
- self.checkPropertiesMethod(self.addressbookUnderTest())
-
-
- def test_addressbookObjectProperties(self):
- """
- L{IAddressBookObject.properties} returns a property store.
- """
- self.checkPropertiesMethod(self.addressbookObjectUnderTest())
-
-
- def test_newAddressBookObjectProperties(self):
- """
- L{IAddressBookObject.properties} returns an empty property store for a
- addressbook object which has been created but not committed.
- """
- addressbook = self.addressbookUnderTest()
- addressbook.createAddressBookObjectWithName(
- "4.vcf", VComponent.fromString(vcard4_text)
- )
- newEvent = addressbook.addressbookObjectWithName("4.vcf")
- self.assertEquals(newEvent.properties().items(), [])
-
-
- def test_setComponentPreservesProperties(self):
- """
- L{IAddressBookObject.setComponent} preserves properties.
-
- (Some implementations must go to extra trouble to provide this
- behavior; for example, file storage must copy extended attributes from
- the existing file to the temporary file replacing it.)
- """
- propertyName = PropertyName("http://example.com/ns", "example")
- propertyContent = WebDAVUnknownElement("sample content")
- propertyContent.name = propertyName.name
- propertyContent.namespace = propertyName.namespace
-
- self.addressbookObjectUnderTest().properties()[
- propertyName] = propertyContent
- self.commit()
- # Sanity check; are properties even readable in a separate transaction?
- # Should probably be a separate test.
- self.assertEquals(
- self.addressbookObjectUnderTest().properties()[propertyName],
- propertyContent)
- obj = self.addressbookObjectUnderTest()
- vcard1_text = obj.vCardText()
- vcard1_text_withDifferentNote = vcard1_text.replace(
- "NOTE:CardDAV protocol updates",
- "NOTE:Changed"
- )
- # Sanity check; make sure the test has the right idea of the subject.
- self.assertNotEquals(vcard1_text, vcard1_text_withDifferentNote)
- newComponent = VComponent.fromString(vcard1_text_withDifferentNote)
- obj.setComponent(newComponent)
-
- # Putting everything into a separate transaction to account for any
- # caching that may take place.
- self.commit()
- self.assertEquals(
- self.addressbookObjectUnderTest().properties()[propertyName],
- propertyContent
- )
-
-
- def test_dontLeakAddressbooks(self):
- """
- Addressbooks in one user's addressbook home should not show up in another
- user's addressbook home.
- """
- home2 = self.transactionUnderTest().addressbookHomeWithUID(
- "home2", create=True)
- self.assertIdentical(home2.addressbookWithName("addressbook_1"), None)
-
-
- def test_dontLeakObjects(self):
- """
- Addressbook objects in one user's addressbook should not show up in another
- user's via uid or name queries.
- """
- home1 = self.homeUnderTest()
- home2 = self.transactionUnderTest().addressbookHomeWithUID(
- "home2", create=True)
- addressbook1 = home1.addressbookWithName("addressbook_1")
- addressbook2 = home2.addressbookWithName("addressbook")
- objects = list(home2.addressbookWithName("addressbook").addressbookObjects())
- self.assertEquals(objects, [])
- for resourceName in self.requirements['home1']['addressbook_1'].keys():
- obj = addressbook1.addressbookObjectWithName(resourceName)
- self.assertIdentical(
- addressbook2.addressbookObjectWithName(resourceName), None)
- self.assertIdentical(
- addressbook2.addressbookObjectWithUID(obj.uid()), None)
-
-
-
-class StubNotifierFactory(object):
-
- """ For testing push notifications without an XMPP server """
-
- def __init__(self):
- self.reset()
-
- def newNotifier(self, label="default", id=None):
- return Notifier(self, label=label, id=id)
-
- def send(self, op, id):
- self.history.append((op, id))
-
- def reset(self):
- self.history = []
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/common.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/common.py)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/common.py (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/common.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,897 @@
+# -*- test-case-name: txdav.carddav.datastore,txdav.carddav.datastore.test.test_sql.AddressBookSQLStorageTests -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+"""
+Tests for common addressbook store API functions.
+"""
+
+from zope.interface.verify import verifyObject
+from zope.interface.exceptions import (
+ BrokenMethodImplementation, DoesNotImplement
+)
+
+from txdav.idav import IPropertyStore, IDataStore
+from txdav.base.propertystore.base import PropertyName
+
+from txdav.common.icommondatastore import (
+ HomeChildNameAlreadyExistsError, ICommonTransaction
+)
+from txdav.common.icommondatastore import InvalidObjectResourceError
+from txdav.common.icommondatastore import NoSuchHomeChildError
+from txdav.common.icommondatastore import NoSuchObjectResourceError
+from txdav.common.icommondatastore import ObjectResourceNameAlreadyExistsError
+
+from txdav.carddav.iaddressbookstore import (
+ IAddressBookObject, IAddressBookHome,
+ IAddressBook, IAddressBookTransaction
+)
+from twistedcaldav.vcard import Component as VComponent
+from twistedcaldav.notify import Notifier
+
+from twext.python.filepath import CachingFilePath as FilePath
+from twext.web2.dav import davxml
+from twext.web2.dav.element.base import WebDAVUnknownElement
+
+
+storePath = FilePath(__file__).parent().child("addressbook_store")
+
+homeRoot = storePath.child("ho").child("me").child("home1")
+
+adbk1Root = homeRoot.child("addressbook_1")
+
+addressbook1_objectNames = [
+ "1.vcf",
+ "2.vcf",
+ "3.vcf",
+]
+
+
+home1_addressbookNames = [
+ "addressbook_1",
+ "addressbook_2",
+ "addressbook_empty",
+]
+
+
+vcard4_text = (
+ """BEGIN:VCARD
+VERSION:3.0
+N:Thompson;Default;;;
+FN:Default Thompson
+EMAIL;type=INTERNET;type=WORK;type=pref:lthompson at example.com
+TEL;type=WORK;type=pref:1-555-555-5555
+TEL;type=CELL:1-444-444-4444
+item1.ADR;type=WORK;type=pref:;;1245 Test;Sesame Street;California;11111;USA
+item1.X-ABADR:us
+UID:uid4
+END:VCARD
+""".replace("\n", "\r\n")
+)
+
+
+
+vcard4notCardDAV_text = ( # Missing UID, N and FN
+"""BEGIN:VCARD
+VERSION:3.0
+EMAIL;type=INTERNET;type=WORK;type=pref:lthompson at example.com
+TEL;type=WORK;type=pref:1-555-555-5555
+TEL;type=CELL:1-444-444-4444
+item1.ADR;type=WORK;type=pref:;;1245 Test;Sesame Street;California;11111;USA
+item1.X-ABADR:us
+END:VCARD
+""".replace("\n", "\r\n")
+)
+
+
+
+vcard1modified_text = vcard4_text.replace(
+ "\r\nUID:uid4\r\n",
+ "\r\nUID:uid1\r\n"
+)
+
+
+def assertProvides(testCase, interface, provider):
+ """
+ Verify that C{provider} properly provides C{interface}
+
+ @type interface: L{zope.interface.Interface}
+ @type provider: C{provider}
+ """
+ try:
+ verifyObject(interface, provider)
+ except BrokenMethodImplementation, e:
+ testCase.fail(e)
+ except DoesNotImplement, e:
+ testCase.fail("%r does not provide %s.%s" %
+ (provider, interface.__module__, interface.getName()))
+
+
+
+class CommonTests(object):
+ """
+ Tests for common functionality of interfaces defined in
+ L{txdav.carddav.iaddressbookstore}.
+ """
+
+ requirements = {
+ "home1": {
+ "addressbook_1": {
+ "1.vcf": adbk1Root.child("1.vcf").getContent(),
+ "2.vcf": adbk1Root.child("2.vcf").getContent(),
+ "3.vcf": adbk1Root.child("3.vcf").getContent()
+ },
+ "addressbook_2": {},
+ "addressbook_empty": {},
+ "not_a_addressbook": None
+ },
+ "not_a_home": None
+ }
+
+ def storeUnderTest(self):
+ """
+ Subclasses must override this to return an L{IAddressBookStore}
+ provider which adheres to the structure detailed by
+ L{CommonTests.requirements}. This attribute is a dict of dict of dicts;
+ the outermost layer representing UIDs mapping to addressbook homes,
+ then addressbook names mapping to addressbook collections, and finally
+ addressbook object names mapping to addressbook object text.
+ """
+ raise NotImplementedError()
+
+
+ lastTransaction = None
+ savedStore = None
+
+ def transactionUnderTest(self):
+ """
+ Create a transaction from C{storeUnderTest} and save it as
+ C[lastTransaction}. Also makes sure to use the same store, saving the
+ value from C{storeUnderTest}.
+ """
+ if self.lastTransaction is not None:
+ return self.lastTransaction
+ if self.savedStore is None:
+ self.savedStore = self.storeUnderTest()
+ txn = self.lastTransaction = self.savedStore.newTransaction(self.id())
+ return txn
+
+
+ def commit(self):
+ """
+ Commit the last transaction created from C{transactionUnderTest}, and
+ clear it.
+ """
+ self.lastTransaction.commit()
+ self.lastTransaction = None
+
+
+ def abort(self):
+ """
+ Abort the last transaction created from C[transactionUnderTest}, and
+ clear it.
+ """
+ self.lastTransaction.abort()
+ self.lastTransaction = None
+
+ def setUp(self):
+ self.notifierFactory = StubNotifierFactory()
+
+ def tearDown(self):
+ if self.lastTransaction is not None:
+ self.commit()
+
+
+
+ def homeUnderTest(self):
+ """
+ Get the addressbook home detailed by C{requirements['home1']}.
+ """
+ return self.transactionUnderTest().addressbookHomeWithUID("home1")
+
+
+ def addressbookUnderTest(self):
+ """
+ Get the addressbook detailed by C{requirements['home1']['addressbook_1']}.
+ """
+ return self.homeUnderTest().addressbookWithName("addressbook_1")
+
+
+ def addressbookObjectUnderTest(self):
+ """
+ Get the addressbook detailed by
+ C{requirements['home1']['addressbook_1']['1.vcf']}.
+ """
+ return self.addressbookUnderTest().addressbookObjectWithName("1.vcf")
+
+
+ assertProvides = assertProvides
+
+
+ def test_addressbookStoreProvides(self):
+ """
+ The addressbook store provides L{IAddressBookStore} and its required
+ attributes.
+ """
+ addressbookStore = self.storeUnderTest()
+ self.assertProvides(IDataStore, addressbookStore)
+
+
+ def test_transactionProvides(self):
+ """
+ The transactions generated by the addressbook store provide
+ L{IAddressBookStoreTransaction} and its required attributes.
+ """
+ txn = self.transactionUnderTest()
+ self.assertProvides(ICommonTransaction, txn)
+ self.assertProvides(IAddressBookTransaction, txn)
+
+
+ def test_homeProvides(self):
+ """
+ The addressbook homes generated by the addressbook store provide
+ L{IAddressBookHome} and its required attributes.
+ """
+ self.assertProvides(IAddressBookHome, self.homeUnderTest())
+
+
+ def test_addressbookProvides(self):
+ """
+ The addressbooks generated by the addressbook store provide L{IAddressBook} and
+ its required attributes.
+ """
+ self.assertProvides(IAddressBook, self.addressbookUnderTest())
+
+
+ def test_addressbookObjectProvides(self):
+ """
+ The addressbook objects generated by the addressbook store provide
+ L{IAddressBookObject} and its required attributes.
+ """
+ self.assertProvides(IAddressBookObject, self.addressbookObjectUnderTest())
+
+ def test_notifierID(self):
+ home = self.homeUnderTest()
+ self.assertEquals(home.notifierID(), "home1")
+ addressbook = home.addressbookWithName("addressbook_1")
+ self.assertEquals(addressbook.notifierID(), "home1")
+ self.assertEquals(addressbook.notifierID(label="collection"), "home1/addressbook_1")
+
+
+ def test_addressbookHomeWithUID_exists(self):
+ """
+ Finding an existing addressbook home by UID results in an object that
+ provides L{IAddressBookHome} and has a C{uid()} method that returns the
+ same value that was passed in.
+ """
+ addressbookHome = (self.transactionUnderTest()
+ .addressbookHomeWithUID("home1"))
+ self.assertEquals(addressbookHome.uid(), "home1")
+ self.assertProvides(IAddressBookHome, addressbookHome)
+
+
+ def test_addressbookHomeWithUID_absent(self):
+ """
+ L{IAddressBookStoreTransaction.addressbookHomeWithUID} should return C{None}
+ when asked for a non-existent addressbook home.
+ """
+ txn = self.transactionUnderTest()
+ self.assertEquals(txn.addressbookHomeWithUID("xyzzy"), None)
+
+
+ def test_addressbookWithName_exists(self):
+ """
+ L{IAddressBookHome.addressbookWithName} returns an L{IAddressBook} provider,
+ whose name matches the one passed in.
+ """
+ home = self.homeUnderTest()
+ for name in home1_addressbookNames:
+ addressbook = home.addressbookWithName(name)
+ if addressbook is None:
+ self.fail("addressbook %r didn't exist" % (name,))
+ self.assertProvides(IAddressBook, addressbook)
+ self.assertEquals(addressbook.name(), name)
+
+
+ def test_addressbookRename(self):
+ """
+ L{IAddressBook.rename} changes the name of the L{IAddressBook}.
+ """
+ home = self.homeUnderTest()
+ addressbook = home.addressbookWithName("addressbook_1")
+ addressbook.rename("some_other_name")
+ def positiveAssertions():
+ self.assertEquals(addressbook.name(), "some_other_name")
+ self.assertEquals(addressbook, home.addressbookWithName("some_other_name"))
+ self.assertEquals(None, home.addressbookWithName("addressbook_1"))
+ positiveAssertions()
+ self.commit()
+ home = self.homeUnderTest()
+ addressbook = home.addressbookWithName("some_other_name")
+ positiveAssertions()
+ # FIXME: revert
+ # FIXME: test for multiple renames
+ # FIXME: test for conflicting renames (a->b, c->a in the same txn)
+
+
+ def test_addressbookWithName_absent(self):
+ """
+ L{IAddressBookHome.addressbookWithName} returns C{None} for addressbooks which
+ do not exist.
+ """
+ self.assertEquals(self.homeUnderTest().addressbookWithName("xyzzy"),
+ None)
+
+
+ def test_createAddressBookWithName_absent(self):
+ """
+ L{IAddressBookHome.createAddressBookWithName} creates a new L{IAddressBook} that
+ can be retrieved with L{IAddressBookHome.addressbookWithName}.
+ """
+ home = self.homeUnderTest()
+ name = "new"
+ self.assertIdentical(home.addressbookWithName(name), None)
+ home.createAddressBookWithName(name)
+ self.assertNotIdentical(home.addressbookWithName(name), None)
+ def checkProperties():
+ addressbookProperties = home.addressbookWithName(name).properties()
+ self.assertEquals(
+ addressbookProperties[
+ PropertyName.fromString(davxml.ResourceType.sname())
+ ],
+ davxml.ResourceType.addressbook
+ ) #@UndefinedVariable
+ checkProperties()
+ self.commit()
+
+ # Make sure notification fired after commit
+ self.assertEquals(self.notifierFactory.history, [("update", "home1")])
+
+ # Make sure it's available in a new transaction; i.e. test the commit.
+ home = self.homeUnderTest()
+ self.assertNotIdentical(home.addressbookWithName(name), None)
+
+ # FIXME: These two lines aren't in the calendar common tests:
+ # home = self.addressbookStore.newTransaction().addressbookHomeWithUID(
+ # "home1")
+
+ # Sanity check: are the properties actually persisted?
+ # FIXME: no independent testing of this right now
+ checkProperties()
+
+
+ def test_createAddressBookWithName_exists(self):
+ """
+ L{IAddressBookHome.createAddressBookWithName} raises
+ L{AddressBookAlreadyExistsError} when the name conflicts with an already-
+ existing address book.
+ """
+ for name in home1_addressbookNames:
+ self.assertRaises(
+ HomeChildNameAlreadyExistsError,
+ self.homeUnderTest().createAddressBookWithName, name
+ )
+
+
+ def test_removeAddressBookWithName_exists(self):
+ """
+ L{IAddressBookHome.removeAddressBookWithName} removes a addressbook that already
+ exists.
+ """
+ home = self.homeUnderTest()
+ # FIXME: test transactions
+ for name in home1_addressbookNames:
+ self.assertNotIdentical(home.addressbookWithName(name), None)
+ home.removeAddressBookWithName(name)
+ self.assertEquals(home.addressbookWithName(name), None)
+
+ self.commit()
+
+ # Make sure notification fired after commit
+ self.assertEquals(
+ self.notifierFactory.history,
+ [("update", "home1"), ("update", "home1"), ("update", "home1")]
+ )
+
+
+ def test_removeAddressBookWithName_absent(self):
+ """
+ Attempt to remove an non-existing addressbook should raise.
+ """
+ home = self.homeUnderTest()
+ self.assertRaises(NoSuchHomeChildError,
+ home.removeAddressBookWithName, "xyzzy")
+
+
+ def test_addressbookObjects(self):
+ """
+ L{IAddressBook.addressbookObjects} will enumerate the addressbook objects present
+ in the filesystem, in name order, but skip those with hidden names.
+ """
+ addressbook1 = self.addressbookUnderTest()
+ addressbookObjects = list(addressbook1.addressbookObjects())
+
+ for addressbookObject in addressbookObjects:
+ self.assertProvides(IAddressBookObject, addressbookObject)
+ self.assertEquals(
+ addressbook1.addressbookObjectWithName(addressbookObject.name()),
+ addressbookObject
+ )
+
+ self.assertEquals(
+ set(o.name() for o in addressbookObjects),
+ set(addressbook1_objectNames)
+ )
+
+
+ def test_addressbookObjectsWithRemovedObject(self):
+ """
+ L{IAddressBook.addressbookObjects} skips those objects which have been
+ removed by L{AddressBook.removeAddressBookObjectWithName} in the same
+ transaction, even if it has not yet been committed.
+ """
+ addressbook1 = self.addressbookUnderTest()
+ addressbook1.removeAddressBookObjectWithName("2.vcf")
+ addressbookObjects = list(addressbook1.addressbookObjects())
+ self.assertEquals(set(o.name() for o in addressbookObjects),
+ set(addressbook1_objectNames) - set(["2.vcf"]))
+
+
+ def test_ownerAddressBookHome(self):
+ """
+ L{IAddressBook.ownerAddressBookHome} should match the home UID.
+ """
+ self.assertEquals(
+ self.addressbookUnderTest().ownerAddressBookHome().uid(),
+ self.homeUnderTest().uid()
+ )
+
+
+ def test_addressbookObjectWithName_exists(self):
+ """
+ L{IAddressBook.addressbookObjectWithName} returns an L{IAddressBookObject}
+ provider for addressbooks which already exist.
+ """
+ addressbook1 = self.addressbookUnderTest()
+ for name in addressbook1_objectNames:
+ addressbookObject = addressbook1.addressbookObjectWithName(name)
+ self.assertProvides(IAddressBookObject, addressbookObject)
+ self.assertEquals(addressbookObject.name(), name)
+ # FIXME: add more tests based on CommonTests.requirements
+
+
+ def test_addressbookObjectWithName_absent(self):
+ """
+ L{IAddressBook.addressbookObjectWithName} returns C{None} for addressbooks which
+ don't exist.
+ """
+ addressbook1 = self.addressbookUnderTest()
+ self.assertEquals(addressbook1.addressbookObjectWithName("xyzzy"), None)
+
+
+ def test_removeAddressBookObjectWithUID_exists(self):
+ """
+ Remove an existing addressbook object.
+ """
+ addressbook = self.addressbookUnderTest()
+ for name in addressbook1_objectNames:
+ uid = (u'uid' + name.rstrip(".vcf"))
+ self.assertNotIdentical(addressbook.addressbookObjectWithUID(uid),
+ None)
+ addressbook.removeAddressBookObjectWithUID(uid)
+ self.assertEquals(
+ addressbook.addressbookObjectWithUID(uid),
+ None
+ )
+ self.assertEquals(
+ addressbook.addressbookObjectWithName(name),
+ None
+ )
+
+
+ def test_removeAddressBookObjectWithName_exists(self):
+ """
+ Remove an existing addressbook object.
+ """
+ addressbook = self.addressbookUnderTest()
+ for name in addressbook1_objectNames:
+ self.assertNotIdentical(
+ addressbook.addressbookObjectWithName(name), None
+ )
+ addressbook.removeAddressBookObjectWithName(name)
+ self.assertIdentical(
+ addressbook.addressbookObjectWithName(name), None
+ )
+
+ # Make sure notifications are fired after commit
+ self.commit()
+ self.assertEquals(
+ self.notifierFactory.history,
+ [
+ ("update", "home1"),
+ ("update", "home1/addressbook_1"),
+ ("update", "home1"),
+ ("update", "home1/addressbook_1"),
+ ("update", "home1"),
+ ("update", "home1/addressbook_1"),
+ ]
+ )
+
+
+ def test_removeAddressBookObjectWithName_absent(self):
+ """
+ Attempt to remove an non-existing addressbook object should raise.
+ """
+ addressbook = self.addressbookUnderTest()
+ self.assertRaises(
+ NoSuchObjectResourceError,
+ addressbook.removeAddressBookObjectWithName, "xyzzy"
+ )
+
+
+ def test_addressbookName(self):
+ """
+ L{AddressBook.name} reflects the name of the addressbook.
+ """
+ self.assertEquals(self.addressbookUnderTest().name(), "addressbook_1")
+
+
+ def test_addressbookObjectName(self):
+ """
+ L{IAddressBookObject.name} reflects the name of the addressbook object.
+ """
+ self.assertEquals(self.addressbookObjectUnderTest().name(), "1.vcf")
+
+
+ def test_component(self):
+ """
+ L{IAddressBookObject.component} returns a L{VComponent} describing the
+ addressbook data underlying that addressbook object.
+ """
+ component = self.addressbookObjectUnderTest().component()
+
+ self.failUnless(
+ isinstance(component, VComponent),
+ component
+ )
+
+ self.assertEquals(component.name(), "VCARD")
+ self.assertEquals(component.resourceUID(), "uid1")
+
+
+ def test_iAddressBookText(self):
+ """
+ L{IAddressBookObject.iAddressBookText} returns a C{str} describing the same
+ data provided by L{IAddressBookObject.component}.
+ """
+ text = self.addressbookObjectUnderTest().vCardText()
+ self.assertIsInstance(text, str)
+ self.failUnless(text.startswith("BEGIN:VCARD\r\n"))
+ self.assertIn("\r\nUID:uid1\r\n", text)
+ self.failUnless(text.endswith("\r\nEND:VCARD\r\n"))
+
+
+ def test_addressbookObjectUID(self):
+ """
+ L{IAddressBookObject.uid} returns a C{str} describing the C{UID} property
+ of the addressbook object's component.
+ """
+ self.assertEquals(self.addressbookObjectUnderTest().uid(), "uid1")
+
+
+ def test_addressbookObjectWithUID_absent(self):
+ """
+ L{IAddressBook.addressbookObjectWithUID} returns C{None} for addressbooks which
+ don't exist.
+ """
+ addressbook1 = self.addressbookUnderTest()
+ self.assertEquals(addressbook1.addressbookObjectWithUID("xyzzy"), None)
+
+
+ def test_addressbooks(self):
+ """
+ L{IAddressBookHome.addressbooks} returns an iterable of L{IAddressBook}
+ providers, which are consistent with the results from
+ L{IAddressBook.addressbookWithName}.
+ """
+ # Add a dot directory to make sure we don't find it
+ # self.home1._path.child(".foo").createDirectory()
+ home = self.homeUnderTest()
+ addressbooks = list(home.addressbooks())
+
+ for addressbook in addressbooks:
+ self.assertProvides(IAddressBook, addressbook)
+ self.assertEquals(addressbook,
+ home.addressbookWithName(addressbook.name()))
+
+ self.assertEquals(
+ set(c.name() for c in addressbooks),
+ set(home1_addressbookNames)
+ )
+
+
+ def test_addressbooksAfterAddAddressBook(self):
+ """
+ L{IAddressBookHome.addressbooks} includes addressbooks recently added with
+ L{IAddressBookHome.createAddressBookWithName}.
+ """
+ home = self.homeUnderTest()
+ before = set(x.name() for x in home.addressbooks())
+ home.createAddressBookWithName("new-name")
+ after = set(x.name() for x in home.addressbooks())
+ self.assertEquals(before | set(['new-name']), after)
+
+
+ def test_createAddressBookObjectWithName_absent(self):
+ """
+ L{IAddressBook.createAddressBookObjectWithName} creates a new
+ L{IAddressBookObject}.
+ """
+ addressbook1 = self.addressbookUnderTest()
+ name = "4.vcf"
+ self.assertIdentical(addressbook1.addressbookObjectWithName(name), None)
+ component = VComponent.fromString(vcard4_text)
+ addressbook1.createAddressBookObjectWithName(name, component)
+
+ addressbookObject = addressbook1.addressbookObjectWithName(name)
+ self.assertEquals(addressbookObject.component(), component)
+
+ self.commit()
+
+ # Make sure notifications fire after commit
+ self.assertEquals(
+ self.notifierFactory.history,
+ [
+ ("update", "home1"),
+ ("update", "home1/addressbook_1"),
+ ]
+ )
+
+
+ def test_createAddressBookObjectWithName_exists(self):
+ """
+ L{IAddressBook.createAddressBookObjectWithName} raises
+ L{AddressBookObjectNameAlreadyExistsError} if a addressbook object with the
+ given name already exists in that addressbook.
+ """
+ self.assertRaises(
+ ObjectResourceNameAlreadyExistsError,
+ self.addressbookUnderTest().createAddressBookObjectWithName,
+ "1.vcf", VComponent.fromString(vcard4_text)
+ )
+
+
+ def test_createAddressBookObjectWithName_invalid(self):
+ """
+ L{IAddressBook.createAddressBookObjectWithName} raises
+ L{InvalidAddressBookComponentError} if presented with invalid iAddressBook
+ text.
+ """
+ self.assertRaises(
+ InvalidObjectResourceError,
+ self.addressbookUnderTest().createAddressBookObjectWithName,
+ "new", VComponent.fromString(vcard4notCardDAV_text)
+ )
+
+
+ def test_setComponent_invalid(self):
+ """
+ L{IAddressBookObject.setComponent} raises L{InvalidIAddressBookDataError} if
+ presented with invalid iAddressBook text.
+ """
+ addressbookObject = self.addressbookObjectUnderTest()
+ self.assertRaises(
+ InvalidObjectResourceError,
+ addressbookObject.setComponent,
+ VComponent.fromString(vcard4notCardDAV_text)
+ )
+
+
+ def test_setComponent_uidchanged(self):
+ """
+ L{IAddressBookObject.setComponent} raises L{InvalidAddressBookComponentError}
+ when given a L{VComponent} whose UID does not match its existing UID.
+ """
+ addressbook1 = self.addressbookUnderTest()
+ component = VComponent.fromString(vcard4_text)
+ addressbookObject = addressbook1.addressbookObjectWithName("1.vcf")
+ self.assertRaises(
+ InvalidObjectResourceError,
+ addressbookObject.setComponent, component
+ )
+
+
+ def test_addressbookHomeWithUID_create(self):
+ """
+ L{IAddressBookStoreTransaction.addressbookHomeWithUID} with C{create=True}
+ will create a addressbook home that doesn't exist yet.
+ """
+ txn = self.transactionUnderTest()
+ noHomeUID = "xyzzy"
+ addressbookHome = txn.addressbookHomeWithUID(
+ noHomeUID,
+ create=True
+ )
+ def readOtherTxn():
+ otherTxn = self.savedStore.newTransaction()
+ self.addCleanup(otherTxn.commit)
+ return otherTxn.addressbookHomeWithUID(noHomeUID)
+ self.assertProvides(IAddressBookHome, addressbookHome)
+ # A concurrent transaction shouldn't be able to read it yet:
+ self.assertIdentical(readOtherTxn(), None)
+ self.commit()
+ # But once it's committed, other transactions should see it.
+ self.assertProvides(IAddressBookHome, readOtherTxn())
+
+
+ def test_setComponent(self):
+ """
+ L{AddressBookObject.setComponent} changes the result of
+ L{AddressBookObject.component} within the same transaction.
+ """
+ component = VComponent.fromString(vcard1modified_text)
+
+ addressbook1 = self.addressbookUnderTest()
+ addressbookObject = addressbook1.addressbookObjectWithName("1.vcf")
+ oldComponent = addressbookObject.component()
+ self.assertNotEqual(component, oldComponent)
+ addressbookObject.setComponent(component)
+ self.assertEquals(addressbookObject.component(), component)
+
+ # Also check a new instance
+ addressbookObject = addressbook1.addressbookObjectWithName("1.vcf")
+ self.assertEquals(addressbookObject.component(), component)
+
+ self.commit()
+
+ # Make sure notification fired after commit
+ self.assertEquals(
+ self.notifierFactory.history,
+ [
+ ("update", "home1"),
+ ("update", "home1/addressbook_1"),
+ ]
+ )
+
+ def checkPropertiesMethod(self, thunk):
+ """
+ Verify that the given object has a properties method that returns an
+ L{IPropertyStore}.
+ """
+ properties = thunk.properties()
+ self.assertProvides(IPropertyStore, properties)
+
+
+ def test_homeProperties(self):
+ """
+ L{IAddressBookHome.properties} returns a property store.
+ """
+ self.checkPropertiesMethod(self.homeUnderTest())
+
+
+ def test_addressbookProperties(self):
+ """
+ L{IAddressBook.properties} returns a property store.
+ """
+ self.checkPropertiesMethod(self.addressbookUnderTest())
+
+
+ def test_addressbookObjectProperties(self):
+ """
+ L{IAddressBookObject.properties} returns a property store.
+ """
+ self.checkPropertiesMethod(self.addressbookObjectUnderTest())
+
+
+ def test_newAddressBookObjectProperties(self):
+ """
+ L{IAddressBookObject.properties} returns an empty property store for a
+ addressbook object which has been created but not committed.
+ """
+ addressbook = self.addressbookUnderTest()
+ addressbook.createAddressBookObjectWithName(
+ "4.vcf", VComponent.fromString(vcard4_text)
+ )
+ newEvent = addressbook.addressbookObjectWithName("4.vcf")
+ self.assertEquals(newEvent.properties().items(), [])
+
+
+ def test_setComponentPreservesProperties(self):
+ """
+ L{IAddressBookObject.setComponent} preserves properties.
+
+ (Some implementations must go to extra trouble to provide this
+ behavior; for example, file storage must copy extended attributes from
+ the existing file to the temporary file replacing it.)
+ """
+ propertyName = PropertyName("http://example.com/ns", "example")
+ propertyContent = WebDAVUnknownElement("sample content")
+ propertyContent.name = propertyName.name
+ propertyContent.namespace = propertyName.namespace
+
+ self.addressbookObjectUnderTest().properties()[
+ propertyName] = propertyContent
+ self.commit()
+ # Sanity check; are properties even readable in a separate transaction?
+ # Should probably be a separate test.
+ self.assertEquals(
+ self.addressbookObjectUnderTest().properties()[propertyName],
+ propertyContent)
+ obj = self.addressbookObjectUnderTest()
+ vcard1_text = obj.vCardText()
+ vcard1_text_withDifferentNote = vcard1_text.replace(
+ "NOTE:CardDAV protocol updates",
+ "NOTE:Changed"
+ )
+ # Sanity check; make sure the test has the right idea of the subject.
+ self.assertNotEquals(vcard1_text, vcard1_text_withDifferentNote)
+ newComponent = VComponent.fromString(vcard1_text_withDifferentNote)
+ obj.setComponent(newComponent)
+
+ # Putting everything into a separate transaction to account for any
+ # caching that may take place.
+ self.commit()
+ self.assertEquals(
+ self.addressbookObjectUnderTest().properties()[propertyName],
+ propertyContent
+ )
+
+
+ def test_dontLeakAddressbooks(self):
+ """
+ Addressbooks in one user's addressbook home should not show up in another
+ user's addressbook home.
+ """
+ home2 = self.transactionUnderTest().addressbookHomeWithUID(
+ "home2", create=True)
+ self.assertIdentical(home2.addressbookWithName("addressbook_1"), None)
+
+
+ def test_dontLeakObjects(self):
+ """
+ Addressbook objects in one user's addressbook should not show up in another
+ user's via uid or name queries.
+ """
+ home1 = self.homeUnderTest()
+ home2 = self.transactionUnderTest().addressbookHomeWithUID(
+ "home2", create=True)
+ addressbook1 = home1.addressbookWithName("addressbook_1")
+ addressbook2 = home2.addressbookWithName("addressbook")
+ objects = list(home2.addressbookWithName("addressbook").addressbookObjects())
+ self.assertEquals(objects, [])
+ for resourceName in self.requirements['home1']['addressbook_1'].keys():
+ obj = addressbook1.addressbookObjectWithName(resourceName)
+ self.assertIdentical(
+ addressbook2.addressbookObjectWithName(resourceName), None)
+ self.assertIdentical(
+ addressbook2.addressbookObjectWithUID(obj.uid()), None)
+
+
+
+class StubNotifierFactory(object):
+
+ """ For testing push notifications without an XMPP server """
+
+ def __init__(self):
+ self.reset()
+
+ def newNotifier(self, label="default", id=None):
+ return Notifier(self, label=label, id=id)
+
+ def send(self, op, id):
+ self.history.append((op, id))
+
+ def reset(self):
+ self.history = []
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/test_file.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/test_file.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/test_file.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,451 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-File addressbook store tests.
-"""
-
-from twext.python.filepath import CachingFilePath as FilePath
-from twisted.trial import unittest
-
-from twistedcaldav.vcard import Component as VComponent
-
-from txdav.common.icommondatastore import HomeChildNameNotAllowedError
-from txdav.common.icommondatastore import ObjectResourceNameNotAllowedError
-from txdav.common.icommondatastore import ObjectResourceUIDAlreadyExistsError
-from txdav.common.icommondatastore import NoSuchHomeChildError
-from txdav.common.icommondatastore import NoSuchObjectResourceError
-
-from txdav.carddav.datastore.file import AddressBookStore, AddressBookHome
-from txdav.carddav.datastore.file import AddressBook, AddressBookObject
-
-from txdav.carddav.datastore.test.common import (
- CommonTests, vcard4_text, vcard1modified_text, StubNotifierFactory)
-
-storePath = FilePath(__file__).parent().child("addressbook_store")
-
-def _todo(f, why):
- f.todo = why
- return f
-
-
-
-featureUnimplemented = lambda f: _todo(f, "Feature unimplemented")
-testUnimplemented = lambda f: _todo(f, "Test unimplemented")
-todo = lambda why: lambda f: _todo(f, why)
-
-
-def setUpAddressBookStore(test):
- test.root = FilePath(test.mktemp())
- test.root.createDirectory()
-
- storeRootPath = test.storeRootPath = test.root.child("store")
- addressbookPath = storeRootPath.child("addressbooks").child("__uids__")
- addressbookPath.parent().makedirs()
- storePath.copyTo(addressbookPath)
-
- test.notifierFactory = StubNotifierFactory()
- test.addressbookStore = AddressBookStore(storeRootPath, test.notifierFactory)
- test.txn = test.addressbookStore.newTransaction()
- assert test.addressbookStore is not None, "No addressbook store?"
-
-
-
-def setUpHome1(test):
- setUpAddressBookStore(test)
- test.home1 = test.txn.addressbookHomeWithUID("home1")
- assert test.home1 is not None, "No addressbook home?"
-
-
-
-def setUpAddressBook1(test):
- setUpHome1(test)
- test.addressbook1 = test.home1.addressbookWithName("addressbook_1")
- assert test.addressbook1 is not None, "No addressbook?"
-
-
-
-class AddressBookStoreTest(unittest.TestCase):
- """
- Test cases for L{AddressBookStore}.
- """
-
- def setUp(self):
- setUpAddressBookStore(self)
-
-
- def test_addressbookHomeWithUID_dot(self):
- """
- Filenames starting with "." are reserved by this
- implementation, so no UIDs may start with ".".
- """
- self.assertEquals(
- self.addressbookStore.newTransaction().addressbookHomeWithUID("xyzzy"),
- None
- )
-
-
-
-class AddressBookHomeTest(unittest.TestCase):
-
- def setUp(self):
- setUpHome1(self)
-
-
- def test_init(self):
- """
- L{AddressBookHome} has C{_path} and L{_addressbookStore} attributes,
- indicating its location on disk and parent store, respectively.
- """
- self.failUnless(
- isinstance(self.home1._path, FilePath),
- self.home1._path
- )
- self.assertEquals(
- self.home1._addressbookStore,
- self.addressbookStore
- )
-
-
- def test_addressbookWithName_dot(self):
- """
- Filenames starting with "." are reserved by this
- implementation, so no addressbook names may start with ".".
- """
- name = ".foo"
- self.home1._path.child(name).createDirectory()
- self.assertEquals(self.home1.addressbookWithName(name), None)
-
-
- def test_createAddressBookWithName_dot(self):
- """
- Filenames starting with "." are reserved by this
- implementation, so no addressbook names may start with ".".
- """
- self.assertRaises(
- HomeChildNameNotAllowedError,
- self.home1.createAddressBookWithName, ".foo"
- )
-
-
- def test_removeAddressBookWithName_dot(self):
- """
- Filenames starting with "." are reserved by this
- implementation, so no addressbook names may start with ".".
- """
- name = ".foo"
- self.home1._path.child(name).createDirectory()
- self.assertRaises(
- NoSuchHomeChildError,
- self.home1.removeAddressBookWithName, name
- )
-
-
-
-class AddressBookTest(unittest.TestCase):
-
- def setUp(self):
- setUpAddressBook1(self)
-
-
- def test_init(self):
- """
- L{AddressBook.__init__} sets private attributes to reflect its constructor
- arguments.
- """
- self.failUnless(
- isinstance(self.addressbook1._path, FilePath),
- self.addressbook1
- )
- self.failUnless(
- isinstance(self.addressbook1._addressbookHome, AddressBookHome),
- self.addressbook1._addressbookHome
- )
-
-
- def test_useIndexImmediately(self):
- """
- L{AddressBook._index} is usable in the same transaction it is created, with
- a temporary filename.
- """
- self.home1.createAddressBookWithName("addressbook2")
- addressbook = self.home1.addressbookWithName("addressbook2")
- index = addressbook._index
- self.assertEquals(set(index.addressbookObjects()),
- set(addressbook.addressbookObjects()))
- self.txn.commit()
- self.txn = self.addressbookStore.newTransaction()
- self.home1 = self.txn.addressbookHomeWithUID("home1")
- addressbook = self.home1.addressbookWithName("addressbook2")
- # FIXME: we should be curating our own index here, but in order to fix
- # that the code in the old implicit scheduler needs to change. This
- # test would be more effective if there were actually some objects in
- # this list.
- index = addressbook._index
- self.assertEquals(set(index.addressbookObjects()),
- set(addressbook.addressbookObjects()))
-
-
- def test_addressbookObjectWithName_dot(self):
- """
- Filenames starting with "." are reserved by this
- implementation, so no addressbook object names may start with
- ".".
- """
- name = ".foo.vcf"
- self.home1._path.child(name).touch()
- self.assertEquals(self.addressbook1.addressbookObjectWithName(name), None)
-
-
- @featureUnimplemented
- def test_addressbookObjectWithUID_exists(self):
- """
- Find existing addressbook object by name.
- """
- addressbookObject = self.addressbook1.addressbookObjectWithUID("1")
- self.failUnless(
- isinstance(addressbookObject, AddressBookObject),
- addressbookObject
- )
- self.assertEquals(
- addressbookObject.component(),
- self.addressbook1.addressbookObjectWithName("1.vcf").component()
- )
-
-
- def test_createAddressBookObjectWithName_dot(self):
- """
- Filenames starting with "." are reserved by this
- implementation, so no addressbook object names may start with
- ".".
- """
- self.assertRaises(
- ObjectResourceNameNotAllowedError,
- self.addressbook1.createAddressBookObjectWithName,
- ".foo", VComponent.fromString(vcard4_text)
- )
-
-
- @featureUnimplemented
- def test_createAddressBookObjectWithName_uidconflict(self):
- """
- Attempt to create a addressbook object with a conflicting UID
- should raise.
- """
- name = "foo.vcf"
- assert self.addressbook1.addressbookObjectWithName(name) is None
- component = VComponent.fromString(vcard1modified_text)
- self.assertRaises(
- ObjectResourceUIDAlreadyExistsError,
- self.addressbook1.createAddressBookObjectWithName,
- name, component
- )
-
-
- def test_removeAddressBookObject_delayedEffect(self):
- """
- Removing a addressbook object should not immediately remove the underlying
- file; it should only be removed upon commit() of the transaction.
- """
- self.addressbook1.removeAddressBookObjectWithName("2.vcf")
- self.failUnless(self.addressbook1._path.child("2.vcf").exists())
- self.txn.commit()
- self.failIf(self.addressbook1._path.child("2.vcf").exists())
-
-
- def test_removeAddressBookObjectWithName_dot(self):
- """
- Filenames starting with "." are reserved by this
- implementation, so no addressbook object names may start with
- ".".
- """
- name = ".foo"
- self.addressbook1._path.child(name).touch()
- self.assertRaises(
- NoSuchObjectResourceError,
- self.addressbook1.removeAddressBookObjectWithName, name
- )
-
-
- def _refresh(self):
- """
- Re-read the (committed) home1 and addressbook1 objects in a new
- transaction.
- """
- self.txn = self.addressbookStore.newTransaction()
- self.home1 = self.txn.addressbookHomeWithUID("home1")
- self.addressbook1 = self.home1.addressbookWithName("addressbook_1")
-
-
- def test_undoCreateAddressBookObject(self):
- """
- If a addressbook object is created as part of a transaction, it will be
- removed if that transaction has to be aborted.
- """
- # Make sure that the addressbook home is actually committed; rolling back
- # addressbook home creation will remove the whole directory.
- self.txn.commit()
- self._refresh()
- self.addressbook1.createAddressBookObjectWithName(
- "sample.vcf",
- VComponent.fromString(vcard4_text)
- )
- self._refresh()
- self.assertIdentical(
- self.addressbook1.addressbookObjectWithName("sample.vcf"),
- None
- )
-
-
- def doThenUndo(self):
- """
- Commit the current transaction, but add an operation that will cause it
- to fail at the end. Finally, refresh all attributes with a new
- transaction so that further oparations can be performed in a valid
- context.
- """
- def fail():
- raise RuntimeError("oops")
- self.txn.addOperation(fail, "dummy failing operation")
- self.assertRaises(RuntimeError, self.txn.commit)
- self._refresh()
-
-
- def test_undoModifyAddressBookObject(self):
- """
- If an existing addressbook object is modified as part of a transaction, it
- should be restored to its previous status if the transaction aborts.
- """
- originalComponent = self.addressbook1.addressbookObjectWithName(
- "1.vcf").component()
- self.addressbook1.addressbookObjectWithName("1.vcf").setComponent(
- VComponent.fromString(vcard1modified_text)
- )
- # Sanity check.
- self.assertEquals(
- self.addressbook1.addressbookObjectWithName("1.vcf").component(),
- VComponent.fromString(vcard1modified_text)
- )
- self.doThenUndo()
- self.assertEquals(
- self.addressbook1.addressbookObjectWithName("1.vcf").component(),
- originalComponent
- )
-
-
- def test_modifyAddressBookObjectCaches(self):
- """
- Modifying a addressbook object should cache the modified component in
- memory, to avoid unnecessary parsing round-trips.
- """
- modifiedComponent = VComponent.fromString(vcard1modified_text)
- self.addressbook1.addressbookObjectWithName("1.vcf").setComponent(
- modifiedComponent
- )
- self.assertIdentical(
- modifiedComponent,
- self.addressbook1.addressbookObjectWithName("1.vcf").component()
- )
-
-
- @featureUnimplemented
- def test_removeAddressBookObjectWithUID_absent(self):
- """
- Attempt to remove an non-existing addressbook object should raise.
- """
- self.assertRaises(
- NoSuchObjectResourceError,
- self.addressbook1.removeAddressBookObjectWithUID, "xyzzy"
- )
-
-
- @testUnimplemented
- def test_syncToken(self):
- """
- Sync token is correct.
- """
- raise NotImplementedError()
-
-
- @testUnimplemented
- def test_addressbookObjectsInTimeRange(self):
- """
- Find addressbook objects occuring in a given time range.
- """
- raise NotImplementedError()
-
-
- @testUnimplemented
- def test_addressbookObjectsSinceToken(self):
- """
- Find addressbook objects that have been modified since a given
- sync token.
- """
- raise NotImplementedError()
-
-
-
-class AddressBookObjectTest(unittest.TestCase):
- def setUp(self):
- setUpAddressBook1(self)
- self.object1 = self.addressbook1.addressbookObjectWithName("1.vcf")
-
-
- def test_init(self):
- """
- L{AddressBookObject} has instance attributes, C{_path} and C{_addressbook},
- which refer to its position in the filesystem and the addressbook in which
- it is contained, respectively.
- """
- self.failUnless(
- isinstance(self.object1._path, FilePath),
- self.object1._path
- )
- self.failUnless(
- isinstance(self.object1._addressbook, AddressBook),
- self.object1._addressbook
- )
-
-
-class FileStorageTests(unittest.TestCase, CommonTests):
- """
- File storage tests.
- """
-
- def storeUnderTest(self):
- """
- Create and return a L{AddressBookStore} for testing.
- """
- setUpAddressBookStore(self)
- return self.addressbookStore
-
-
- def test_init(self):
- """
- L{AddressBookStore} has a C{_path} attribute which refers to its
- constructor argument.
- """
- self.assertEquals(self.storeUnderTest()._path,
- self.storeRootPath)
-
-
- def test_addressbookObjectsWithDotFile(self):
- """
- Adding a dotfile to the addressbook home should not increase
- """
- self.homeUnderTest()._path.child(".foo").createDirectory()
- self.test_addressbookObjects()
-
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/test_file.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/test_file.py)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/test_file.py (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/test_file.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,451 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+File addressbook store tests.
+"""
+
+from twext.python.filepath import CachingFilePath as FilePath
+from twisted.trial import unittest
+
+from twistedcaldav.vcard import Component as VComponent
+
+from txdav.common.icommondatastore import HomeChildNameNotAllowedError
+from txdav.common.icommondatastore import ObjectResourceNameNotAllowedError
+from txdav.common.icommondatastore import ObjectResourceUIDAlreadyExistsError
+from txdav.common.icommondatastore import NoSuchHomeChildError
+from txdav.common.icommondatastore import NoSuchObjectResourceError
+
+from txdav.carddav.datastore.file import AddressBookStore, AddressBookHome
+from txdav.carddav.datastore.file import AddressBook, AddressBookObject
+
+from txdav.carddav.datastore.test.common import (
+ CommonTests, vcard4_text, vcard1modified_text, StubNotifierFactory)
+
+storePath = FilePath(__file__).parent().child("addressbook_store")
+
+def _todo(f, why):
+ f.todo = why
+ return f
+
+
+
+featureUnimplemented = lambda f: _todo(f, "Feature unimplemented")
+testUnimplemented = lambda f: _todo(f, "Test unimplemented")
+todo = lambda why: lambda f: _todo(f, why)
+
+
+def setUpAddressBookStore(test):
+ test.root = FilePath(test.mktemp())
+ test.root.createDirectory()
+
+ storeRootPath = test.storeRootPath = test.root.child("store")
+ addressbookPath = storeRootPath.child("addressbooks").child("__uids__")
+ addressbookPath.parent().makedirs()
+ storePath.copyTo(addressbookPath)
+
+ test.notifierFactory = StubNotifierFactory()
+ test.addressbookStore = AddressBookStore(storeRootPath, test.notifierFactory)
+ test.txn = test.addressbookStore.newTransaction()
+ assert test.addressbookStore is not None, "No addressbook store?"
+
+
+
+def setUpHome1(test):
+ setUpAddressBookStore(test)
+ test.home1 = test.txn.addressbookHomeWithUID("home1")
+ assert test.home1 is not None, "No addressbook home?"
+
+
+
+def setUpAddressBook1(test):
+ setUpHome1(test)
+ test.addressbook1 = test.home1.addressbookWithName("addressbook_1")
+ assert test.addressbook1 is not None, "No addressbook?"
+
+
+
+class AddressBookStoreTest(unittest.TestCase):
+ """
+ Test cases for L{AddressBookStore}.
+ """
+
+ def setUp(self):
+ setUpAddressBookStore(self)
+
+
+ def test_addressbookHomeWithUID_dot(self):
+ """
+ Filenames starting with "." are reserved by this
+ implementation, so no UIDs may start with ".".
+ """
+ self.assertEquals(
+ self.addressbookStore.newTransaction().addressbookHomeWithUID("xyzzy"),
+ None
+ )
+
+
+
+class AddressBookHomeTest(unittest.TestCase):
+
+ def setUp(self):
+ setUpHome1(self)
+
+
+ def test_init(self):
+ """
+ L{AddressBookHome} has C{_path} and L{_addressbookStore} attributes,
+ indicating its location on disk and parent store, respectively.
+ """
+ self.failUnless(
+ isinstance(self.home1._path, FilePath),
+ self.home1._path
+ )
+ self.assertEquals(
+ self.home1._addressbookStore,
+ self.addressbookStore
+ )
+
+
+ def test_addressbookWithName_dot(self):
+ """
+ Filenames starting with "." are reserved by this
+ implementation, so no addressbook names may start with ".".
+ """
+ name = ".foo"
+ self.home1._path.child(name).createDirectory()
+ self.assertEquals(self.home1.addressbookWithName(name), None)
+
+
+ def test_createAddressBookWithName_dot(self):
+ """
+ Filenames starting with "." are reserved by this
+ implementation, so no addressbook names may start with ".".
+ """
+ self.assertRaises(
+ HomeChildNameNotAllowedError,
+ self.home1.createAddressBookWithName, ".foo"
+ )
+
+
+ def test_removeAddressBookWithName_dot(self):
+ """
+ Filenames starting with "." are reserved by this
+ implementation, so no addressbook names may start with ".".
+ """
+ name = ".foo"
+ self.home1._path.child(name).createDirectory()
+ self.assertRaises(
+ NoSuchHomeChildError,
+ self.home1.removeAddressBookWithName, name
+ )
+
+
+
+class AddressBookTest(unittest.TestCase):
+
+ def setUp(self):
+ setUpAddressBook1(self)
+
+
+ def test_init(self):
+ """
+ L{AddressBook.__init__} sets private attributes to reflect its constructor
+ arguments.
+ """
+ self.failUnless(
+ isinstance(self.addressbook1._path, FilePath),
+ self.addressbook1
+ )
+ self.failUnless(
+ isinstance(self.addressbook1._addressbookHome, AddressBookHome),
+ self.addressbook1._addressbookHome
+ )
+
+
+ def test_useIndexImmediately(self):
+ """
+ L{AddressBook._index} is usable in the same transaction it is created, with
+ a temporary filename.
+ """
+ self.home1.createAddressBookWithName("addressbook2")
+ addressbook = self.home1.addressbookWithName("addressbook2")
+ index = addressbook._index
+ self.assertEquals(set(index.addressbookObjects()),
+ set(addressbook.addressbookObjects()))
+ self.txn.commit()
+ self.txn = self.addressbookStore.newTransaction()
+ self.home1 = self.txn.addressbookHomeWithUID("home1")
+ addressbook = self.home1.addressbookWithName("addressbook2")
+ # FIXME: we should be curating our own index here, but in order to fix
+ # that the code in the old implicit scheduler needs to change. This
+ # test would be more effective if there were actually some objects in
+ # this list.
+ index = addressbook._index
+ self.assertEquals(set(index.addressbookObjects()),
+ set(addressbook.addressbookObjects()))
+
+
+ def test_addressbookObjectWithName_dot(self):
+ """
+ Filenames starting with "." are reserved by this
+ implementation, so no addressbook object names may start with
+ ".".
+ """
+ name = ".foo.vcf"
+ self.home1._path.child(name).touch()
+ self.assertEquals(self.addressbook1.addressbookObjectWithName(name), None)
+
+
+ @featureUnimplemented
+ def test_addressbookObjectWithUID_exists(self):
+ """
+ Find existing addressbook object by name.
+ """
+ addressbookObject = self.addressbook1.addressbookObjectWithUID("1")
+ self.failUnless(
+ isinstance(addressbookObject, AddressBookObject),
+ addressbookObject
+ )
+ self.assertEquals(
+ addressbookObject.component(),
+ self.addressbook1.addressbookObjectWithName("1.vcf").component()
+ )
+
+
+ def test_createAddressBookObjectWithName_dot(self):
+ """
+ Filenames starting with "." are reserved by this
+ implementation, so no addressbook object names may start with
+ ".".
+ """
+ self.assertRaises(
+ ObjectResourceNameNotAllowedError,
+ self.addressbook1.createAddressBookObjectWithName,
+ ".foo", VComponent.fromString(vcard4_text)
+ )
+
+
+ @featureUnimplemented
+ def test_createAddressBookObjectWithName_uidconflict(self):
+ """
+ Attempt to create a addressbook object with a conflicting UID
+ should raise.
+ """
+ name = "foo.vcf"
+ assert self.addressbook1.addressbookObjectWithName(name) is None
+ component = VComponent.fromString(vcard1modified_text)
+ self.assertRaises(
+ ObjectResourceUIDAlreadyExistsError,
+ self.addressbook1.createAddressBookObjectWithName,
+ name, component
+ )
+
+
+ def test_removeAddressBookObject_delayedEffect(self):
+ """
+ Removing a addressbook object should not immediately remove the underlying
+ file; it should only be removed upon commit() of the transaction.
+ """
+ self.addressbook1.removeAddressBookObjectWithName("2.vcf")
+ self.failUnless(self.addressbook1._path.child("2.vcf").exists())
+ self.txn.commit()
+ self.failIf(self.addressbook1._path.child("2.vcf").exists())
+
+
+ def test_removeAddressBookObjectWithName_dot(self):
+ """
+ Filenames starting with "." are reserved by this
+ implementation, so no addressbook object names may start with
+ ".".
+ """
+ name = ".foo"
+ self.addressbook1._path.child(name).touch()
+ self.assertRaises(
+ NoSuchObjectResourceError,
+ self.addressbook1.removeAddressBookObjectWithName, name
+ )
+
+
+ def _refresh(self):
+ """
+ Re-read the (committed) home1 and addressbook1 objects in a new
+ transaction.
+ """
+ self.txn = self.addressbookStore.newTransaction()
+ self.home1 = self.txn.addressbookHomeWithUID("home1")
+ self.addressbook1 = self.home1.addressbookWithName("addressbook_1")
+
+
+ def test_undoCreateAddressBookObject(self):
+ """
+ If a addressbook object is created as part of a transaction, it will be
+ removed if that transaction has to be aborted.
+ """
+ # Make sure that the addressbook home is actually committed; rolling back
+ # addressbook home creation will remove the whole directory.
+ self.txn.commit()
+ self._refresh()
+ self.addressbook1.createAddressBookObjectWithName(
+ "sample.vcf",
+ VComponent.fromString(vcard4_text)
+ )
+ self._refresh()
+ self.assertIdentical(
+ self.addressbook1.addressbookObjectWithName("sample.vcf"),
+ None
+ )
+
+
+ def doThenUndo(self):
+ """
+ Commit the current transaction, but add an operation that will cause it
+ to fail at the end. Finally, refresh all attributes with a new
+ transaction so that further oparations can be performed in a valid
+ context.
+ """
+ def fail():
+ raise RuntimeError("oops")
+ self.txn.addOperation(fail, "dummy failing operation")
+ self.assertRaises(RuntimeError, self.txn.commit)
+ self._refresh()
+
+
+ def test_undoModifyAddressBookObject(self):
+ """
+ If an existing addressbook object is modified as part of a transaction, it
+ should be restored to its previous status if the transaction aborts.
+ """
+ originalComponent = self.addressbook1.addressbookObjectWithName(
+ "1.vcf").component()
+ self.addressbook1.addressbookObjectWithName("1.vcf").setComponent(
+ VComponent.fromString(vcard1modified_text)
+ )
+ # Sanity check.
+ self.assertEquals(
+ self.addressbook1.addressbookObjectWithName("1.vcf").component(),
+ VComponent.fromString(vcard1modified_text)
+ )
+ self.doThenUndo()
+ self.assertEquals(
+ self.addressbook1.addressbookObjectWithName("1.vcf").component(),
+ originalComponent
+ )
+
+
+ def test_modifyAddressBookObjectCaches(self):
+ """
+ Modifying a addressbook object should cache the modified component in
+ memory, to avoid unnecessary parsing round-trips.
+ """
+ modifiedComponent = VComponent.fromString(vcard1modified_text)
+ self.addressbook1.addressbookObjectWithName("1.vcf").setComponent(
+ modifiedComponent
+ )
+ self.assertIdentical(
+ modifiedComponent,
+ self.addressbook1.addressbookObjectWithName("1.vcf").component()
+ )
+
+
+ @featureUnimplemented
+ def test_removeAddressBookObjectWithUID_absent(self):
+ """
+ Attempt to remove an non-existing addressbook object should raise.
+ """
+ self.assertRaises(
+ NoSuchObjectResourceError,
+ self.addressbook1.removeAddressBookObjectWithUID, "xyzzy"
+ )
+
+
+ @testUnimplemented
+ def test_syncToken(self):
+ """
+ Sync token is correct.
+ """
+ raise NotImplementedError()
+
+
+ @testUnimplemented
+ def test_addressbookObjectsInTimeRange(self):
+ """
+ Find addressbook objects occuring in a given time range.
+ """
+ raise NotImplementedError()
+
+
+ @testUnimplemented
+ def test_addressbookObjectsSinceToken(self):
+ """
+ Find addressbook objects that have been modified since a given
+ sync token.
+ """
+ raise NotImplementedError()
+
+
+
+class AddressBookObjectTest(unittest.TestCase):
+ def setUp(self):
+ setUpAddressBook1(self)
+ self.object1 = self.addressbook1.addressbookObjectWithName("1.vcf")
+
+
+ def test_init(self):
+ """
+ L{AddressBookObject} has instance attributes, C{_path} and C{_addressbook},
+ which refer to its position in the filesystem and the addressbook in which
+ it is contained, respectively.
+ """
+ self.failUnless(
+ isinstance(self.object1._path, FilePath),
+ self.object1._path
+ )
+ self.failUnless(
+ isinstance(self.object1._addressbook, AddressBook),
+ self.object1._addressbook
+ )
+
+
+class FileStorageTests(unittest.TestCase, CommonTests):
+ """
+ File storage tests.
+ """
+
+ def storeUnderTest(self):
+ """
+ Create and return a L{AddressBookStore} for testing.
+ """
+ setUpAddressBookStore(self)
+ return self.addressbookStore
+
+
+ def test_init(self):
+ """
+ L{AddressBookStore} has a C{_path} attribute which refers to its
+ constructor argument.
+ """
+ self.assertEquals(self.storeUnderTest()._path,
+ self.storeRootPath)
+
+
+ def test_addressbookObjectsWithDotFile(self):
+ """
+ Adding a dotfile to the addressbook home should not increase
+ """
+ self.homeUnderTest()._path.child(".foo").createDirectory()
+ self.test_addressbookObjects()
+
Deleted: CalendarServer/trunk/txdav/carddav/datastore/test/test_sql.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/test_sql.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/test_sql.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,75 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Tests for txdav.caldav.datastore.postgres, mostly based on
-L{txdav.caldav.datastore.test.common}.
-"""
-
-from txdav.carddav.datastore.test.common import CommonTests as AddressBookCommonTests
-
-from txdav.common.datastore.test.util import SQLStoreBuilder
-
-from twisted.trial import unittest
-from twisted.internet.defer import inlineCallbacks
-from twistedcaldav.vcard import Component as VCard
-
-
-theStoreBuilder = SQLStoreBuilder()
-buildStore = theStoreBuilder.buildStore
-
-class AddressBookSQLStorageTests(AddressBookCommonTests, unittest.TestCase):
- """
- AddressBook SQL storage tests.
- """
-
- @inlineCallbacks
- def setUp(self):
- super(AddressBookSQLStorageTests, self).setUp()
- self.addressbookStore = yield buildStore(self, self.notifierFactory)
- self.populate()
-
- def populate(self):
- populateTxn = self.addressbookStore.newTransaction()
- for homeUID in self.requirements:
- addressbooks = self.requirements[homeUID]
- if addressbooks is not None:
- home = populateTxn.addressbookHomeWithUID(homeUID, True)
- # We don't want the default addressbook to appear unless it's
- # explicitly listed.
- home.removeAddressBookWithName("addressbook")
- for addressbookName in addressbooks:
- addressbookObjNames = addressbooks[addressbookName]
- if addressbookObjNames is not None:
- home.createAddressBookWithName(addressbookName)
- addressbook = home.addressbookWithName(addressbookName)
- for objectName in addressbookObjNames:
- objData = addressbookObjNames[objectName]
- addressbook.createAddressBookObjectWithName(
- objectName, VCard.fromString(objData)
- )
-
- populateTxn.commit()
- self.notifierFactory.reset()
-
-
-
- def storeUnderTest(self):
- """
- Create and return a L{AddressBookStore} for testing.
- """
- return self.addressbookStore
-
Copied: CalendarServer/trunk/txdav/carddav/datastore/test/test_sql.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/test/test_sql.py)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/test/test_sql.py (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/test/test_sql.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,75 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Tests for txdav.caldav.datastore.postgres, mostly based on
+L{txdav.caldav.datastore.test.common}.
+"""
+
+from txdav.carddav.datastore.test.common import CommonTests as AddressBookCommonTests
+
+from txdav.common.datastore.test.util import SQLStoreBuilder
+
+from twisted.trial import unittest
+from twisted.internet.defer import inlineCallbacks
+from twistedcaldav.vcard import Component as VCard
+
+
+theStoreBuilder = SQLStoreBuilder()
+buildStore = theStoreBuilder.buildStore
+
+class AddressBookSQLStorageTests(AddressBookCommonTests, unittest.TestCase):
+ """
+ AddressBook SQL storage tests.
+ """
+
+ @inlineCallbacks
+ def setUp(self):
+ super(AddressBookSQLStorageTests, self).setUp()
+ self.addressbookStore = yield buildStore(self, self.notifierFactory)
+ self.populate()
+
+ def populate(self):
+ populateTxn = self.addressbookStore.newTransaction()
+ for homeUID in self.requirements:
+ addressbooks = self.requirements[homeUID]
+ if addressbooks is not None:
+ home = populateTxn.addressbookHomeWithUID(homeUID, True)
+ # We don't want the default addressbook to appear unless it's
+ # explicitly listed.
+ home.removeAddressBookWithName("addressbook")
+ for addressbookName in addressbooks:
+ addressbookObjNames = addressbooks[addressbookName]
+ if addressbookObjNames is not None:
+ home.createAddressBookWithName(addressbookName)
+ addressbook = home.addressbookWithName(addressbookName)
+ for objectName in addressbookObjNames:
+ objData = addressbookObjNames[objectName]
+ addressbook.createAddressBookObjectWithName(
+ objectName, VCard.fromString(objData)
+ )
+
+ populateTxn.commit()
+ self.notifierFactory.reset()
+
+
+
+ def storeUnderTest(self):
+ """
+ Create and return a L{AddressBookStore} for testing.
+ """
+ return self.addressbookStore
+
Deleted: CalendarServer/trunk/txdav/carddav/datastore/util.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/util.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/datastore/util.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,58 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Utility logic common to multiple backend implementations.
-"""
-
-from twistedcaldav.vcard import Component as VCard
-from twistedcaldav.vcard import InvalidVCardDataError
-
-from txdav.common.icommondatastore import InvalidObjectResourceError,\
- NoSuchObjectResourceError
-
-def validateAddressBookComponent(addressbookObject, vcard, component, inserting):
- """
- Validate an addressbook component for a particular addressbook.
-
- @param addressbookObject: The addressbook object whose component will be replaced.
- @type addressbookObject: L{IAddressBookObject}
-
- @param addressbook: The addressbook which the L{IAddressBookObject} is present in.
- @type addressbook: L{IAddressBook}
-
- @param component: The VComponent to be validated.
- @type component: L{VComponent}
- """
-
- if not isinstance(component, VCard):
- raise TypeError(type(component))
-
- try:
- if not inserting and component.resourceUID() != addressbookObject.uid():
- raise InvalidObjectResourceError(
- "UID may not change (%s != %s)" % (
- component.resourceUID(), addressbookObject.uid()
- )
- )
- except NoSuchObjectResourceError:
- pass
-
- try:
- component.validForCardDAV()
- except InvalidVCardDataError, e:
- raise InvalidObjectResourceError(e)
-
\ No newline at end of file
Copied: CalendarServer/trunk/txdav/carddav/datastore/util.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/datastore/util.py)
===================================================================
--- CalendarServer/trunk/txdav/carddav/datastore/util.py (rev 0)
+++ CalendarServer/trunk/txdav/carddav/datastore/util.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,58 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Utility logic common to multiple backend implementations.
+"""
+
+from twistedcaldav.vcard import Component as VCard
+from twistedcaldav.vcard import InvalidVCardDataError
+
+from txdav.common.icommondatastore import InvalidObjectResourceError,\
+ NoSuchObjectResourceError
+
+def validateAddressBookComponent(addressbookObject, vcard, component, inserting):
+ """
+ Validate an addressbook component for a particular addressbook.
+
+ @param addressbookObject: The addressbook object whose component will be replaced.
+ @type addressbookObject: L{IAddressBookObject}
+
+ @param addressbook: The addressbook which the L{IAddressBookObject} is present in.
+ @type addressbook: L{IAddressBook}
+
+ @param component: The VComponent to be validated.
+ @type component: L{VComponent}
+ """
+
+ if not isinstance(component, VCard):
+ raise TypeError(type(component))
+
+ try:
+ if not inserting and component.resourceUID() != addressbookObject.uid():
+ raise InvalidObjectResourceError(
+ "UID may not change (%s != %s)" % (
+ component.resourceUID(), addressbookObject.uid()
+ )
+ )
+ except NoSuchObjectResourceError:
+ pass
+
+ try:
+ component.validForCardDAV()
+ except InvalidVCardDataError, e:
+ raise InvalidObjectResourceError(e)
+
\ No newline at end of file
Deleted: CalendarServer/trunk/txdav/carddav/iaddressbookstore.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/iaddressbookstore.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/iaddressbookstore.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,265 +0,0 @@
-# -*- test-case-name: txdav.carddav.datastore,txdav.carddav.datastore.test.test_sql.AddressBookSQLStorageTests -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Address book store interfaces
-"""
-
-from txdav.common.icommondatastore import ICommonTransaction,\
- IShareableCollection
-from txdav.idav import INotifier
-from txdav.idav import IDataStoreResource
-
-__all__ = [
- # Classes
- "IAddressBookTransaction",
- "IAddressBookHome",
- "IAddressBook",
- "IAddressBookObject",
-]
-
-class IAddressBookTransaction(ICommonTransaction):
- """
- Transaction interface that addressbook stores must provide.
- """
-
- def addressbookHomeWithUID(uid, create=False):
- """
- Retrieve the addressbook home for the principal with the given C{uid}.
-
- If C{create} is C{True}, create the addressbook home if it doesn't
- already exist.
-
- @return: an L{IAddressBookHome} or C{None} if no such addressbook
- home exists.
- """
-
-
-
-#
-# Interfaces
-#
-
-class IAddressBookHome(INotifier, IDataStoreResource):
- """
- AddressBook home
-
- An addressbook home belongs to a specific principal and contains the
- addressbooks which that principal has direct access to. This
- includes both addressbooks owned by the principal as well as
- addressbooks that have been shared with and accepts by the principal.
- """
-
- def uid():
- """
- Retrieve the unique identifier for this addressbook home.
-
- @return: a string.
- """
-
-
- def addressbooks():
- """
- Retrieve addressbooks contained in this addressbook home.
-
- @return: an iterable of L{IAddressBook}s.
- """
-
- def addressbookWithName(name):
- """
- Retrieve the addressbook with the given C{name} contained in this
- addressbook home.
-
- @param name: a string.
- @return: an L{IAddressBook} or C{None} if no such addressbook
- exists.
- """
-
- def createAddressBookWithName(name):
- """
- Create an addressbook with the given C{name} in this addressbook
- home.
-
- @param name: a string.
- @raise AddressBookAlreadyExistsError: if an addressbook with the
- given C{name} already exists.
- """
-
- def removeAddressBookWithName(name):
- """
- Remove the addressbook with the given C{name} from this addressbook
- home. If this addressbook home owns the addressbook, also remove
- the addressbook from all addressbook homes.
-
- @param name: a string.
- @raise NoSuchAddressBookObjectError: if no such addressbook exists.
- """
-
-
-class IAddressBook(INotifier, IShareableCollection, IDataStoreResource):
- """
- AddressBook
-
- An addressbook is a container for addressbook objects (contacts),
- An addressbook belongs to a specific principal but may be
- shared with other principals, granting them read-only or
- read/write access.
- """
-
- def rename(name):
- """
- Change the name of this addressbook.
- """
-
- def ownerAddressBookHome():
- """
- Retrieve the addressbook home for the owner of this addressbook.
- AddressBooks may be shared from one (the owner's) addressbook home
- to other (the sharee's) addressbook homes.
-
- @return: an L{IAddressBookHome}.
- """
-
- def addressbookObjects():
- """
- Retrieve the addressbook objects contained in this addressbook.
-
- @return: an iterable of L{IAddressBookObject}s.
- """
-
- def addressbookObjectWithName(name):
- """
- Retrieve the addressbook object with the given C{name} contained
- in this addressbook.
-
- @param name: a string.
- @return: an L{IAddressBookObject} or C{None} if no such addressbook
- object exists.
- """
-
- def addressbookObjectWithUID(uid):
- """
- Retrieve the addressbook object with the given C{uid} contained
- in this addressbook.
-
- @param uid: a string.
- @return: an L{IAddressBookObject} or C{None} if no such addressbook
- object exists.
- """
-
- def createAddressBookObjectWithName(name, component):
- """
- Create an addressbook component with the given C{name} in this
- addressbook from the given C{component}.
-
- @param name: a string.
- @param component: a C{VCARD} L{Component}
- @raise AddressBookObjectNameAlreadyExistsError: if an addressbook
- object with the given C{name} already exists.
- @raise AddressBookObjectUIDAlreadyExistsError: if an addressbook
- object with the same UID as the given C{component} already
- exists.
- @raise InvalidAddressBookComponentError: if the given
- C{component} is not a valid C{VCARD} L{VComponent} for
- an addressbook object.
- """
-
- def removeAddressBookObjectWithName(name):
- """
- Remove the addressbook object with the given C{name} from this
- addressbook.
-
- @param name: a string.
- @raise NoSuchAddressBookObjectError: if no such addressbook object
- exists.
- """
-
- def removeAddressBookObjectWithUID(uid):
- """
- Remove the addressbook object with the given C{uid} from this
- addressbook.
-
- @param uid: a string.
- @raise NoSuchAddressBookObjectError: if the addressbook object does
- not exist.
- """
-
- def syncToken():
- """
- Retrieve the current sync token for this addressbook.
-
- @return: a string containing a sync token.
- """
-
- def addressbookObjectsSinceToken(token):
- """
- Retrieve all addressbook objects in this addressbook that have
- changed since the given C{token} was last valid.
-
- @param token: a sync token.
- @return: a 3-tuple containing an iterable of
- L{IAddressBookObject}s that have changed, an iterable of uids
- that have been removed, and the current sync token.
- """
-
-
-class IAddressBookObject(IDataStoreResource):
- """
- AddressBook object
-
- An addressbook object describes a contact (vCard).
- """
-
- def addressbook():
- """
- @return: The address book which this address book object is a part of.
- @rtype: L{IAddressBook}
- """
-
-
- def setComponent(component):
- """
- Rewrite this addressbook object to match the given C{component}.
- C{component} must have the same UID as this addressbook object.
-
- @param component: a C{VCARD} L{VComponent}.
- @raise InvalidAddressBookComponentError: if the given
- C{component} is not a valid C{VCARD} L{VComponent} for
- an addressbook object.
- """
-
- def component():
- """
- Retrieve the addressbook component for this addressbook object.
-
- @return: a C{VCARD} L{VComponent}.
- """
-
- def vCardText():
- """
- Retrieve the vCard text data for this addressbook object.
-
- @return: a string containing vCard data for a single
- addressbook object.
- """
-
- def uid():
- """
- Retrieve the UID for this addressbook object.
-
- @return: a string containing a UID.
- """
Copied: CalendarServer/trunk/txdav/carddav/iaddressbookstore.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/iaddressbookstore.py)
===================================================================
--- CalendarServer/trunk/txdav/carddav/iaddressbookstore.py (rev 0)
+++ CalendarServer/trunk/txdav/carddav/iaddressbookstore.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,265 @@
+# -*- test-case-name: txdav.carddav.datastore,txdav.carddav.datastore.test.test_sql.AddressBookSQLStorageTests -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Address book store interfaces
+"""
+
+from txdav.common.icommondatastore import ICommonTransaction,\
+ IShareableCollection
+from txdav.idav import INotifier
+from txdav.idav import IDataStoreResource
+
+__all__ = [
+ # Classes
+ "IAddressBookTransaction",
+ "IAddressBookHome",
+ "IAddressBook",
+ "IAddressBookObject",
+]
+
+class IAddressBookTransaction(ICommonTransaction):
+ """
+ Transaction interface that addressbook stores must provide.
+ """
+
+ def addressbookHomeWithUID(uid, create=False):
+ """
+ Retrieve the addressbook home for the principal with the given C{uid}.
+
+ If C{create} is C{True}, create the addressbook home if it doesn't
+ already exist.
+
+ @return: an L{IAddressBookHome} or C{None} if no such addressbook
+ home exists.
+ """
+
+
+
+#
+# Interfaces
+#
+
+class IAddressBookHome(INotifier, IDataStoreResource):
+ """
+ AddressBook home
+
+ An addressbook home belongs to a specific principal and contains the
+ addressbooks which that principal has direct access to. This
+ includes both addressbooks owned by the principal as well as
+ addressbooks that have been shared with and accepts by the principal.
+ """
+
+ def uid():
+ """
+ Retrieve the unique identifier for this addressbook home.
+
+ @return: a string.
+ """
+
+
+ def addressbooks():
+ """
+ Retrieve addressbooks contained in this addressbook home.
+
+ @return: an iterable of L{IAddressBook}s.
+ """
+
+ def addressbookWithName(name):
+ """
+ Retrieve the addressbook with the given C{name} contained in this
+ addressbook home.
+
+ @param name: a string.
+ @return: an L{IAddressBook} or C{None} if no such addressbook
+ exists.
+ """
+
+ def createAddressBookWithName(name):
+ """
+ Create an addressbook with the given C{name} in this addressbook
+ home.
+
+ @param name: a string.
+ @raise AddressBookAlreadyExistsError: if an addressbook with the
+ given C{name} already exists.
+ """
+
+ def removeAddressBookWithName(name):
+ """
+ Remove the addressbook with the given C{name} from this addressbook
+ home. If this addressbook home owns the addressbook, also remove
+ the addressbook from all addressbook homes.
+
+ @param name: a string.
+ @raise NoSuchAddressBookObjectError: if no such addressbook exists.
+ """
+
+
+class IAddressBook(INotifier, IShareableCollection, IDataStoreResource):
+ """
+ AddressBook
+
+ An addressbook is a container for addressbook objects (contacts),
+ An addressbook belongs to a specific principal but may be
+ shared with other principals, granting them read-only or
+ read/write access.
+ """
+
+ def rename(name):
+ """
+ Change the name of this addressbook.
+ """
+
+ def ownerAddressBookHome():
+ """
+ Retrieve the addressbook home for the owner of this addressbook.
+ AddressBooks may be shared from one (the owner's) addressbook home
+ to other (the sharee's) addressbook homes.
+
+ @return: an L{IAddressBookHome}.
+ """
+
+ def addressbookObjects():
+ """
+ Retrieve the addressbook objects contained in this addressbook.
+
+ @return: an iterable of L{IAddressBookObject}s.
+ """
+
+ def addressbookObjectWithName(name):
+ """
+ Retrieve the addressbook object with the given C{name} contained
+ in this addressbook.
+
+ @param name: a string.
+ @return: an L{IAddressBookObject} or C{None} if no such addressbook
+ object exists.
+ """
+
+ def addressbookObjectWithUID(uid):
+ """
+ Retrieve the addressbook object with the given C{uid} contained
+ in this addressbook.
+
+ @param uid: a string.
+ @return: an L{IAddressBookObject} or C{None} if no such addressbook
+ object exists.
+ """
+
+ def createAddressBookObjectWithName(name, component):
+ """
+ Create an addressbook component with the given C{name} in this
+ addressbook from the given C{component}.
+
+ @param name: a string.
+ @param component: a C{VCARD} L{Component}
+ @raise AddressBookObjectNameAlreadyExistsError: if an addressbook
+ object with the given C{name} already exists.
+ @raise AddressBookObjectUIDAlreadyExistsError: if an addressbook
+ object with the same UID as the given C{component} already
+ exists.
+ @raise InvalidAddressBookComponentError: if the given
+ C{component} is not a valid C{VCARD} L{VComponent} for
+ an addressbook object.
+ """
+
+ def removeAddressBookObjectWithName(name):
+ """
+ Remove the addressbook object with the given C{name} from this
+ addressbook.
+
+ @param name: a string.
+ @raise NoSuchAddressBookObjectError: if no such addressbook object
+ exists.
+ """
+
+ def removeAddressBookObjectWithUID(uid):
+ """
+ Remove the addressbook object with the given C{uid} from this
+ addressbook.
+
+ @param uid: a string.
+ @raise NoSuchAddressBookObjectError: if the addressbook object does
+ not exist.
+ """
+
+ def syncToken():
+ """
+ Retrieve the current sync token for this addressbook.
+
+ @return: a string containing a sync token.
+ """
+
+ def addressbookObjectsSinceToken(token):
+ """
+ Retrieve all addressbook objects in this addressbook that have
+ changed since the given C{token} was last valid.
+
+ @param token: a sync token.
+ @return: a 3-tuple containing an iterable of
+ L{IAddressBookObject}s that have changed, an iterable of uids
+ that have been removed, and the current sync token.
+ """
+
+
+class IAddressBookObject(IDataStoreResource):
+ """
+ AddressBook object
+
+ An addressbook object describes a contact (vCard).
+ """
+
+ def addressbook():
+ """
+ @return: The address book which this address book object is a part of.
+ @rtype: L{IAddressBook}
+ """
+
+
+ def setComponent(component):
+ """
+ Rewrite this addressbook object to match the given C{component}.
+ C{component} must have the same UID as this addressbook object.
+
+ @param component: a C{VCARD} L{VComponent}.
+ @raise InvalidAddressBookComponentError: if the given
+ C{component} is not a valid C{VCARD} L{VComponent} for
+ an addressbook object.
+ """
+
+ def component():
+ """
+ Retrieve the addressbook component for this addressbook object.
+
+ @return: a C{VCARD} L{VComponent}.
+ """
+
+ def vCardText():
+ """
+ Retrieve the vCard text data for this addressbook object.
+
+ @return: a string containing vCard data for a single
+ addressbook object.
+ """
+
+ def uid():
+ """
+ Retrieve the UID for this addressbook object.
+
+ @return: a string containing a UID.
+ """
Deleted: CalendarServer/trunk/txdav/carddav/resource.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/carddav/resource.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/carddav/resource.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,83 +0,0 @@
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-CardDAV resources.
-"""
-
-__all__ = [
- "CardDAVResource",
- "AddressBookHomeResource",
- "AddressBookCollectionResource",
- "AddressBookObjectResource",
-]
-
-from twext.python.log import LoggingMixIn
-from twext.web2.dav.element.base import dav_namespace
-
-from twistedcaldav.carddavxml import carddav_namespace
-from twistedcaldav.config import config
-from twistedcaldav.extensions import DAVResource
-
-class CardDAVResource(DAVResource, LoggingMixIn):
- """
- CardDAV resource.
- """
- def davComplianceClasses(self):
- return (
- tuple(super(CardDAVResource, self).davComplianceClasses())
- + config.CardDAVComplianceClasses
- )
-
-
-class AddressBookHomeResource(CardDAVResource):
- """
- AddressBook home resource.
-
- This resource is backed by an L{IAddressBookHome} implementation.
- """
-
-
-class AddressBookCollectionResource(CardDAVResource):
- """
- AddressBook collection resource.
-
- This resource is backed by an L{IAddressBook} implementation.
- """
- #
- # HTTP
- #
-
- #
- # WebDAV
- #
-
- def liveProperties(self):
-
- return super(AddressBookCollectionResource, self).liveProperties() + (
- (dav_namespace, "owner"),
- (carddav_namespace, "supported-addressbook-data"),
- )
-
-
-
-
-class AddressBookObjectResource(CardDAVResource):
- """
- AddressBook object resource.
-
- This resource is backed by an L{IAddressBookObject} implementation.
- """
Copied: CalendarServer/trunk/txdav/carddav/resource.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/carddav/resource.py)
===================================================================
--- CalendarServer/trunk/txdav/carddav/resource.py (rev 0)
+++ CalendarServer/trunk/txdav/carddav/resource.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,83 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+CardDAV resources.
+"""
+
+__all__ = [
+ "CardDAVResource",
+ "AddressBookHomeResource",
+ "AddressBookCollectionResource",
+ "AddressBookObjectResource",
+]
+
+from twext.python.log import LoggingMixIn
+from twext.web2.dav.element.base import dav_namespace
+
+from twistedcaldav.carddavxml import carddav_namespace
+from twistedcaldav.config import config
+from twistedcaldav.extensions import DAVResource
+
+class CardDAVResource(DAVResource, LoggingMixIn):
+ """
+ CardDAV resource.
+ """
+ def davComplianceClasses(self):
+ return (
+ tuple(super(CardDAVResource, self).davComplianceClasses())
+ + config.CardDAVComplianceClasses
+ )
+
+
+class AddressBookHomeResource(CardDAVResource):
+ """
+ AddressBook home resource.
+
+ This resource is backed by an L{IAddressBookHome} implementation.
+ """
+
+
+class AddressBookCollectionResource(CardDAVResource):
+ """
+ AddressBook collection resource.
+
+ This resource is backed by an L{IAddressBook} implementation.
+ """
+ #
+ # HTTP
+ #
+
+ #
+ # WebDAV
+ #
+
+ def liveProperties(self):
+
+ return super(AddressBookCollectionResource, self).liveProperties() + (
+ (dav_namespace, "owner"),
+ (carddav_namespace, "supported-addressbook-data"),
+ )
+
+
+
+
+class AddressBookObjectResource(CardDAVResource):
+ """
+ AddressBook object resource.
+
+ This resource is backed by an L{IAddressBookObject} implementation.
+ """
Modified: CalendarServer/trunk/txdav/common/datastore/file.py
===================================================================
--- CalendarServer/trunk/txdav/common/datastore/file.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/common/datastore/file.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,4 +1,4 @@
-# -*- test-case-name: txcaldav.calendarstore.test.test_file -*-
+# -*- test-case-name: txdav.caldav.datastore.test.test_file -*-
##
# Copyright (c) 2010 Apple Inc. All rights reserved.
#
@@ -38,11 +38,12 @@
ObjectResourceNameAlreadyExistsError, NoSuchObjectResourceError
from txdav.common.inotifications import INotificationCollection, \
INotificationObject
-from txdav.datastore.file import DataStoreTransaction, DataStore, writeOperation, \
- hidden, isValidName, cached, FileMetaDataMixin
+from txdav.base.datastore.file import DataStoreTransaction, DataStore, writeOperation, \
+ hidden, isValidName, FileMetaDataMixin
+from txdav.base.datastore.util import cached
from txdav.idav import IDataStore
-from txdav.propertystore.base import PropertyName
-from txdav.propertystore.xattr import PropertyStore
+from txdav.base.propertystore.base import PropertyName
+from txdav.base.propertystore.xattr import PropertyStore
from errno import EEXIST, ENOENT
from zope.interface import implements, directlyProvides
@@ -109,10 +110,10 @@
@type dataStore: L{CommonDataStore}
"""
- from txcaldav.icalendarstore import ICalendarTransaction
- from txcarddav.iaddressbookstore import IAddressBookTransaction
- from txcaldav.calendarstore.file import CalendarHome
- from txcarddav.addressbookstore.file import AddressBookHome
+ from txdav.caldav.icalendarstore import ICalendarTransaction
+ from txdav.carddav.iaddressbookstore import IAddressBookTransaction
+ from txdav.caldav.datastore.file import CalendarHome
+ from txdav.carddav.datastore.file import AddressBookHome
super(CommonStoreTransaction, self).__init__(dataStore, name)
self._homes = {}
@@ -200,7 +201,7 @@
notifier)
self._homes[storeType][(uid, self)] = home
if creating:
- home.created()
+ home.createdHome()
# Create notification collection
if storeType == ECALENDARTYPE:
@@ -635,7 +636,7 @@
raise ObjectResourceNameAlreadyExistsError(name)
objectResource = self._objectResourceClass(name, self)
- objectResource.setComponent(component)
+ objectResource.setComponent(component, inserting=True)
self._cachedObjectResources[name] = objectResource
# Note: setComponent triggers a notification, so we don't need to
@@ -704,8 +705,8 @@
@param props: the L{PropertyStore} from C{properties()}.
"""
+ pass
-
def _doValidate(self, component):
raise NotImplementedError
@@ -742,7 +743,7 @@
@writeOperation
- def setComponent(self, component):
+ def setComponent(self, component, inserting=False):
raise NotImplementedError
@@ -761,9 +762,19 @@
def properties(self):
uid = self._parentCollection._home.uid()
props = PropertyStore(uid, lambda : self._path)
+ self.initPropertyStore(props)
self._transaction.addOperation(props.flush, "object properties flush")
return props
+ def initPropertyStore(self, props):
+ """
+ A hook for subclasses to override in order to set up their property
+ store after it's been created.
+
+ @param props: the L{PropertyStore} from C{properties()}.
+ """
+ pass
+
class CommonStubResource(object):
"""
Just enough resource to keep the collection sql DB classes going.
@@ -877,7 +888,7 @@
@writeOperation
- def setData(self, uid, xmltype, xmldata):
+ def setData(self, uid, xmltype, xmldata, inserting=False):
rname = uid + ".xml"
self._parentCollection.retrieveOldIndex().addOrUpdateRecord(
Copied: CalendarServer/trunk/txdav/common/datastore/sql.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/common/datastore/sql.py)
===================================================================
--- CalendarServer/trunk/txdav/common/datastore/sql.py (rev 0)
+++ CalendarServer/trunk/txdav/common/datastore/sql.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,1071 @@
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+SQL data store.
+"""
+
+__all__ = [
+ "CommonDataStore",
+ "CommonStoreTransaction",
+ "CommonHome",
+]
+
+from twext.python.log import Logger, LoggingMixIn
+from twext.web2.dav.element.rfc2518 import ResourceType
+from twext.web2.http_headers import MimeType
+
+from twisted.application.service import Service
+from twisted.python import hashlib
+from twisted.python.modules import getModule
+from twisted.python.util import FancyEqMixin
+
+from twistedcaldav.customxml import NotificationType
+
+from txdav.common.datastore.sql_legacy import PostgresLegacyNotificationsEmulator
+from txdav.caldav.icalendarstore import ICalendarTransaction
+
+from txdav.carddav.iaddressbookstore import IAddressBookTransaction
+
+from txdav.common.datastore.sql_tables import CALENDAR_HOME_TABLE,\
+ ADDRESSBOOK_HOME_TABLE, NOTIFICATION_HOME_TABLE, _BIND_MODE_OWN,\
+ _BIND_STATUS_ACCEPTED
+from txdav.common.icommondatastore import HomeChildNameNotAllowedError,\
+ HomeChildNameAlreadyExistsError, NoSuchHomeChildError,\
+ ObjectResourceNameNotAllowedError, ObjectResourceNameAlreadyExistsError,\
+ NoSuchObjectResourceError
+from txdav.common.inotifications import INotificationCollection,\
+ INotificationObject
+from txdav.base.datastore.sql import memoized
+from txdav.base.datastore.util import cached
+from txdav.idav import IDataStore, AlreadyFinishedError
+from txdav.base.propertystore.base import PropertyName
+from txdav.base.propertystore.sql import PropertyStore
+
+from zope.interface.declarations import implements, directlyProvides
+
+v1_schema = getModule(__name__).filePath.sibling(
+ "sql_schema_v1.sql").getContent()
+
+log = Logger()
+
+ECALENDARTYPE = 0
+EADDRESSBOOKTYPE = 1
+
+class CommonDataStore(Service, object):
+
+ implements(IDataStore)
+
+ def __init__(self, connectionFactory, notifierFactory, attachmentsPath,
+ enableCalendars=True, enableAddressBooks=True):
+ assert enableCalendars or enableAddressBooks
+
+ self.connectionFactory = connectionFactory
+ self.notifierFactory = notifierFactory
+ self.attachmentsPath = attachmentsPath
+ self.enableCalendars = enableCalendars
+ self.enableAddressBooks = enableAddressBooks
+
+
+ def newTransaction(self, label="unlabeled"):
+ return CommonStoreTransaction(
+ self,
+ self.connectionFactory(),
+ self.enableCalendars,
+ self.enableAddressBooks,
+ self.notifierFactory,
+ label
+ )
+
+class CommonStoreTransaction(object):
+ """
+ Transaction implementation for SQL database.
+ """
+
+ _homeClass = {}
+
+ def __init__(self, store, connection, enableCalendars, enableAddressBooks, notifierFactory, label):
+
+ self._store = store
+ self._connection = connection
+ self._cursor = connection.cursor()
+ self._completed = False
+ self._calendarHomes = {}
+ self._addressbookHomes = {}
+ self._notificationHomes = {}
+ self._postCommitOperations = []
+ self._notifierFactory = notifierFactory
+ self._label = label
+
+ extraInterfaces = []
+ if enableCalendars:
+ extraInterfaces.append(ICalendarTransaction)
+ if enableAddressBooks:
+ extraInterfaces.append(IAddressBookTransaction)
+ directlyProvides(self, *extraInterfaces)
+
+ from txdav.caldav.datastore.sql import CalendarHome
+ from txdav.carddav.datastore.sql import AddressBookHome
+ CommonStoreTransaction._homeClass[ECALENDARTYPE] = CalendarHome
+ CommonStoreTransaction._homeClass[EADDRESSBOOKTYPE] = AddressBookHome
+
+ def store(self):
+ return self._store
+
+
+ def __repr__(self):
+ return 'PG-TXN<%s>' % (self._label,)
+
+
+ def execSQL(self, sql, args=[], raiseOnZeroRowCount=None):
+ # print 'EXECUTE %s: %s' % (self._label, sql)
+ self._cursor.execute(sql, args)
+ if raiseOnZeroRowCount is not None and self._cursor.rowcount == 0:
+ raise raiseOnZeroRowCount()
+ if self._cursor.description:
+ return self._cursor.fetchall()
+ else:
+ return None
+
+
+ def __del__(self):
+ if not self._completed:
+ self._connection.rollback()
+ self._connection.close()
+
+
+ @memoized('uid', '_calendarHomes')
+ def calendarHomeWithUID(self, uid, create=False):
+ return self.homeWithUID(ECALENDARTYPE, uid, create=create)
+
+ @memoized('uid', '_addressbookHomes')
+ def addressbookHomeWithUID(self, uid, create=False):
+ return self.homeWithUID(EADDRESSBOOKTYPE, uid, create=create)
+
+ def homeWithUID(self, storeType, uid, create=False):
+
+ if storeType == ECALENDARTYPE:
+ homeTable = CALENDAR_HOME_TABLE
+ elif storeType == EADDRESSBOOKTYPE:
+ homeTable = ADDRESSBOOK_HOME_TABLE
+
+ data = self.execSQL(
+ "select %(column_RESOURCE_ID)s from %(name)s where %(column_OWNER_UID)s = %%s" % homeTable,
+ [uid]
+ )
+ if not data:
+ if not create:
+ return None
+ self.execSQL(
+ "insert into %(name)s (%(column_OWNER_UID)s) values (%%s)" % homeTable,
+ [uid]
+ )
+ home = self.homeWithUID(storeType, uid)
+ home.createdHome()
+ return home
+ resid = data[0][0]
+
+ if self._notifierFactory:
+ notifier = self._notifierFactory.newNotifier(id=uid)
+ else:
+ notifier = None
+
+ return self._homeClass[storeType](self, uid, resid, notifier)
+
+
+ @memoized('uid', '_notificationHomes')
+ def notificationsWithUID(self, uid):
+ """
+ Implement notificationsWithUID.
+ """
+ rows = self.execSQL(
+ """
+ select %(column_RESOURCE_ID)s from %(name)s where
+ %(column_OWNER_UID)s = %%s
+ """ % NOTIFICATION_HOME_TABLE, [uid]
+ )
+ if rows:
+ resourceID = rows[0][0]
+ else:
+ resourceID = str(self.execSQL(
+ "insert into %(name)s (%(column_OWNER_UID)s) values (%%s) returning %(column_RESOURCE_ID)s" % NOTIFICATION_HOME_TABLE,
+ [uid]
+ )[0][0])
+ return NotificationCollection(self, uid, resourceID)
+
+
+ def abort(self):
+ if not self._completed:
+ # print 'ABORTING', self._label
+ self._completed = True
+ self._connection.rollback()
+ self._connection.close()
+ else:
+ raise AlreadyFinishedError()
+
+
+ def commit(self):
+ if not self._completed:
+ # print 'COMPLETING', self._label
+ self._completed = True
+ self._connection.commit()
+ self._connection.close()
+ for operation in self._postCommitOperations:
+ operation()
+ else:
+ raise AlreadyFinishedError()
+
+
+ def postCommit(self, operation):
+ """
+ Run things after 'commit.'
+ """
+ self._postCommitOperations.append(operation)
+ # FIXME: implement.
+
+class CommonHome(LoggingMixIn):
+
+ _childClass = None
+ _childTable = None
+ _bindTable = None
+
+ def __init__(self, transaction, ownerUID, resourceID, notifier):
+ self._txn = transaction
+ self._ownerUID = ownerUID
+ self._resourceID = resourceID
+ self._shares = None
+ self._children = {}
+ self._notifier = notifier
+
+
+ def __repr__(self):
+ return "<%s: %s>" % (self.__class__.__name__, self._resourceID)
+
+ def uid(self):
+ """
+ Retrieve the unique identifier for this home.
+
+ @return: a string.
+ """
+ return self._ownerUID
+
+
+ def transaction(self):
+ return self._txn
+
+
+ def retrieveOldShares(self):
+ return self._shares
+
+
+ def name(self):
+ """
+ Implement L{IDataStoreResource.name} to return the uid.
+ """
+ return self.uid()
+
+
+ def children(self):
+ """
+ Retrieve children contained in this home.
+ """
+ names = self.listChildren()
+ for name in names:
+ yield self.childWithName(name)
+
+
+ def listChildren(self):
+ """
+ Retrieve the names of the children in this home.
+
+ @return: an iterable of C{str}s.
+ """
+ # FIXME: not specified on the interface or exercised by the tests, but
+ # required by clients of the implementation!
+ rows = self._txn.execSQL(
+ "select %(column_RESOURCE_NAME)s from %(name)s where "
+ "%(column_HOME_RESOURCE_ID)s = %%s "
+ "and %(column_BIND_MODE)s = %%s " % self._bindTable,
+ # Right now, we only show owned calendars.
+ [self._resourceID, _BIND_MODE_OWN]
+ )
+ names = [row[0] for row in rows]
+ return names
+
+
+ @memoized('name', '_children')
+ def childWithName(self, name):
+ """
+ Retrieve the child with the given C{name} contained in this
+ home.
+
+ @param name: a string.
+ @return: an L{ICalendar} or C{None} if no such child
+ exists.
+ """
+ data = self._txn.execSQL(
+ "select %(column_RESOURCE_ID)s from %(name)s where "
+ "%(column_RESOURCE_NAME)s = %%s and %(column_HOME_RESOURCE_ID)s = %%s "
+ "and %(column_BIND_MODE)s = %%s" % self._bindTable,
+ [name, self._resourceID, _BIND_MODE_OWN]
+ )
+ if not data:
+ return None
+ resourceID = data[0][0]
+ if self._notifier:
+ childID = "%s/%s" % (self.uid(), name)
+ notifier = self._notifier.clone(label="collection", id=childID)
+ else:
+ notifier = None
+ return self._childClass(self, name, resourceID, notifier)
+
+
+ def createChildWithName(self, name):
+ if name.startswith("."):
+ raise HomeChildNameNotAllowedError(name)
+
+ rows = self._txn.execSQL(
+ "select %(column_RESOURCE_NAME)s from %(name)s where "
+ "%(column_RESOURCE_NAME)s = %%s AND "
+ "%(column_HOME_RESOURCE_ID)s = %%s" % self._bindTable,
+ [name, self._resourceID]
+ )
+ if rows:
+ raise HomeChildNameAlreadyExistsError()
+
+ rows = self._txn.execSQL("select nextval('RESOURCE_ID_SEQ')")
+ resourceID = rows[0][0]
+ self._txn.execSQL(
+ "insert into %(name)s (%(column_RESOURCE_ID)s) values "
+ "(%%s)" % self._childTable,
+ [resourceID])
+
+ self._txn.execSQL("""
+ insert into %(name)s (
+ %(column_HOME_RESOURCE_ID)s,
+ %(column_RESOURCE_ID)s, %(column_RESOURCE_NAME)s, %(column_BIND_MODE)s,
+ %(column_SEEN_BY_OWNER)s, %(column_SEEN_BY_SHAREE)s, %(column_BIND_STATUS)s) values (
+ %%s, %%s, %%s, %%s, %%s, %%s, %%s)
+ """ % self._bindTable,
+ [self._resourceID, resourceID, name, _BIND_MODE_OWN, True, True,
+ _BIND_STATUS_ACCEPTED]
+ )
+
+ newChild = self.childWithName(name)
+ newChild.properties()[
+ PropertyName.fromElement(ResourceType)] = newChild.resourceType()
+ newChild._updateSyncToken()
+ self.createdChild(newChild)
+
+ if self._notifier:
+ self._txn.postCommit(self._notifier.notify)
+
+
+ def createdChild(self, child):
+ pass
+
+
+ def removeChildWithName(self, name):
+ rows = self._txn.execSQL(
+ """select %(column_RESOURCE_ID)s from %(name)s
+ where %(column_RESOURCE_NAME)s = %%s and %(column_HOME_RESOURCE_ID)s = %%s""" % self._bindTable,
+ [name, self._resourceID]
+ )
+ if not rows:
+ raise NoSuchHomeChildError()
+ resourceID = rows[0][0]
+
+ self._txn.execSQL(
+ "delete from %(name)s where %(column_RESOURCE_ID)s = %%s" % self._childTable,
+ [resourceID]
+ )
+ self._children.pop(name, None)
+ if self._txn._cursor.rowcount == 0:
+ raise NoSuchHomeChildError()
+ if self._notifier:
+ self._txn.postCommit(self._notifier.notify)
+
+
+ @cached
+ def properties(self):
+ return PropertyStore(
+ self.uid(),
+ self._txn,
+ self._resourceID
+ )
+
+
+ # IDataStoreResource
+ def contentType(self):
+ """
+ The content type of objects
+ """
+ return None
+
+
+ def md5(self):
+ return None
+
+
+ def size(self):
+ return 0
+
+
+ def created(self):
+ return None
+
+
+ def modified(self):
+ return None
+
+
+ def notifierID(self, label="default"):
+ if self._notifier:
+ return self._notifier.getID(label)
+ else:
+ return None
+
+class CommonHomeChild(LoggingMixIn, FancyEqMixin):
+ """
+ Common ancestor class of AddressBooks and Calendars.
+ """
+
+ compareAttributes = '_name _home _resourceID'.split()
+
+ _objectResourceClass = None
+ _bindTable = None
+ _homeChildTable = None
+ _revisionsTable = None
+ _objectTable = None
+
+ def __init__(self, home, name, resourceID, notifier):
+ self._home = home
+ self._name = name
+ self._resourceID = resourceID
+ self._objects = {}
+ self._notifier = notifier
+
+ self._index = None # Derived classes need to set this
+ self._invites = None # Derived classes need to set this
+
+
+ @property
+ def _txn(self):
+ return self._home._txn
+
+
+ def resourceType(self):
+ return NotImplementedError
+
+
+ def retrieveOldIndex(self):
+ return self._index
+
+
+ def retrieveOldInvites(self):
+ return self._invites
+
+ def __repr__(self):
+ return "<%s: %s>" % (self.__class__.__name__, self._resourceID)
+
+ def name(self):
+ return self._name
+
+
+ def rename(self, name):
+ oldName = self._name
+ self._txn.execSQL(
+ "update %(name)s set %(column_RESOURCE_NAME)s = %%s "
+ "where %(column_RESOURCE_ID)s = %%s AND "
+ "%(column_HOME_RESOURCE_ID)s = %%s" % self._bindTable,
+ [name, self._resourceID, self._home._resourceID]
+ )
+ self._name = name
+ # update memos
+ del self._home._children[oldName]
+ self._home._children[name] = self
+ self._updateSyncToken()
+
+ if self._notifier:
+ self._txn.postCommit(self._notifier.notify)
+
+
+ def ownerHome(self):
+ return self._home
+
+
+ def setSharingUID(self, uid):
+ self.properties()._setPerUserUID(uid)
+
+
+ def objectResources(self):
+ for name in self.listObjectResources():
+ yield self.objectResourceWithName(name)
+
+
+ def listObjectResources(self):
+ rows = self._txn.execSQL(
+ "select %(column_RESOURCE_NAME)s from %(name)s "
+ "where %(column_PARENT_RESOURCE_ID)s = %%s" % self._objectTable,
+ [self._resourceID])
+ return sorted([row[0] for row in rows])
+
+
+ @memoized('name', '_objects')
+ def objectResourceWithName(self, name):
+ rows = self._txn.execSQL(
+ "select %(column_RESOURCE_ID)s from %(name)s "
+ "where %(column_RESOURCE_NAME)s = %%s and %(column_PARENT_RESOURCE_ID)s = %%s" % self._objectTable,
+ [name, self._resourceID]
+ )
+ if not rows:
+ return None
+ resid = rows[0][0]
+ return self._objectResourceClass(name, self, resid)
+
+
+ @memoized('uid', '_objects')
+ def objectResourceWithUID(self, uid):
+ rows = self._txn.execSQL(
+ "select %(column_RESOURCE_ID)s, %(column_RESOURCE_NAME)s from %(name)s "
+ "where %(column_UID)s = %%s and %(column_PARENT_RESOURCE_ID)s = %%s" % self._objectTable,
+ [uid, self._resourceID]
+ )
+ if not rows:
+ return None
+ resid = rows[0][0]
+ name = rows[0][1]
+ return self._objectResourceClass(name, self, resid)
+
+
+ def createObjectResourceWithName(self, name, component):
+ if name.startswith("."):
+ raise ObjectResourceNameNotAllowedError(name)
+
+ rows = self._txn.execSQL(
+ "select %(column_RESOURCE_ID)s from %(name)s "
+ "where %(column_RESOURCE_NAME)s = %%s and %(column_PARENT_RESOURCE_ID)s = %%s" % self._objectTable,
+ [name, self._resourceID]
+ )
+ if rows:
+ raise ObjectResourceNameAlreadyExistsError()
+
+ objectResource = self._objectResourceClass(name, self, None)
+ objectResource.setComponent(component, inserting=True)
+
+ # Note: setComponent triggers a notification, so we don't need to
+ # call notify( ) here like we do for object removal.
+
+
+ def removeObjectResourceWithName(self, name):
+ rows = self._txn.execSQL(
+ "delete from %(name)s "
+ "where %(column_RESOURCE_NAME)s = %%s and %(column_PARENT_RESOURCE_ID)s = %%s "
+ "returning %(column_UID)s" % self._objectTable,
+ [name, self._resourceID],
+ raiseOnZeroRowCount=lambda:NoSuchObjectResourceError()
+ )
+ uid = rows[0][0]
+ self._objects.pop(name, None)
+ self._objects.pop(uid, None)
+ self._deleteRevision(name)
+
+ if self._notifier:
+ self._txn.postCommit(self._notifier.notify)
+
+
+ def removeObjectResourceWithUID(self, uid):
+ rows = self._txn.execSQL(
+ "delete from %(name)s "
+ "where %(column_UID)s = %%s and %(column_PARENT_RESOURCE_ID)s = %%s "
+ "returning %(column_RESOURCE_NAME)s" % self._objectTable,
+ [uid, self._resourceID],
+ raiseOnZeroRowCount=lambda:NoSuchObjectResourceError()
+ )
+ name = rows[0][0]
+ self._objects.pop(name, None)
+ self._objects.pop(uid, None)
+ self._deleteRevision(name)
+
+ if self._notifier:
+ self._txn.postCommit(self._notifier.notify)
+
+
+ def syncToken(self):
+ revision = self._txn.execSQL(
+ "select %(column_REVISION)s from %(name)s where %(column_RESOURCE_ID)s = %%s" % self._homeChildTable,
+ [self._resourceID])[0][0]
+ return "%s#%s" % (self._resourceID, revision,)
+
+ def objectResourcesSinceToken(self, token):
+ raise NotImplementedError()
+
+
+ def _updateSyncToken(self):
+
+ self._txn.execSQL("""
+ update %(name)s
+ set (%(column_REVISION)s) = (nextval('%(sequence)s'))
+ where %(column_RESOURCE_ID)s = %%s
+ """ % self._homeChildTable,
+ [self._resourceID]
+ )
+
+ def _insertRevision(self, name):
+ self._changeRevision("insert", name)
+
+ def _updateRevision(self, name):
+ self._changeRevision("update", name)
+
+ def _deleteRevision(self, name):
+ self._changeRevision("delete", name)
+
+ def _changeRevision(self, action, name):
+
+ nextrevision = self._txn.execSQL("""
+ select nextval('%(sequence)s')
+ """ % self._homeChildTable
+ )
+
+ if action == "delete":
+ self._txn.execSQL("""
+ update %(name)s
+ set (%(column_REVISION)s, %(column_DELETED)s) = (%%s, TRUE)
+ where %(column_RESOURCE_ID)s = %%s and %(column_RESOURCE_NAME)s = %%s
+ """ % self._revisionsTable,
+ [nextrevision, self._resourceID, name]
+ )
+ self._txn.execSQL("""
+ update %(name)s
+ set (%(column_REVISION)s) = (%%s)
+ where %(column_RESOURCE_ID)s = %%s
+ """ % self._homeChildTable,
+ [nextrevision, self._resourceID]
+ )
+ elif action == "update":
+ self._txn.execSQL("""
+ update %(name)s
+ set (%(column_REVISION)s) = (%%s)
+ where %(column_RESOURCE_ID)s = %%s and %(column_RESOURCE_NAME)s = %%s
+ """ % self._revisionsTable,
+ [nextrevision, self._resourceID, name]
+ )
+ self._txn.execSQL("""
+ update %(name)s
+ set (%(column_REVISION)s) = (%%s)
+ where %(column_RESOURCE_ID)s = %%s
+ """ % self._homeChildTable,
+ [nextrevision, self._resourceID]
+ )
+ elif action == "insert":
+ self._txn.execSQL("""
+ delete from %(name)s
+ where %(column_RESOURCE_ID)s = %%s and %(column_RESOURCE_NAME)s = %%s
+ """ % self._revisionsTable,
+ [self._resourceID, name,]
+ )
+ self._txn.execSQL("""
+ insert into %(name)s
+ (%(column_RESOURCE_ID)s, %(column_RESOURCE_NAME)s, %(column_REVISION)s, %(column_DELETED)s)
+ values (%%s, %%s, %%s, FALSE)
+ """ % self._revisionsTable,
+ [self._resourceID, name, nextrevision]
+ )
+ self._txn.execSQL("""
+ update %(name)s
+ set (%(column_REVISION)s) = (%%s)
+ where %(column_RESOURCE_ID)s = %%s
+ """ % self._homeChildTable,
+ [nextrevision, self._resourceID]
+ )
+ @cached
+ def properties(self):
+ props = PropertyStore(
+ self.ownerHome().uid(),
+ self._txn,
+ self._resourceID
+ )
+ self.initPropertyStore(props)
+ return props
+
+ def initPropertyStore(self, props):
+ """
+ A hook for subclasses to override in order to set up their property
+ store after it's been created.
+
+ @param props: the L{PropertyStore} from C{properties()}.
+ """
+ pass
+
+ def _doValidate(self, component):
+ raise NotImplementedError
+
+ def notifierID(self, label="default"):
+ if self._notifier:
+ return self._notifier.getID(label)
+ else:
+ return None
+
+
+
+ # IDataStoreResource
+ def contentType(self):
+ raise NotImplementedError()
+
+
+ def md5(self):
+ return None
+
+
+ def size(self):
+ return 0
+
+
+ def created(self):
+ created = self._txn.execSQL(
+ "select extract(EPOCH from %(column_CREATED)s) from %(name)s "
+ "where %(column_RESOURCE_ID)s = %%s" % self._homeChildTable,
+ [self._resourceID]
+ )[0][0]
+ return int(created)
+
+ def modified(self):
+ modified = self._txn.execSQL(
+ "select extract(EPOCH from %(column_MODIFIED)s) from %(name)s "
+ "where %(column_RESOURCE_ID)s = %%s" % self._homeChildTable,
+ [self._resourceID]
+ )[0][0]
+ return int(modified)
+
+class CommonObjectResource(LoggingMixIn, FancyEqMixin):
+ """
+ @ivar _path: The path of the file on disk
+
+ @type _path: L{FilePath}
+ """
+
+ compareAttributes = '_name _parentCollection'.split()
+
+ _objectTable = None
+
+ def __init__(self, name, parent, resid):
+ self._name = name
+ self._parentCollection = parent
+ self._resourceID = resid
+ self._objectText = None
+
+ def __repr__(self):
+ return "<%s: %s>" % (self.__class__.__name__, self._resourceID)
+
+ @property
+ def _txn(self):
+ return self._parentCollection._txn
+
+ def setComponent(self, component, inserting=False):
+ raise NotImplementedError
+
+
+ def component(self):
+ raise NotImplementedError
+
+
+ def text(self):
+ raise NotImplementedError
+
+
+ def uid(self):
+ raise NotImplementedError
+
+ @cached
+ def properties(self):
+ props = PropertyStore(
+ self.uid(),
+ self._txn,
+ self._resourceID
+ )
+ self.initPropertyStore(props)
+ return props
+
+ def initPropertyStore(self, props):
+ """
+ A hook for subclasses to override in order to set up their property
+ store after it's been created.
+
+ @param props: the L{PropertyStore} from C{properties()}.
+ """
+ pass
+
+ # IDataStoreResource
+ def contentType(self):
+ raise NotImplementedError()
+
+ def md5(self):
+ return None
+
+ def size(self):
+ size = self._txn.execSQL(
+ "select character_length(%(column_TEXT)s) from %(name)s "
+ "where %(column_RESOURCE_ID)s = %%s" % self._objectTable,
+ [self._resourceID]
+ )[0][0]
+ return size
+
+
+ def created(self):
+ created = self._txn.execSQL(
+ "select extract(EPOCH from %(column_CREATED)s) from %(name)s "
+ "where %(column_RESOURCE_ID)s = %%s" % self._objectTable,
+ [self._resourceID]
+ )[0][0]
+ return int(created)
+
+ def modified(self):
+ modified = self._txn.execSQL(
+ "select extract(EPOCH from %(column_MODIFIED)s) from %(name)s "
+ "where %(column_RESOURCE_ID)s = %%s" % self._objectTable,
+ [self._resourceID]
+ )[0][0]
+ return int(modified)
+
+class NotificationCollection(LoggingMixIn, FancyEqMixin):
+
+ implements(INotificationCollection)
+
+ compareAttributes = '_uid _resourceID'.split()
+
+ _objectResourceClass = None
+ _bindTable = None
+ _homeChildTable = None
+ _revisionsTable = None
+ _objectTable = None
+
+ def __init__(self, txn, uid, resourceID):
+
+ self._txn = txn
+ self._uid = uid
+ self._resourceID = resourceID
+ self._notifications = {}
+
+
+ def resourceType(self):
+ return ResourceType.notification #@UndefinedVariable
+
+ def retrieveOldIndex(self):
+ return PostgresLegacyNotificationsEmulator(self)
+
+ def __repr__(self):
+ return "<%s: %s>" % (self.__class__.__name__, self._resourceID)
+
+ def name(self):
+ return 'notification'
+
+ def uid(self):
+ return self._uid
+
+ def notificationObjects(self):
+ for name in self.listNotificationObjects():
+ yield self.notificationObjectWithName(name)
+
+ def listNotificationObjects(self):
+ rows = self._txn.execSQL(
+ "select (NOTIFICATION_UID) from NOTIFICATION "
+ "where NOTIFICATION_HOME_RESOURCE_ID = %s",
+ [self._resourceID])
+ return sorted(["%s.xml" % row[0] for row in rows])
+
+ def _nameToUID(self, name):
+ """
+ Based on the file-backed implementation, the 'name' is just uid +
+ ".xml".
+ """
+ return name.rsplit(".", 1)[0]
+
+
+ def notificationObjectWithName(self, name):
+ return self.notificationObjectWithUID(self._nameToUID(name))
+
+ @memoized('uid', '_notifications')
+ def notificationObjectWithUID(self, uid):
+ rows = self._txn.execSQL(
+ "select RESOURCE_ID from NOTIFICATION "
+ "where NOTIFICATION_UID = %s and NOTIFICATION_HOME_RESOURCE_ID = %s",
+ [uid, self._resourceID])
+ if rows:
+ resourceID = rows[0][0]
+ return NotificationObject(self, resourceID)
+ else:
+ return None
+
+
+ def writeNotificationObject(self, uid, xmltype, xmldata):
+
+ inserting = False
+ notificationObject = self.notificationObjectWithUID(uid)
+ if notificationObject is None:
+ notificationObject = NotificationObject(self, None)
+ inserting = True
+ notificationObject.setData(uid, xmltype, xmldata, inserting=inserting)
+
+
+ def removeNotificationObjectWithName(self, name):
+ self.removeNotificationObjectWithUID(self._nameToUID(name))
+
+
+ def removeNotificationObjectWithUID(self, uid):
+ self._txn.execSQL(
+ "delete from NOTIFICATION "
+ "where NOTIFICATION_UID = %s and NOTIFICATION_HOME_RESOURCE_ID = %s",
+ [uid, self._resourceID]
+ )
+ self._notifications.pop(uid, None)
+
+
+ def syncToken(self):
+ return 'dummy-sync-token'
+
+
+ def notificationObjectsSinceToken(self, token):
+ changed = []
+ removed = []
+ token = self.syncToken()
+ return (changed, removed, token)
+
+
+ @cached
+ def properties(self):
+ return PropertyStore(
+ self._uid,
+ self._txn,
+ self._resourceID
+ )
+
+class NotificationObject(LoggingMixIn, FancyEqMixin):
+ implements(INotificationObject)
+
+ compareAttributes = '_resourceID _home'.split()
+
+ def __init__(self, home, resourceID):
+ self._home = home
+ self._resourceID = resourceID
+
+
+ def __repr__(self):
+ return "<%s: %s>" % (self.__class__.__name__, self._resourceID)
+
+
+ @property
+ def _txn(self):
+ return self._home._txn
+
+
+ def notificationCollection(self):
+ return self._home
+
+
+ def name(self):
+ return self.uid() + ".xml"
+
+
+ def setData(self, uid, xmltype, xmldata, inserting=False):
+
+ xmltypeString = xmltype.toxml()
+ if inserting:
+ rows = self._txn.execSQL(
+ "insert into NOTIFICATION (NOTIFICATION_HOME_RESOURCE_ID, NOTIFICATION_UID, XML_TYPE, XML_DATA) "
+ "values (%s, %s, %s, %s) returning RESOURCE_ID",
+ [self._home._resourceID, uid, xmltypeString, xmldata]
+ )
+ self._resourceID = rows[0][0]
+ else:
+ self._txn.execSQL(
+ "update NOTIFICATION set XML_TYPE = %s, XML_DATA = %s "
+ "where NOTIFICATION_HOME_RESOURCE_ID = %s and NOTIFICATION_UID = %s",
+ [xmltypeString, xmldata, self._home._resourceID, uid])
+
+ self.properties()[PropertyName.fromElement(NotificationType)] = NotificationType(xmltype)
+
+
+ def _fieldQuery(self, field):
+ data = self._txn.execSQL(
+ "select " + field + " from NOTIFICATION "
+ "where RESOURCE_ID = %s",
+ [self._resourceID]
+ )
+ return data[0][0]
+
+
+ def xmldata(self):
+ return self._fieldQuery("XML_DATA")
+
+
+ def uid(self):
+ return self._fieldQuery("NOTIFICATION_UID")
+
+
+ @cached
+ def properties(self):
+ props = PropertyStore(
+ self._home.uid(),
+ self._txn,
+ self._resourceID
+ )
+ self.initPropertyStore(props)
+ return props
+
+ def initPropertyStore(self, props):
+ # Setup peruser special properties
+ props.setSpecialProperties(
+ (
+ ),
+ (
+ PropertyName.fromElement(NotificationType),
+ ),
+ )
+
+ def contentType(self):
+ """
+ The content type of NotificationObjects is text/xml.
+ """
+ return MimeType.fromString("text/xml")
+
+
+ def md5(self):
+ return hashlib.md5(self.xmldata()).hexdigest()
+
+
+ def size(self):
+ size = self._txn.execSQL(
+ "select character_length(XML_DATA) from NOTIFICATION "
+ "where RESOURCE_ID = %s",
+ [self._resourceID]
+ )[0][0]
+ return size
+
+
+ def created(self):
+ modified = self._txn.execSQL(
+ "select extract(EPOCH from CREATED) from NOTIFICATION "
+ "where RESOURCE_ID = %s",
+ [self._resourceID]
+ )[0][0]
+ return int(modified)
+
+ def modified(self):
+ modified = self._txn.execSQL(
+ "select extract(EPOCH from MODIFIED) from NOTIFICATION "
+ "where RESOURCE_ID = %s", [self._resourceID]
+ )[0][0]
+ return int(modified)
Copied: CalendarServer/trunk/txdav/common/datastore/sql_legacy.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/common/datastore/sql_legacy.py)
===================================================================
--- CalendarServer/trunk/txdav/common/datastore/sql_legacy.py (rev 0)
+++ CalendarServer/trunk/txdav/common/datastore/sql_legacy.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,1325 @@
+# -*- test-case-name: txdav.caldav.datastore.test.test_sql -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+
+"""
+PostgreSQL data store.
+"""
+
+import datetime
+import StringIO
+
+from twistedcaldav.sharing import SharedCollectionRecord
+
+from twisted.python import hashlib
+from twisted.internet.defer import succeed
+
+from twext.python.log import Logger, LoggingMixIn
+
+from twistedcaldav import carddavxml
+from twistedcaldav.config import config
+from twistedcaldav.dateops import normalizeForIndex
+from twistedcaldav.index import IndexedSearchException, ReservationError,\
+ SyncTokenValidException
+from twistedcaldav.memcachepool import CachePoolUserMixIn
+from twistedcaldav.notifications import NotificationRecord
+from twistedcaldav.query import calendarqueryfilter, calendarquery, \
+ addressbookquery
+from twistedcaldav.query.sqlgenerator import sqlgenerator
+from twistedcaldav.sharing import Invite
+
+from txdav.common.datastore.sql_tables import \
+ _BIND_MODE_OWN, _BIND_MODE_READ, _BIND_MODE_WRITE, _BIND_STATUS_INVITED,\
+ _BIND_STATUS_ACCEPTED, _BIND_STATUS_DECLINED, _BIND_STATUS_INVALID
+
+log = Logger()
+
+indexfbtype_to_icalfbtype = {
+ 0: '?',
+ 1: 'F',
+ 2: 'B',
+ 3: 'U',
+ 4: 'T',
+}
+
+class PostgresLegacyInvitesEmulator(object):
+ """
+ Emulator for the implicit interface specified by
+ L{twistedcaldav.sharing.InvitesDatabase}.
+ """
+
+
+ def __init__(self, calendar):
+ self._calendar = calendar
+
+
+ @property
+ def _txn(self):
+ return self._calendar._txn
+
+
+ def create(self):
+ "No-op, because the index implicitly always exists in the database."
+
+
+ def remove(self):
+ "No-op, because the index implicitly always exists in the database."
+
+
+ def allRecords(self):
+ for row in self._txn.execSQL(
+ """
+ select
+ INVITE.INVITE_UID, INVITE.NAME, INVITE.RECIPIENT_ADDRESS,
+ CALENDAR_HOME.OWNER_UID, CALENDAR_BIND.BIND_MODE,
+ CALENDAR_BIND.BIND_STATUS, CALENDAR_BIND.MESSAGE
+ from
+ INVITE, CALENDAR_HOME, CALENDAR_BIND
+ where
+ INVITE.RESOURCE_ID = %s and
+ INVITE.HOME_RESOURCE_ID =
+ CALENDAR_HOME.RESOURCE_ID and
+ CALENDAR_BIND.CALENDAR_RESOURCE_ID =
+ INVITE.RESOURCE_ID and
+ CALENDAR_BIND.CALENDAR_HOME_RESOURCE_ID =
+ INVITE.HOME_RESOURCE_ID
+ order by
+ INVITE.NAME asc
+ """, [self._calendar._resourceID]):
+ [inviteuid, common_name, userid, ownerUID,
+ bindMode, bindStatus, summary] = row
+ # FIXME: this is really the responsibility of the protocol layer.
+ state = {
+ _BIND_STATUS_INVITED: "NEEDS-ACTION",
+ _BIND_STATUS_ACCEPTED: "ACCEPTED",
+ _BIND_STATUS_DECLINED: "DECLINED",
+ _BIND_STATUS_INVALID: "INVALID",
+ }[bindStatus]
+ access = {
+ _BIND_MODE_READ: "read-only",
+ _BIND_MODE_WRITE: "read-write"
+ }[bindMode]
+ principalURL = "/principals/__uids__/%s/" % (ownerUID,)
+ yield Invite(
+ inviteuid, userid, principalURL, common_name,
+ access, state, summary
+ )
+
+
+ def recordForUserID(self, userid):
+ for record in self.allRecords():
+ if record.userid == userid:
+ return record
+
+
+ def recordForPrincipalURL(self, principalURL):
+ for record in self.allRecords():
+ if record.principalURL == principalURL:
+ return record
+
+
+ def recordForInviteUID(self, inviteUID):
+ for record in self.allRecords():
+ if record.inviteuid == inviteUID:
+ return record
+
+
+ def addOrUpdateRecord(self, record):
+ bindMode = {'read-only': _BIND_MODE_READ,
+ 'read-write': _BIND_MODE_WRITE}[record.access]
+ bindStatus = {
+ "NEEDS-ACTION": _BIND_STATUS_INVITED,
+ "ACCEPTED": _BIND_STATUS_ACCEPTED,
+ "DECLINED": _BIND_STATUS_DECLINED,
+ "INVALID": _BIND_STATUS_INVALID,
+ }[record.state]
+ # principalURL is derived from a directory record's principalURL() so
+ # it will always contain the UID. The form is '/principals/__uids__/x'
+ # (and may contain a trailing slash).
+ principalUID = record.principalURL.split("/")[3]
+ shareeHome = self._txn.calendarHomeWithUID(principalUID, create=True)
+ rows = self._txn.execSQL(
+ "select RESOURCE_ID, HOME_RESOURCE_ID from INVITE where RECIPIENT_ADDRESS = %s",
+ [record.userid]
+ )
+ if rows:
+ [[resourceID, homeResourceID]] = rows
+ # Invite(inviteuid, userid, principalURL, common_name, access, state, summary)
+ self._txn.execSQL("""
+ update CALENDAR_BIND set BIND_MODE = %s,
+ BIND_STATUS = %s, MESSAGE = %s
+ where
+ CALENDAR_RESOURCE_ID = %s and
+ CALENDAR_HOME_RESOURCE_ID = %s
+ """, [bindMode, bindStatus, record.summary,
+ resourceID, homeResourceID])
+ self._txn.execSQL("""
+ update INVITE set NAME = %s, INVITE_UID = %s
+ where RECIPIENT_ADDRESS = %s
+ """,
+ [record.name, record.inviteuid, record.userid]
+ )
+ else:
+ self._txn.execSQL(
+ """
+ insert into INVITE (
+ INVITE_UID, NAME,
+ HOME_RESOURCE_ID, RESOURCE_ID,
+ RECIPIENT_ADDRESS
+ )
+ values (%s, %s, %s, %s, %s)
+ """,
+ [
+ record.inviteuid, record.name,
+ shareeHome._resourceID, self._calendar._resourceID,
+ record.userid
+ ])
+ self._txn.execSQL(
+ """
+ insert into CALENDAR_BIND
+ (
+ CALENDAR_HOME_RESOURCE_ID, CALENDAR_RESOURCE_ID,
+ CALENDAR_RESOURCE_NAME, BIND_MODE, BIND_STATUS,
+ SEEN_BY_OWNER, SEEN_BY_SHAREE, MESSAGE
+ )
+ values (%s, %s, %s, %s, %s, %s, %s, %s)
+ """,
+ [
+ shareeHome._resourceID,
+ self._calendar._resourceID,
+ None, # this is NULL because it is not bound yet, let's be
+ # explicit about that.
+ bindMode,
+ bindStatus,
+ False,
+ False,
+ record.summary
+ ])
+
+
+ def removeRecordForUserID(self, userid):
+ rec = self.recordForUserID(userid)
+ self.removeRecordForInviteUID(rec.inviteuid)
+
+
+ def removeRecordForPrincipalURL(self, principalURL):
+ raise NotImplementedError("removeRecordForPrincipalURL")
+
+
+ def removeRecordForInviteUID(self, inviteUID):
+ rows = self._txn.execSQL("""
+ select HOME_RESOURCE_ID, RESOURCE_ID from INVITE where
+ INVITE_UID = %s
+ """, [inviteUID])
+ if rows:
+ [[homeID, resourceID]] = rows
+ self._txn.execSQL(
+ "delete from CALENDAR_BIND where "
+ "CALENDAR_HOME_RESOURCE_ID = %s and CALENDAR_RESOURCE_ID = %s",
+ [homeID, resourceID])
+ self._txn.execSQL("delete from INVITE where INVITE_UID = %s",
+ [inviteUID])
+
+
+
+class PostgresLegacySharesEmulator(object):
+
+ def __init__(self, home):
+ self._home = home
+
+
+ @property
+ def _txn(self):
+ return self._home._txn
+
+
+ def create(self):
+ pass
+
+
+ def remove(self):
+ pass
+
+
+ def allRecords(self):
+ # This should have been a smart join that got all these columns at
+ # once, but let's not bother to fix it, since the actual query we
+ # _want_ to do (just look for calendar binds in a particular homes) is
+ # much simpler anyway; we should just do that.
+ shareRows = self._txn.execSQL(
+ """
+ select CALENDAR_RESOURCE_ID, CALENDAR_RESOURCE_NAME, MESSAGE
+ from CALENDAR_BIND
+ where CALENDAR_HOME_RESOURCE_ID = %s and
+ BIND_MODE != %s and
+ CALENDAR_RESOURCE_NAME is not null
+ """, [self._home._resourceID, _BIND_MODE_OWN])
+ for resourceID, resourceName, summary in shareRows:
+ [[shareuid]] = self._txn.execSQL(
+ """
+ select INVITE_UID
+ from INVITE
+ where RESOURCE_ID = %s and HOME_RESOURCE_ID = %s
+ """, [resourceID, self._home._resourceID])
+ sharetype = 'I'
+ [[ownerHomeID, ownerResourceName]] = self._txn.execSQL(
+ """
+ select CALENDAR_HOME_RESOURCE_ID, CALENDAR_RESOURCE_NAME
+ from CALENDAR_BIND
+ where CALENDAR_RESOURCE_ID = %s and
+ BIND_MODE = %s
+ """, [resourceID, _BIND_MODE_OWN]
+ )
+ [[ownerUID]] = self._txn.execSQL(
+ "select OWNER_UID from CALENDAR_HOME where RESOURCE_ID = %s",
+ [ownerHomeID])
+ hosturl = '/calendars/__uids__/%s/%s' % (
+ ownerUID, ownerResourceName
+ )
+ localname = resourceName
+ record = SharedCollectionRecord(
+ shareuid, sharetype, hosturl, localname, summary
+ )
+ yield record
+
+
+ def _search(self, **kw):
+ [[key, value]] = kw.items()
+ for record in self.allRecords():
+ if getattr(record, key) == value:
+ return record
+
+ def recordForLocalName(self, localname):
+ return self._search(localname=localname)
+
+ def recordForShareUID(self, shareUID):
+ return self._search(shareuid=shareUID)
+
+
+ def addOrUpdateRecord(self, record):
+# print '*** SHARING***: Adding or updating this record:'
+# import pprint
+# pprint.pprint(record.__dict__)
+ # record.hosturl -> /calendars/__uids__/<uid>/<calendarname>
+ splithost = record.hosturl.split('/')
+ ownerUID = splithost[3]
+ ownerCalendarName = splithost[4]
+ ownerHome = self._txn.calendarHomeWithUID(ownerUID)
+ ownerCalendar = ownerHome.calendarWithName(ownerCalendarName)
+ calendarResourceID = ownerCalendar._resourceID
+
+ # There needs to be a bind already, one that corresponds to the
+ # invitation. The invitation's UID is the same as the share UID. I
+ # just need to update its 'localname', i.e.
+ # CALENDAR_BIND.CALENDAR_RESOURCE_NAME.
+
+ self._txn.execSQL(
+ """
+ update CALENDAR_BIND set CALENDAR_RESOURCE_NAME = %s
+ where CALENDAR_HOME_RESOURCE_ID = %s and CALENDAR_RESOURCE_ID = %s
+ """,
+ [record.localname, self._home._resourceID, calendarResourceID]
+ )
+
+
+ def removeRecordForLocalName(self, localname):
+ self._txn.execSQL(
+ "delete from CALENDAR_BIND where CALENDAR_RESOURCE_NAME = %s "
+ "and CALENDAR_HOME_RESOURCE_ID = %s",
+ [localname, self._home._resourceID]
+ )
+
+
+ def removeRecordForShareUID(self, shareUID):
+ pass
+# c = self._home._cursor()
+# c.execute(
+# "delete from CALENDAR_BIND where CALENDAR_RESOURCE_NAME = %s "
+# "and CALENDAR_HOME_RESOURCE_ID = %s",
+# [self._home._resourceID]
+# )
+
+
+
+class postgresqlgenerator(sqlgenerator):
+ """
+ Query generator for postgreSQL indexed searches. (Currently unused: work
+ in progress.)
+ """
+
+ ISOP = " = "
+ CONTAINSOP = " LIKE "
+ NOTCONTAINSOP = " NOT LIKE "
+ FIELDS = {
+ "TYPE": "CALENDAR_OBJECT.ICALENDAR_TYPE",
+ "UID": "CALENDAR_OBJECT.ICALENDAR_UID",
+ }
+
+ def __init__(self, expr, calendarid, userid):
+ self.RESOURCEDB = "CALENDAR_OBJECT"
+ self.TIMESPANDB = "TIME_RANGE"
+ self.TIMESPANTEST = "((TIME_RANGE.FLOATING = FALSE AND TIME_RANGE.START_DATE < %s AND TIME_RANGE.END_DATE > %s) OR (TIME_RANGE.FLOATING = TRUE AND TIME_RANGE.START_DATE < %s AND TIME_RANGE.END_DATE > %s))"
+ self.TIMESPANTEST_NOEND = "((TIME_RANGE.FLOATING = FALSE AND TIME_RANGE.END_DATE > %s) OR (TIME_RANGE.FLOATING = TRUE AND TIME_RANGE.END_DATE > %s))"
+ self.TIMESPANTEST_NOSTART = "((TIME_RANGE.FLOATING = FALSE AND TIME_RANGE.START_DATE < %s) OR (TIME_RANGE.FLOATING = TRUE AND TIME_RANGE.START_DATE < %s))"
+ self.TIMESPANTEST_TAIL_PIECE = " AND TIME_RANGE.CALENDAR_OBJECT_RESOURCE_ID = CALENDAR_OBJECT.RESOURCE_ID AND CALENDAR_OBJECT.CALENDAR_RESOURCE_ID = %s"
+ self.TIMESPANTEST_JOIN_ON_PIECE = "TIME_RANGE.INSTANCE_ID = TRANSPARENCY.TIME_RANGE_INSTANCE_ID AND TRANSPARENCY.USER_ID = %s"
+
+ super(postgresqlgenerator, self).__init__(expr, calendarid, userid)
+
+
+ def generate(self):
+ """
+ Generate the actual SQL 'where ...' expression from the passed in
+ expression tree.
+
+ @return: a C{tuple} of (C{str}, C{list}), where the C{str} is the
+ partial SQL statement, and the C{list} is the list of argument
+ substitutions to use with the SQL API execute method.
+ """
+
+ # Init state
+ self.sout = StringIO.StringIO()
+ self.arguments = []
+ self.substitutions = []
+ self.usedtimespan = False
+
+ # Generate ' where ...' partial statement
+ self.sout.write(self.WHERE)
+ self.generateExpression(self.expression)
+
+ # Prefix with ' from ...' partial statement
+ select = self.FROM + self.RESOURCEDB
+ if self.usedtimespan:
+ self.frontArgument(self.userid)
+ select += ", %s LEFT OUTER JOIN %s ON (%s)" % (
+ self.TIMESPANDB,
+ self.TRANSPARENCYDB,
+ self.TIMESPANTEST_JOIN_ON_PIECE
+ )
+ select += self.sout.getvalue()
+
+ select = select % tuple(self.substitutions)
+
+ return select, self.arguments
+
+
+ def addArgument(self, arg):
+ self.arguments.append(arg)
+ self.substitutions.append("%s")
+ self.sout.write("%s")
+
+ def setArgument(self, arg):
+ self.arguments.append(arg)
+ self.substitutions.append("%s")
+
+ def frontArgument(self, arg):
+ self.arguments.insert(0, arg)
+ self.substitutions.insert(0, "%s")
+
+ def containsArgument(self, arg):
+ return "%%%s%%" % (arg,)
+
+
+class MemcachedUIDReserver(CachePoolUserMixIn, LoggingMixIn):
+ def __init__(self, index, cachePool=None):
+ self.index = index
+ self._cachePool = cachePool
+
+ def _key(self, uid):
+ return 'reservation:%s' % (
+ hashlib.md5('%s:%s' % (uid,
+ self.index.resource._resourceID)).hexdigest())
+
+ def reserveUID(self, uid):
+ uid = uid.encode('utf-8')
+ self.log_debug("Reserving UID %r @ %r" % (
+ uid,
+ self.index.resource))
+
+ def _handleFalse(result):
+ if result is False:
+ raise ReservationError(
+ "UID %s already reserved for calendar collection %s."
+ % (uid, self.index.resource._name)
+ )
+
+ d = self.getCachePool().add(self._key(uid),
+ 'reserved',
+ expireTime=config.UIDReservationTimeOut)
+ d.addCallback(_handleFalse)
+ return d
+
+
+ def unreserveUID(self, uid):
+ uid = uid.encode('utf-8')
+ self.log_debug("Unreserving UID %r @ %r" % (
+ uid,
+ self.index.resource))
+
+ def _handleFalse(result):
+ if result is False:
+ raise ReservationError(
+ "UID %s is not reserved for calendar collection %s."
+ % (uid, self.index.resource._resourceID)
+ )
+
+ d = self.getCachePool().delete(self._key(uid))
+ d.addCallback(_handleFalse)
+ return d
+
+
+ def isReservedUID(self, uid):
+ uid = uid.encode('utf-8')
+ self.log_debug("Is reserved UID %r @ %r" % (
+ uid,
+ self.index.resource))
+
+ def _checkValue((flags, value)):
+ if value is None:
+ return False
+ else:
+ return True
+
+ d = self.getCachePool().get(self._key(uid))
+ d.addCallback(_checkValue)
+ return d
+
+class DummyUIDReserver(LoggingMixIn):
+
+ def __init__(self, index):
+ self.index = index
+ self.reservations = set()
+
+ def _key(self, uid):
+ return 'reservation:%s' % (
+ hashlib.md5('%s:%s' % (uid,
+ self.index.resource._resourceID)).hexdigest())
+
+ def reserveUID(self, uid):
+ uid = uid.encode('utf-8')
+ self.log_debug("Reserving UID %r @ %r" % (
+ uid,
+ self.index.resource))
+
+ key = self._key(uid)
+ if key in self.reservations:
+ raise ReservationError(
+ "UID %s already reserved for calendar collection %s."
+ % (uid, self.index.resource._name)
+ )
+ self.reservations.add(key)
+ return succeed(None)
+
+
+ def unreserveUID(self, uid):
+ uid = uid.encode('utf-8')
+ self.log_debug("Unreserving UID %r @ %r" % (
+ uid,
+ self.index.resource))
+
+ key = self._key(uid)
+ if key in self.reservations:
+ self.reservations.remove(key)
+ return succeed(None)
+
+
+ def isReservedUID(self, uid):
+ uid = uid.encode('utf-8')
+ self.log_debug("Is reserved UID %r @ %r" % (
+ uid,
+ self.index.resource))
+ key = self._key(uid)
+ return succeed(key in self.reservations)
+
+class PostgresLegacyIndexEmulator(LoggingMixIn):
+ """
+ Emulator for L{twistedcaldv.index.Index} and
+ L{twistedcaldv.index.IndexSchedule}.
+ """
+
+ def __init__(self, calendar):
+ self.resource = self.calendar = calendar
+ if (
+ hasattr(config, "Memcached") and
+ config.Memcached.Pools.Default.ClientEnabled
+ ):
+ self.reserver = MemcachedUIDReserver(self)
+ else:
+ # This is only used with unit tests
+ self.reserver = DummyUIDReserver(self)
+
+ @property
+ def _txn(self):
+ return self.calendar._txn
+
+
+ def reserveUID(self, uid):
+ if self.calendar._name == "inbox":
+ return succeed(None)
+ else:
+ return self.reserver.reserveUID(uid)
+
+
+ def unreserveUID(self, uid):
+ if self.calendar._name == "inbox":
+ return succeed(None)
+ else:
+ return self.reserver.unreserveUID(uid)
+
+
+ def isReservedUID(self, uid):
+ if self.calendar._name == "inbox":
+ return succeed(False)
+ else:
+ return self.reserver.isReservedUID(uid)
+
+
+ def isAllowedUID(self, uid, *names):
+ """
+ Checks to see whether to allow an operation which would add the
+ specified UID to the index. Specifically, the operation may not
+ violate the constraint that UIDs must be unique.
+ @param uid: the UID to check
+ @param names: the names of resources being replaced or deleted by the
+ operation; UIDs associated with these resources are not checked.
+ @return: True if the UID is not in the index and is not reserved,
+ False otherwise.
+ """
+ if self.calendar._name == "inbox":
+ return True
+ else:
+ rname = self.resourceNameForUID(uid)
+ return (rname is None or rname in names)
+
+ def resourceUIDForName(self, name):
+ obj = self.calendar.calendarObjectWithName(name)
+ if obj is None:
+ return None
+ return obj.uid()
+
+
+ def resourceNameForUID(self, uid):
+ obj = self.calendar.calendarObjectWithUID(uid)
+ if obj is None:
+ return None
+ return obj.name()
+
+
+ def notExpandedBeyond(self, minDate):
+ """
+ Gives all resources which have not been expanded beyond a given date
+ in the database. (Unused; see above L{postgresqlgenerator}.
+ """
+ return [row[0] for row in self._txn.execSQL(
+ "select RESOURCE_NAME from CALENDAR_OBJECT "
+ "where RECURRANCE_MAX < %s and CALENDAR_RESOURCE_ID = %s",
+ [normalizeForIndex(minDate), self.calendar._resourceID]
+ )]
+
+
+ def reExpandResource(self, name, expand_until):
+ """
+ Given a resource name, remove it from the database and re-add it
+ with a longer expansion.
+ """
+ obj = self.calendar.calendarObjectWithName(name)
+ obj.updateDatabase(obj.component(), expand_until=expand_until, reCreate=True)
+
+ def testAndUpdateIndex(self, minDate):
+ # Find out if the index is expanded far enough
+ names = self.notExpandedBeyond(minDate)
+
+ # Actually expand recurrence max
+ for name in names:
+ self.log_info("Search falls outside range of index for %s %s" % (name, minDate))
+ self.reExpandResource(name, minDate)
+
+ def whatchanged(self, revision):
+
+ results = [
+ (name.encode("utf-8"), deleted)
+ for name, deleted in
+ self._txn.execSQL(
+ """select RESOURCE_NAME, DELETED from CALENDAR_OBJECT_REVISIONS
+ where REVISION > %s and CALENDAR_RESOURCE_ID = %s""",
+ [revision, self.calendar._resourceID],
+ )
+ ]
+ results.sort(key=lambda x:x[1])
+
+ changed = []
+ deleted = []
+ for name, wasdeleted in results:
+ if name:
+ if wasdeleted:
+ if revision:
+ deleted.append(name)
+ else:
+ changed.append(name)
+ else:
+ raise SyncTokenValidException
+
+ return changed, deleted,
+
+ def indexedSearch(self, filter, useruid='', fbtype=False):
+ """
+ Finds resources matching the given qualifiers.
+ @param filter: the L{Filter} for the calendar-query to execute.
+ @return: an iterable of tuples for each resource matching the
+ given C{qualifiers}. The tuples are C{(name, uid, type)}, where
+ C{name} is the resource name, C{uid} is the resource UID, and
+ C{type} is the resource iCalendar component type.x
+ """
+
+ # Make sure we have a proper Filter element and get the partial SQL
+ # statement to use.
+ if isinstance(filter, calendarqueryfilter.Filter):
+ qualifiers = calendarquery.sqlcalendarquery(filter, self.calendar._resourceID, useruid, generator=postgresqlgenerator)
+ if qualifiers is not None:
+ # Determine how far we need to extend the current expansion of
+ # events. If we have an open-ended time-range we will expand one
+ # year past the start. That should catch bounded recurrences - unbounded
+ # will have been indexed with an "infinite" value always included.
+ maxDate, isStartDate = filter.getmaxtimerange()
+ if maxDate:
+ maxDate = maxDate.date()
+ if isStartDate:
+ maxDate += datetime.timedelta(days=365)
+ self.testAndUpdateIndex(maxDate)
+ else:
+ # We cannot handler this filter in an indexed search
+ raise IndexedSearchException()
+
+ else:
+ qualifiers = None
+
+ # Perform the search
+ if qualifiers is None:
+ rowiter = self._txn.execSQL(
+ "select RESOURCE_NAME, ICALENDAR_UID, ICALENDAR_TYPE from CALENDAR_OBJECT where CALENDAR_RESOURCE_ID = %s",
+ [self.calendar._resourceID, ],
+ )
+ else:
+ if fbtype:
+ # For a free-busy time-range query we return all instances
+ rowiter = self._txn.execSQL(
+ """select DISTINCT
+ CALENDAR_OBJECT.RESOURCE_NAME, CALENDAR_OBJECT.ICALENDAR_UID, CALENDAR_OBJECT.ICALENDAR_TYPE, CALENDAR_OBJECT.ORGANIZER,
+ TIME_RANGE.FLOATING, TIME_RANGE.START_DATE, TIME_RANGE.END_DATE, TIME_RANGE.FBTYPE, TIME_RANGE.TRANSPARENT, TRANSPARENCY.TRANSPARENT""" +
+ qualifiers[0],
+ qualifiers[1]
+ )
+ else:
+ rowiter = self._txn.execSQL(
+ "select DISTINCT CALENDAR_OBJECT.RESOURCE_NAME, CALENDAR_OBJECT.ICALENDAR_UID, CALENDAR_OBJECT.ICALENDAR_TYPE" +
+ qualifiers[0],
+ qualifiers[1]
+ )
+
+ # Check result for missing resources
+
+ for row in rowiter:
+ if fbtype:
+ row = list(row)
+ row[4] = 'Y' if row[4] else 'N'
+ row[7] = indexfbtype_to_icalfbtype[row[7]]
+ row[8] = 'T' if row[9] else 'F'
+ del row[9]
+ yield row
+
+
+ def bruteForceSearch(self):
+ return self._txn.execSQL(
+ "select RESOURCE_NAME, ICALENDAR_UID, ICALENDAR_TYPE from "
+ "CALENDAR_OBJECT where CALENDAR_RESOURCE_ID = %s",
+ [self.calendar._resourceID]
+ )
+
+
+ def resourcesExist(self, names):
+ return list(set(names).intersection(
+ set(self.calendar.listCalendarObjects())))
+
+
+ def resourceExists(self, name):
+ return bool(
+ self._txn.execSQL(
+ "select RESOURCE_NAME from CALENDAR_OBJECT where "
+ "RESOURCE_NAME = %s and CALENDAR_RESOURCE_ID = %s",
+ [name, self.calendar._resourceID]
+ )
+ )
+
+
+
+
+class PostgresLegacyNotificationsEmulator(object):
+ def __init__(self, notificationsCollection):
+ self._collection = notificationsCollection
+
+
+ def _recordForObject(self, notificationObject):
+ return NotificationRecord(
+ notificationObject.uid(),
+ notificationObject.name(),
+ notificationObject._fieldQuery("XML_TYPE"))
+
+
+ def recordForName(self, name):
+ return self._recordForObject(
+ self._collection.notificationObjectWithName(name)
+ )
+
+
+ def recordForUID(self, uid):
+ return self._recordForObject(
+ self._collection.notificationObjectWithUID(uid)
+ )
+
+
+ def removeRecordForUID(self, uid):
+ self._collection.removeNotificationObjectWithUID(uid)
+
+
+ def removeRecordForName(self, name):
+ self._collection.removeNotificationObjectWithName(name)
+
+
+
+
+# CARDDAV
+
+class postgresqladbkgenerator(sqlgenerator):
+ """
+ Query generator for postgreSQL indexed searches. (Currently unused: work
+ in progress.)
+ """
+
+ ISOP = " = "
+ CONTAINSOP = " LIKE "
+ NOTCONTAINSOP = " NOT LIKE "
+ FIELDS = {
+ "UID": "ADDRESSBOOK_OBJECT.VCARD_UID",
+ }
+
+ def __init__(self, expr, addressbookid):
+ self.RESOURCEDB = "ADDRESSBOOK_OBJECT"
+
+ super(postgresqladbkgenerator, self).__init__(expr, addressbookid)
+
+
+ def generate(self):
+ """
+ Generate the actual SQL 'where ...' expression from the passed in
+ expression tree.
+
+ @return: a C{tuple} of (C{str}, C{list}), where the C{str} is the
+ partial SQL statement, and the C{list} is the list of argument
+ substitutions to use with the SQL API execute method.
+ """
+
+ # Init state
+ self.sout = StringIO.StringIO()
+ self.arguments = []
+ self.substitutions = []
+
+ # Generate ' where ...' partial statement
+ self.sout.write(self.WHERE)
+ self.generateExpression(self.expression)
+
+ # Prefix with ' from ...' partial statement
+ select = self.FROM + self.RESOURCEDB
+ select += self.sout.getvalue()
+
+ select = select % tuple(self.substitutions)
+
+ return select, self.arguments
+
+
+ def addArgument(self, arg):
+ self.arguments.append(arg)
+ self.substitutions.append("%s")
+ self.sout.write("%s")
+
+ def setArgument(self, arg):
+ self.arguments.append(arg)
+ self.substitutions.append("%s")
+
+ def frontArgument(self, arg):
+ self.arguments.insert(0, arg)
+ self.substitutions.insert(0, "%s")
+
+ def containsArgument(self, arg):
+ return "%%%s%%" % (arg,)
+
+
+class PostgresLegacyABIndexEmulator(object):
+ """
+ Emulator for L{twistedcaldv.index.Index} and
+ L{twistedcaldv.index.IndexSchedule}.
+ """
+
+ def __init__(self, addressbook):
+ self.resource = self.addressbook = addressbook
+ if (
+ hasattr(config, "Memcached") and
+ config.Memcached.Pools.Default.ClientEnabled
+ ):
+ self.reserver = MemcachedUIDReserver(self)
+ else:
+ # This is only used with unit tests
+ self.reserver = DummyUIDReserver(self)
+
+
+ @property
+ def _txn(self):
+ return self.addressbook._txn
+
+
+ def reserveUID(self, uid):
+ return self.reserver.reserveUID(uid)
+
+
+ def unreserveUID(self, uid):
+ return self.reserver.unreserveUID(uid)
+
+
+ def isReservedUID(self, uid):
+ return self.reserver.isReservedUID(uid)
+
+
+ def isAllowedUID(self, uid, *names):
+ """
+ Checks to see whether to allow an operation which would add the
+ specified UID to the index. Specifically, the operation may not
+ violate the constraint that UIDs must be unique.
+ @param uid: the UID to check
+ @param names: the names of resources being replaced or deleted by the
+ operation; UIDs associated with these resources are not checked.
+ @return: True if the UID is not in the index and is not reserved,
+ False otherwise.
+ """
+ rname = self.resourceNameForUID(uid)
+ return (rname is None or rname in names)
+
+
+ def resourceUIDForName(self, name):
+ obj = self.addressbook.addressbookObjectWithName(name)
+ if obj is None:
+ return None
+ return obj.uid()
+
+
+ def resourceNameForUID(self, uid):
+ obj = self.addressbook.addressbookObjectWithUID(uid)
+ if obj is None:
+ return None
+ return obj.name()
+
+
+ def whatchanged(self, revision):
+
+ results = [
+ (name.encode("utf-8"), deleted)
+ for name, deleted in
+ self._txn.execSQL(
+ """select RESOURCE_NAME, DELETED from ADDRESSBOOK_OBJECT_REVISIONS
+ where REVISION > %s and ADDRESSBOOK_RESOURCE_ID = %s""",
+ [revision, self.addressbook._resourceID],
+ )
+ ]
+ results.sort(key=lambda x:x[1])
+
+ changed = []
+ deleted = []
+ for name, wasdeleted in results:
+ if name:
+ if wasdeleted:
+ if revision:
+ deleted.append(name)
+ else:
+ changed.append(name)
+ else:
+ raise SyncTokenValidException
+
+ return changed, deleted,
+
+ def searchValid(self, filter):
+ if isinstance(filter, carddavxml.Filter):
+ qualifiers = addressbookquery.sqladdressbookquery(filter)
+ else:
+ qualifiers = None
+
+ return qualifiers is not None
+
+ def search(self, filter):
+ """
+ Finds resources matching the given qualifiers.
+ @param filter: the L{Filter} for the addressbook-query to execute.
+ @return: an iterable of tuples for each resource matching the
+ given C{qualifiers}. The tuples are C{(name, uid, type)}, where
+ C{name} is the resource name, C{uid} is the resource UID, and
+ C{type} is the resource iCalendar component type.x
+ """
+
+ # Make sure we have a proper Filter element and get the partial SQL statement to use.
+ if isinstance(filter, carddavxml.Filter):
+ qualifiers = addressbookquery.sqladdressbookquery(filter, self.addressbook._resourceID, generator=postgresqladbkgenerator)
+ else:
+ qualifiers = None
+ if qualifiers is not None:
+ rowiter = self._txn.execSQL(
+ "select DISTINCT ADDRESSBOOK_OBJECT.RESOURCE_NAME, ADDRESSBOOK_OBJECT.VCARD_UID" +
+ qualifiers[0],
+ qualifiers[1]
+ )
+ else:
+ rowiter = self._txn.execSQL(
+ "select RESOURCE_NAME, VCARD_UID from ADDRESSBOOK_OBJECT where ADDRESSBOOK_RESOURCE_ID = %s",
+ [self.addressbook._resourceID, ],
+ )
+
+ for row in rowiter:
+ yield row
+
+ def indexedSearch(self, filter, useruid='', fbtype=False):
+ """
+ Always raise L{IndexedSearchException}, since these indexes are not
+ fully implemented yet.
+ """
+ raise IndexedSearchException()
+
+
+ def bruteForceSearch(self):
+ return self._txn.execSQL(
+ "select RESOURCE_NAME, VCARD_UID from "
+ "ADDRESSBOOK_OBJECT where ADDRESSBOOK_RESOURCE_ID = %s",
+ [self.addressbook._resourceID]
+ )
+
+
+ def resourcesExist(self, names):
+ return list(set(names).intersection(
+ set(self.addressbook.listAddressbookObjects())))
+
+
+ def resourceExists(self, name):
+ return bool(
+ self._txn.execSQL(
+ "select RESOURCE_NAME from ADDRESSBOOK_OBJECT where "
+ "RESOURCE_NAME = %s and ADDRESSBOOK_RESOURCE_ID = %s",
+ [name, self.addressbook._resourceID]
+ )
+ )
+
+
+class PostgresLegacyABInvitesEmulator(object):
+ """
+ Emulator for the implicit interface specified by
+ L{twistedcaldav.sharing.InvitesDatabase}.
+ """
+
+
+ def __init__(self, addressbook):
+ self._addressbook = addressbook
+
+
+ @property
+ def _txn(self):
+ return self._addressbook._txn
+
+
+ def create(self):
+ "No-op, because the index implicitly always exists in the database."
+
+
+ def remove(self):
+ "No-op, because the index implicitly always exists in the database."
+
+
+ def allRecords(self):
+ for row in self._txn.execSQL(
+ """
+ select
+ INVITE.INVITE_UID, INVITE.NAME, INVITE.RECIPIENT_ADDRESS,
+ ADDRESSBOOK_HOME.OWNER_UID, ADDRESSBOOK_BIND.BIND_MODE,
+ ADDRESSBOOK_BIND.BIND_STATUS, ADDRESSBOOK_BIND.MESSAGE
+ from
+ INVITE, ADDRESSBOOK_HOME, ADDRESSBOOK_BIND
+ where
+ INVITE.RESOURCE_ID = %s and
+ INVITE.HOME_RESOURCE_ID =
+ ADDRESSBOOK_HOME.RESOURCE_ID and
+ ADDRESSBOOK_BIND.ADDRESSBOOK_RESOURCE_ID =
+ INVITE.RESOURCE_ID and
+ ADDRESSBOOK_BIND.ADDRESSBOOK_HOME_RESOURCE_ID =
+ INVITE.HOME_RESOURCE_ID
+ order by
+ INVITE.NAME asc
+ """, [self._addressbook._resourceID]):
+ [inviteuid, common_name, userid, ownerUID,
+ bindMode, bindStatus, summary] = row
+ # FIXME: this is really the responsibility of the protocol layer.
+ state = {
+ _BIND_STATUS_INVITED: "NEEDS-ACTION",
+ _BIND_STATUS_ACCEPTED: "ACCEPTED",
+ _BIND_STATUS_DECLINED: "DECLINED",
+ _BIND_STATUS_INVALID: "INVALID",
+ }[bindStatus]
+ access = {
+ _BIND_MODE_READ: "read-only",
+ _BIND_MODE_WRITE: "read-write"
+ }[bindMode]
+ principalURL = "/principals/__uids__/%s/" % (ownerUID,)
+ yield Invite(
+ inviteuid, userid, principalURL, common_name,
+ access, state, summary
+ )
+
+
+ def recordForUserID(self, userid):
+ for record in self.allRecords():
+ if record.userid == userid:
+ return record
+
+
+ def recordForPrincipalURL(self, principalURL):
+ for record in self.allRecords():
+ if record.principalURL == principalURL:
+ return record
+
+
+ def recordForInviteUID(self, inviteUID):
+ for record in self.allRecords():
+ if record.inviteuid == inviteUID:
+ return record
+
+
+ def addOrUpdateRecord(self, record):
+ bindMode = {'read-only': _BIND_MODE_READ,
+ 'read-write': _BIND_MODE_WRITE}[record.access]
+ bindStatus = {
+ "NEEDS-ACTION": _BIND_STATUS_INVITED,
+ "ACCEPTED": _BIND_STATUS_ACCEPTED,
+ "DECLINED": _BIND_STATUS_DECLINED,
+ "INVALID": _BIND_STATUS_INVALID,
+ }[record.state]
+ # principalURL is derived from a directory record's principalURL() so
+ # it will always contain the UID. The form is '/principals/__uids__/x'
+ # (and may contain a trailing slash).
+ principalUID = record.principalURL.split("/")[3]
+ shareeHome = self._txn.addressbookHomeWithUID(principalUID, create=True)
+ rows = self._txn.execSQL(
+ "select RESOURCE_ID, HOME_RESOURCE_ID from INVITE where RECIPIENT_ADDRESS = %s",
+ [record.userid]
+ )
+ if rows:
+ [[resourceID, homeResourceID]] = rows
+ # Invite(inviteuid, userid, principalURL, common_name, access, state, summary)
+ self._txn.execSQL("""
+ update ADDRESSBOOK_BIND set BIND_MODE = %s,
+ BIND_STATUS = %s, MESSAGE = %s
+ where
+ ADDRESSBOOK_RESOURCE_ID = %s and
+ ADDRESSBOOK_HOME_RESOURCE_ID = %s
+ """, [bindMode, bindStatus, record.summary,
+ resourceID, homeResourceID])
+ self._txn.execSQL("""
+ update INVITE set NAME = %s, INVITE_UID = %s
+ where RECIPIENT_ADDRESS = %s
+ """,
+ [record.name, record.inviteuid, record.userid]
+ )
+ else:
+ self._txn.execSQL(
+ """
+ insert into INVITE (
+ INVITE_UID, NAME,
+ HOME_RESOURCE_ID, RESOURCE_ID,
+ RECIPIENT_ADDRESS
+ )
+ values (%s, %s, %s, %s, %s)
+ """,
+ [
+ record.inviteuid, record.name,
+ shareeHome._resourceID, self._addressbook._resourceID,
+ record.userid
+ ])
+ self._txn.execSQL(
+ """
+ insert into ADDRESSBOOK_BIND
+ (
+ ADDRESSBOOK_HOME_RESOURCE_ID, ADDRESSBOOK_RESOURCE_ID,
+ ADDRESSBOOK_RESOURCE_NAME, BIND_MODE, BIND_STATUS,
+ SEEN_BY_OWNER, SEEN_BY_SHAREE, MESSAGE
+ )
+ values (%s, %s, %s, %s, %s, %s, %s, %s)
+ """,
+ [
+ shareeHome._resourceID,
+ self._addressbook._resourceID,
+ None, # this is NULL because it is not bound yet, let's be
+ # explicit about that.
+ bindMode,
+ bindStatus,
+ False,
+ False,
+ record.summary
+ ])
+
+
+ def removeRecordForUserID(self, userid):
+ rec = self.recordForUserID(userid)
+ self.removeRecordForInviteUID(rec.inviteuid)
+
+
+ def removeRecordForPrincipalURL(self, principalURL):
+ raise NotImplementedError("removeRecordForPrincipalURL")
+
+
+ def removeRecordForInviteUID(self, inviteUID):
+ rows = self._txn.execSQL("""
+ select HOME_RESOURCE_ID, RESOURCE_ID from INVITE where
+ INVITE_UID = %s
+ """, [inviteUID])
+ if rows:
+ [[homeID, resourceID]] = rows
+ self._txn.execSQL(
+ "delete from ADDRESSBOOK_BIND where "
+ "ADDRESSBOOK_HOME_RESOURCE_ID = %s and ADDRESSBOOK_RESOURCE_ID = %s",
+ [homeID, resourceID])
+ self._txn.execSQL("delete from INVITE where INVITE_UID = %s",
+ [inviteUID])
+
+
+
+class PostgresLegacyABSharesEmulator(object):
+
+ def __init__(self, home):
+ self._home = home
+
+
+ @property
+ def _txn(self):
+ return self._home._txn
+
+
+ def create(self):
+ pass
+
+
+ def remove(self):
+ pass
+
+
+ def allRecords(self):
+ # This should have been a smart join that got all these columns at
+ # once, but let's not bother to fix it, since the actual query we
+ # _want_ to do (just look for addressbook binds in a particular homes) is
+ # much simpler anyway; we should just do that.
+ shareRows = self._txn.execSQL(
+ """
+ select ADDRESSBOOK_RESOURCE_ID, ADDRESSBOOK_RESOURCE_NAME, MESSAGE
+ from ADDRESSBOOK_BIND
+ where ADDRESSBOOK_HOME_RESOURCE_ID = %s and
+ BIND_MODE != %s and
+ ADDRESSBOOK_RESOURCE_NAME is not null
+ """, [self._home._resourceID, _BIND_MODE_OWN])
+ for resourceID, resourceName, summary in shareRows:
+ [[shareuid]] = self._txn.execSQL(
+ """
+ select INVITE_UID
+ from INVITE
+ where RESOURCE_ID = %s and HOME_RESOURCE_ID = %s
+ """, [resourceID, self._home._resourceID])
+ sharetype = 'I'
+ [[ownerHomeID, ownerResourceName]] = self._txn.execSQL(
+ """
+ select ADDRESSBOOK_HOME_RESOURCE_ID, ADDRESSBOOK_RESOURCE_NAME
+ from ADDRESSBOOK_BIND
+ where ADDRESSBOOK_RESOURCE_ID = %s and
+ BIND_MODE = %s
+ """, [resourceID, _BIND_MODE_OWN]
+ )
+ [[ownerUID]] = self._txn.execSQL(
+ "select OWNER_UID from ADDRESSBOOK_HOME where RESOURCE_ID = %s",
+ [ownerHomeID])
+ hosturl = '/addressbooks/__uids__/%s/%s' % (
+ ownerUID, ownerResourceName
+ )
+ localname = resourceName
+ record = SharedCollectionRecord(
+ shareuid, sharetype, hosturl, localname, summary
+ )
+ yield record
+
+
+ def _search(self, **kw):
+ [[key, value]] = kw.items()
+ for record in self.allRecords():
+ if getattr(record, key) == value:
+ return record
+
+ def recordForLocalName(self, localname):
+ return self._search(localname=localname)
+
+ def recordForShareUID(self, shareUID):
+ return self._search(shareuid=shareUID)
+
+
+ def addOrUpdateRecord(self, record):
+# print '*** SHARING***: Adding or updating this record:'
+# import pprint
+# pprint.pprint(record.__dict__)
+ # record.hosturl -> /addressbooks/__uids__/<uid>/<addressbookname>
+ splithost = record.hosturl.split('/')
+ ownerUID = splithost[3]
+ ownerAddressBookName = splithost[4]
+ ownerHome = self._txn.addressbookHomeWithUID(ownerUID)
+ ownerAddressBook = ownerHome.addressbookWithName(ownerAddressBookName)
+ addressbookResourceID = ownerAddressBook._resourceID
+
+ # There needs to be a bind already, one that corresponds to the
+ # invitation. The invitation's UID is the same as the share UID. I
+ # just need to update its 'localname', i.e.
+ # ADDRESSBOOK_BIND.ADDRESSBOOK_RESOURCE_NAME.
+
+ self._txn.execSQL(
+ """
+ update ADDRESSBOOK_BIND set ADDRESSBOOK_RESOURCE_NAME = %s
+ where ADDRESSBOOK_HOME_RESOURCE_ID = %s and ADDRESSBOOK_RESOURCE_ID = %s
+ """,
+ [record.localname, self._home._resourceID, addressbookResourceID]
+ )
+
+
+ def removeRecordForLocalName(self, localname):
+ self._txn.execSQL(
+ "delete from ADDRESSBOOK_BIND where ADDRESSBOOK_RESOURCE_NAME = %s "
+ "and ADDRESSBOOK_HOME_RESOURCE_ID = %s",
+ [localname, self._home._resourceID]
+ )
+
+
+ def removeRecordForShareUID(self, shareUID):
+ pass
+# c = self._home._cursor()
+# c.execute(
+# "delete from ADDRESSBOOK_BIND where ADDRESSBOOK_RESOURCE_NAME = %s "
+# "and ADDRESSBOOK_HOME_RESOURCE_ID = %s",
+# [self._home._resourceID]
+# )
Copied: CalendarServer/trunk/txdav/common/datastore/sql_schema_v1.sql (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/common/datastore/sql_schema_v1.sql)
===================================================================
--- CalendarServer/trunk/txdav/common/datastore/sql_schema_v1.sql (rev 0)
+++ CalendarServer/trunk/txdav/common/datastore/sql_schema_v1.sql 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,342 @@
+-----------------
+-- Resource ID --
+-----------------
+
+create sequence RESOURCE_ID_SEQ;
+
+
+-------------------
+-- Calendar Home --
+-------------------
+
+create table CALENDAR_HOME (
+ RESOURCE_ID integer primary key default nextval('RESOURCE_ID_SEQ'),
+ OWNER_UID varchar(255) not null unique
+);
+
+
+--------------
+-- Calendar --
+--------------
+
+create table CALENDAR (
+ RESOURCE_ID integer primary key default nextval('RESOURCE_ID_SEQ'),
+ REVISION integer default 0,
+ CREATED timestamp default timezone('UTC', CURRENT_TIMESTAMP),
+ MODIFIED timestamp default timezone('UTC', CURRENT_TIMESTAMP)
+);
+
+
+------------------------
+-- Sharing Invitation --
+------------------------
+
+create table INVITE (
+ INVITE_UID varchar(255) not null,
+ NAME varchar(255) not null,
+ RECIPIENT_ADDRESS varchar(255) not null,
+ HOME_RESOURCE_ID integer not null,
+ RESOURCE_ID integer not null
+);
+
+
+---------------------------
+-- Sharing Notifications --
+---------------------------
+
+create table NOTIFICATION_HOME (
+ RESOURCE_ID integer primary key default nextval('RESOURCE_ID_SEQ'),
+ OWNER_UID varchar(255) not null unique
+);
+
+
+create table NOTIFICATION (
+ RESOURCE_ID integer primary key default nextval('RESOURCE_ID_SEQ'),
+ NOTIFICATION_HOME_RESOURCE_ID integer not null references NOTIFICATION_HOME,
+ NOTIFICATION_UID varchar(255) not null,
+ XML_TYPE varchar not null,
+ XML_DATA varchar not null,
+ CREATED timestamp default timezone('UTC', CURRENT_TIMESTAMP),
+ MODIFIED timestamp default timezone('UTC', CURRENT_TIMESTAMP),
+
+ unique(NOTIFICATION_UID, NOTIFICATION_HOME_RESOURCE_ID)
+);
+
+
+-------------------
+-- Calendar Bind --
+-------------------
+
+-- Joins CALENDAR_HOME and CALENDAR
+
+create table CALENDAR_BIND (
+ CALENDAR_HOME_RESOURCE_ID integer not null references CALENDAR_HOME,
+ CALENDAR_RESOURCE_ID integer not null references CALENDAR on delete cascade,
+
+ -- An invitation which hasn't been accepted yet will not yet have a resource
+ -- name, so this field may be null.
+
+ CALENDAR_RESOURCE_NAME varchar(255),
+ BIND_MODE integer not null, -- enum CALENDAR_BIND_MODE
+ BIND_STATUS integer not null, -- enum CALENDAR_BIND_STATUS
+ SEEN_BY_OWNER boolean not null,
+ SEEN_BY_SHAREE boolean not null,
+ MESSAGE text,
+
+ primary key(CALENDAR_HOME_RESOURCE_ID, CALENDAR_RESOURCE_ID),
+ unique(CALENDAR_HOME_RESOURCE_ID, CALENDAR_RESOURCE_NAME)
+);
+
+-- Enumeration of calendar bind modes
+
+create table CALENDAR_BIND_MODE (
+ ID integer primary key,
+ DESCRIPTION varchar(16) not null unique
+);
+
+insert into CALENDAR_BIND_MODE values (0, 'own' );
+insert into CALENDAR_BIND_MODE values (1, 'read' );
+insert into CALENDAR_BIND_MODE values (2, 'write');
+
+-- Enumeration of statuses
+
+create table CALENDAR_BIND_STATUS (
+ ID integer primary key,
+ DESCRIPTION varchar(16) not null unique
+);
+
+insert into CALENDAR_BIND_STATUS values (0, 'invited' );
+insert into CALENDAR_BIND_STATUS values (1, 'accepted');
+insert into CALENDAR_BIND_STATUS values (2, 'declined');
+insert into CALENDAR_BIND_STATUS values (3, 'invalid');
+
+
+---------------------
+-- Calendar Object --
+---------------------
+
+create table CALENDAR_OBJECT (
+ RESOURCE_ID integer primary key default nextval('RESOURCE_ID_SEQ'),
+ CALENDAR_RESOURCE_ID integer not null references CALENDAR on delete cascade,
+ RESOURCE_NAME varchar(255) not null,
+ ICALENDAR_TEXT text not null,
+ ICALENDAR_UID varchar(255) not null,
+ ICALENDAR_TYPE varchar(255) not null,
+ ATTACHMENTS_MODE integer not null, -- enum CALENDAR_OBJECT_ATTACHMENTS_MODE
+ ORGANIZER varchar(255),
+ ORGANIZER_OBJECT integer references CALENDAR_OBJECT,
+ RECURRANCE_MAX date, -- maximum date that recurrences have been expanded to.
+ CREATED timestamp default timezone('UTC', CURRENT_TIMESTAMP),
+ MODIFIED timestamp default timezone('UTC', CURRENT_TIMESTAMP),
+
+ unique(CALENDAR_RESOURCE_ID, RESOURCE_NAME)
+
+ -- since the 'inbox' is a 'calendar resource' for the purpose of storing
+ -- calendar objects, this constraint has to be selectively enforced by the
+ -- application layer.
+
+ -- unique(CALENDAR_RESOURCE_ID, ICALENDAR_UID)
+);
+
+-- Enumeration of attachment modes
+
+create table CALENDAR_OBJECT_ATTACHMENTS_MODE (
+ ID integer primary key,
+ DESCRIPTION varchar(16) not null unique
+);
+
+insert into CALENDAR_OBJECT_ATTACHMENTS_MODE values (0, 'read' );
+insert into CALENDAR_OBJECT_ATTACHMENTS_MODE values (1, 'write');
+
+
+-----------------
+-- Instance ID --
+-----------------
+
+create sequence INSTANCE_ID_SEQ;
+
+
+----------------
+-- Time Range --
+----------------
+
+create table TIME_RANGE (
+ INSTANCE_ID integer primary key default nextval('INSTANCE_ID_SEQ'),
+ CALENDAR_RESOURCE_ID integer not null references CALENDAR on delete cascade,
+ CALENDAR_OBJECT_RESOURCE_ID integer not null references CALENDAR_OBJECT on delete cascade,
+ FLOATING boolean not null,
+ START_DATE timestamp not null,
+ END_DATE timestamp not null,
+ FBTYPE integer not null,
+ TRANSPARENT boolean not null
+);
+
+-- Enumeration of free/busy types
+
+create table FREE_BUSY_TYPE (
+ ID integer primary key,
+ DESCRIPTION varchar(16) not null unique
+);
+
+insert into FREE_BUSY_TYPE values (0, 'unknown' );
+insert into FREE_BUSY_TYPE values (1, 'free' );
+insert into FREE_BUSY_TYPE values (2, 'busy' );
+insert into FREE_BUSY_TYPE values (3, 'busy-unavailable');
+insert into FREE_BUSY_TYPE values (4, 'busy-tentative' );
+
+
+------------------
+-- Transparency --
+------------------
+
+create table TRANSPARENCY (
+ TIME_RANGE_INSTANCE_ID integer not null references TIME_RANGE on delete cascade,
+ USER_ID varchar(255) not null,
+ TRANSPARENT boolean not null
+);
+
+
+----------------
+-- Attachment --
+----------------
+
+create table ATTACHMENT (
+ CALENDAR_OBJECT_RESOURCE_ID integer not null references CALENDAR_OBJECT on delete cascade,
+ CONTENT_TYPE varchar(255) not null,
+ SIZE integer not null,
+ MD5 char(32) not null,
+ CREATED timestamp default timezone('UTC', CURRENT_TIMESTAMP),
+ MODIFIED timestamp default timezone('UTC', CURRENT_TIMESTAMP),
+ PATH varchar(1024) not null unique
+);
+
+
+------------------------------
+-- Calendar Object Revision --
+------------------------------
+
+create sequence CALENDAR_OBJECT_REVISION_SEQ;
+
+
+-------------------------------
+-- Calendar Object Revisions --
+-------------------------------
+
+create table CALENDAR_OBJECT_REVISIONS (
+ CALENDAR_RESOURCE_ID integer not null references CALENDAR on delete cascade,
+ RESOURCE_NAME varchar(255) not null,
+ REVISION integer not null,
+ DELETED boolean not null,
+
+ unique(CALENDAR_RESOURCE_ID, RESOURCE_NAME)
+);
+
+
+------------------
+-- iTIP Message --
+------------------
+
+create table ITIP_MESSAGE (
+ CALENDAR_RESOURCE_ID integer not null references CALENDAR,
+ ICALENDAR_TEXT text not null,
+ ICALENDAR_UID varchar(255) not null,
+ MD5 char(32) not null,
+ CHANGES text not null
+);
+
+
+-----------------------
+-- Resource Property --
+-----------------------
+
+create table RESOURCE_PROPERTY (
+ RESOURCE_ID integer not null, -- foreign key: *.RESOURCE_ID
+ NAME varchar(255) not null,
+ VALUE text not null, -- FIXME: xml?
+ VIEWER_UID varchar(255),
+
+ primary key(RESOURCE_ID, NAME, VIEWER_UID)
+);
+
+
+----------------------
+-- AddressBook Home --
+----------------------
+
+create table ADDRESSBOOK_HOME (
+ RESOURCE_ID integer primary key default nextval('RESOURCE_ID_SEQ'),
+ OWNER_UID varchar(255) not null unique
+);
+
+
+-----------------
+-- AddressBook --
+-----------------
+
+create table ADDRESSBOOK (
+ RESOURCE_ID integer primary key default nextval('RESOURCE_ID_SEQ'),
+ REVISION integer default 0,
+ CREATED timestamp default timezone('UTC', CURRENT_TIMESTAMP),
+ MODIFIED timestamp default timezone('UTC', CURRENT_TIMESTAMP)
+);
+
+
+----------------------
+-- AddressBook Bind --
+----------------------
+
+-- Joins ADDRESSBOOK_HOME and ADDRESSBOOK
+
+create table ADDRESSBOOK_BIND (
+ ADDRESSBOOK_HOME_RESOURCE_ID integer not null references ADDRESSBOOK_HOME,
+ ADDRESSBOOK_RESOURCE_ID integer not null references ADDRESSBOOK on delete cascade,
+
+ -- An invitation which hasn't been accepted yet will not yet have a resource
+ -- name, so this field may be null.
+
+ ADDRESSBOOK_RESOURCE_NAME varchar(255),
+ BIND_MODE integer not null, -- enum CALENDAR_BIND_MODE
+ BIND_STATUS integer not null, -- enum CALENDAR_BIND_STATUS
+ SEEN_BY_OWNER boolean not null,
+ SEEN_BY_SHAREE boolean not null,
+ MESSAGE text, -- FIXME: xml?
+
+ primary key(ADDRESSBOOK_HOME_RESOURCE_ID, ADDRESSBOOK_RESOURCE_ID),
+ unique(ADDRESSBOOK_HOME_RESOURCE_ID, ADDRESSBOOK_RESOURCE_NAME)
+);
+
+
+create table ADDRESSBOOK_OBJECT (
+ RESOURCE_ID integer primary key default nextval('RESOURCE_ID_SEQ'),
+ ADDRESSBOOK_RESOURCE_ID integer not null references ADDRESSBOOK on delete cascade,
+ RESOURCE_NAME varchar(255) not null,
+ VCARD_TEXT text not null,
+ VCARD_UID varchar(255) not null,
+ CREATED timestamp default timezone('UTC', CURRENT_TIMESTAMP),
+ MODIFIED timestamp default timezone('UTC', CURRENT_TIMESTAMP),
+
+ unique(ADDRESSBOOK_RESOURCE_ID, RESOURCE_NAME),
+ unique(ADDRESSBOOK_RESOURCE_ID, VCARD_UID)
+);
+
+------------------------------
+-- AddressBook Object Revision --
+------------------------------
+
+create sequence ADDRESSBOOK_OBJECT_REVISION_SEQ;
+
+
+-------------------------------
+-- AddressBook Object Revisions --
+-------------------------------
+
+create table ADDRESSBOOK_OBJECT_REVISIONS (
+ ADDRESSBOOK_RESOURCE_ID integer not null references ADDRESSBOOK on delete cascade,
+ RESOURCE_NAME varchar(255) not null,
+ REVISION integer not null,
+ DELETED boolean not null,
+
+ unique(ADDRESSBOOK_RESOURCE_ID, RESOURCE_NAME)
+);
+
+
Copied: CalendarServer/trunk/txdav/common/datastore/sql_tables.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/common/datastore/sql_tables.py)
===================================================================
--- CalendarServer/trunk/txdav/common/datastore/sql_tables.py (rev 0)
+++ CalendarServer/trunk/txdav/common/datastore/sql_tables.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,132 @@
+# -*- test-case-name: txdav.caldav.datastore.test.test_sql -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+SQL Table definitions.
+"""
+
+CALENDAR_HOME_TABLE = {
+ "name" : "CALENDAR_HOME",
+ "column_RESOURCE_ID" : "RESOURCE_ID",
+ "column_OWNER_UID" : "OWNER_UID",
+}
+
+ADDRESSBOOK_HOME_TABLE = {
+ "name" : "ADDRESSBOOK_HOME",
+ "column_RESOURCE_ID" : "RESOURCE_ID",
+ "column_OWNER_UID" : "OWNER_UID",
+}
+
+NOTIFICATION_HOME_TABLE = {
+ "name" : "NOTIFICATION_HOME",
+ "column_RESOURCE_ID" : "RESOURCE_ID",
+ "column_OWNER_UID" : "OWNER_UID",
+}
+
+CALENDAR_TABLE = {
+ "name" : "CALENDAR",
+ "column_RESOURCE_ID" : "RESOURCE_ID",
+ "column_REVISION" : "REVISION",
+ "column_CREATED" : "CREATED",
+ "column_MODIFIED" : "MODIFIED",
+ "sequence" : "CALENDAR_OBJECT_REVISION_SEQ",
+}
+
+ADDRESSBOOK_TABLE = {
+ "name" : "ADDRESSBOOK",
+ "column_RESOURCE_ID" : "RESOURCE_ID",
+ "column_REVISION" : "REVISION",
+ "column_CREATED" : "CREATED",
+ "column_MODIFIED" : "MODIFIED",
+ "sequence" : "ADDRESSBOOK_OBJECT_REVISION_SEQ",
+}
+
+CALENDAR_BIND_TABLE = {
+ "name" : "CALENDAR_BIND",
+ "column_HOME_RESOURCE_ID" : "CALENDAR_HOME_RESOURCE_ID",
+ "column_RESOURCE_ID" : "CALENDAR_RESOURCE_ID",
+ "column_RESOURCE_NAME" : "CALENDAR_RESOURCE_NAME",
+ "column_BIND_MODE" : "BIND_MODE",
+ "column_BIND_STATUS" : "BIND_STATUS",
+ "column_SEEN_BY_OWNER" : "SEEN_BY_OWNER",
+ "column_SEEN_BY_SHAREE" : "SEEN_BY_SHAREE",
+ "column_MESSAGE" : "MESSAGE",
+}
+
+ADDRESSBOOK_BIND_TABLE = {
+ "name" : "ADDRESSBOOK_BIND",
+ "column_HOME_RESOURCE_ID" : "ADDRESSBOOK_HOME_RESOURCE_ID",
+ "column_RESOURCE_ID" : "ADDRESSBOOK_RESOURCE_ID",
+ "column_RESOURCE_NAME" : "ADDRESSBOOK_RESOURCE_NAME",
+ "column_BIND_MODE" : "BIND_MODE",
+ "column_BIND_STATUS" : "BIND_STATUS",
+ "column_SEEN_BY_OWNER" : "SEEN_BY_OWNER",
+ "column_SEEN_BY_SHAREE" : "SEEN_BY_SHAREE",
+ "column_MESSAGE" : "MESSAGE",
+}
+
+CALENDAR_OBJECT_REVISIONS_TABLE = {
+ "name" : "CALENDAR_OBJECT_REVISIONS",
+ "column_RESOURCE_ID" : "CALENDAR_RESOURCE_ID",
+ "column_RESOURCE_NAME" : "RESOURCE_NAME",
+ "column_REVISION" : "REVISION",
+ "column_DELETED" : "DELETED",
+}
+
+ADDRESSBOOK_OBJECT_REVISIONS_TABLE = {
+ "name" : "ADDRESSBOOK_OBJECT_REVISIONS",
+ "column_RESOURCE_ID" : "ADDRESSBOOK_RESOURCE_ID",
+ "column_RESOURCE_NAME" : "RESOURCE_NAME",
+ "column_REVISION" : "REVISION",
+ "column_DELETED" : "DELETED",
+}
+
+CALENDAR_OBJECT_TABLE = {
+ "name" : "CALENDAR_OBJECT",
+ "column_RESOURCE_ID" : "RESOURCE_ID",
+ "column_PARENT_RESOURCE_ID" : "CALENDAR_RESOURCE_ID",
+ "column_RESOURCE_NAME" : "RESOURCE_NAME",
+ "column_TEXT" : "ICALENDAR_TEXT",
+ "column_UID" : "ICALENDAR_UID",
+ "column_CREATED" : "CREATED",
+ "column_MODIFIED" : "MODIFIED",
+}
+
+ADDRESSBOOK_OBJECT_TABLE = {
+ "name" : "ADDRESSBOOK_OBJECT",
+ "column_RESOURCE_ID" : "RESOURCE_ID",
+ "column_PARENT_RESOURCE_ID" : "ADDRESSBOOK_RESOURCE_ID",
+ "column_RESOURCE_NAME" : "RESOURCE_NAME",
+ "column_TEXT" : "VCARD_TEXT",
+ "column_UID" : "VCARD_UID",
+ "column_CREATED" : "CREATED",
+ "column_MODIFIED" : "MODIFIED",
+}
+
+
+# Various constants
+
+_BIND_STATUS_INVITED = 0
+_BIND_STATUS_ACCEPTED = 1
+_BIND_STATUS_DECLINED = 2
+_BIND_STATUS_INVALID = 3
+
+_ATTACHMENTS_MODE_WRITE = 1
+
+_BIND_MODE_OWN = 0
+_BIND_MODE_READ = 1
+_BIND_MODE_WRITE = 2
Deleted: CalendarServer/trunk/txdav/common/datastore/test/__init__.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/common/datastore/test/__init__.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/common/datastore/test/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,20 +0,0 @@
-# -*- test-case-name: txdav.carddav.datastore.test -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Store tests
-"""
Copied: CalendarServer/trunk/txdav/common/datastore/test/__init__.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/common/datastore/test/__init__.py)
===================================================================
--- CalendarServer/trunk/txdav/common/datastore/test/__init__.py (rev 0)
+++ CalendarServer/trunk/txdav/common/datastore/test/__init__.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,20 @@
+# -*- test-case-name: txdav.carddav.datastore.test -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Store tests
+"""
Deleted: CalendarServer/trunk/txdav/common/datastore/test/util.py
===================================================================
--- CalendarServer/branches/generic-sqlstore/txdav/common/datastore/test/util.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/common/datastore/test/util.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -1,139 +0,0 @@
-# -*- test-case-name: txdav.carddav.datastore.test -*-
-##
-# Copyright (c) 2010 Apple Inc. All rights reserved.
-#
-# Licensed under the Apache License, Version 2.0 (the "License");
-# you may not use this file except in compliance with the License.
-# You may obtain a copy of the License at
-#
-# http://www.apache.org/licenses/LICENSE-2.0
-#
-# Unless required by applicable law or agreed to in writing, software
-# distributed under the License is distributed on an "AS IS" BASIS,
-# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-# See the License for the specific language governing permissions and
-# limitations under the License.
-##
-
-"""
-Store test utility functions
-"""
-
-import gc
-
-from twext.python.filepath import CachingFilePath
-
-from twisted.internet import reactor
-from twisted.internet.defer import Deferred, succeed
-from twisted.internet.task import deferLater
-from twisted.python import log
-
-from txdav.common.datastore.sql import CommonDataStore, v1_schema
-from txdav.base.datastore.subpostgres import PostgresService,\
- DiagnosticConnectionWrapper
-
-def allInstancesOf(cls):
- for o in gc.get_referrers(cls):
- if isinstance(o, cls):
- yield o
-
-
-
-def dumpConnectionStatus():
- print '+++ ALL CONNECTIONS +++'
- for connection in allInstancesOf(DiagnosticConnectionWrapper):
- print connection.label, connection.state
- print '--- CONNECTIONS END ---'
-
-class SQLStoreBuilder(object):
- """
- Test-fixture-builder which can construct a PostgresStore.
- """
- sharedService = None
- currentTestID = None
-
- SHARED_DB_PATH = "../_test_sql_db"
-
- def buildStore(self, testCase, notifierFactory):
- """
- Do the necessary work to build a store for a particular test case.
-
- @return: a L{Deferred} which fires with an L{IDataStore}.
- """
- currentTestID = testCase.id()
- dbRoot = CachingFilePath(self.SHARED_DB_PATH)
- if self.sharedService is None:
- ready = Deferred()
- def getReady(connectionFactory):
- attachmentRoot = dbRoot.child("attachments")
- try:
- attachmentRoot.createDirectory()
- except OSError:
- pass
- try:
- self.store = CommonDataStore(
- lambda label=None: connectionFactory(
- label or currentTestID
- ),
- notifierFactory,
- attachmentRoot
- )
- except:
- ready.errback()
- raise
- else:
- self.cleanDatabase(testCase)
- ready.callback(self.store)
- return self.store
- self.sharedService = PostgresService(
- dbRoot, getReady, v1_schema, "caldav", resetSchema=True,
- testMode=True
- )
- self.sharedService.startService()
- def startStopping():
- log.msg("Starting stopping.")
- self.sharedService.unpauseMonitor()
- return self.sharedService.stopService()
- reactor.addSystemEventTrigger(#@UndefinedVariable
- "before", "shutdown", startStopping)
- result = ready
- else:
- self.store.notifierFactory = notifierFactory
- self.cleanDatabase(testCase)
- result = succeed(self.store)
-
- def cleanUp():
- # FIXME: clean up any leaked connections and report them with an
- # immediate test failure.
- def stopit():
- self.sharedService.pauseMonitor()
- return deferLater(reactor, 0.1, stopit)
- testCase.addCleanup(cleanUp)
- return result
-
-
- def cleanDatabase(self, testCase):
- cleanupConn = self.store.connectionFactory(
- "%s schema-cleanup" % (testCase.id(),)
- )
- cursor = cleanupConn.cursor()
- tables = ['INVITE',
- 'RESOURCE_PROPERTY',
- 'ATTACHMENT',
- 'ADDRESSBOOK_OBJECT',
- 'CALENDAR_OBJECT',
- 'CALENDAR_BIND',
- 'ADDRESSBOOK_BIND',
- 'CALENDAR',
- 'ADDRESSBOOK',
- 'CALENDAR_HOME',
- 'ADDRESSBOOK_HOME',
- 'NOTIFICATION',
- 'NOTIFICATION_HOME']
- for table in tables:
- try:
- cursor.execute("delete from "+table)
- except:
- log.err()
- cleanupConn.commit()
- cleanupConn.close()
Copied: CalendarServer/trunk/txdav/common/datastore/test/util.py (from rev 6191, CalendarServer/branches/generic-sqlstore/txdav/common/datastore/test/util.py)
===================================================================
--- CalendarServer/trunk/txdav/common/datastore/test/util.py (rev 0)
+++ CalendarServer/trunk/txdav/common/datastore/test/util.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -0,0 +1,139 @@
+# -*- test-case-name: txdav.carddav.datastore.test -*-
+##
+# Copyright (c) 2010 Apple Inc. All rights reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+##
+
+"""
+Store test utility functions
+"""
+
+import gc
+
+from twext.python.filepath import CachingFilePath
+
+from twisted.internet import reactor
+from twisted.internet.defer import Deferred, succeed
+from twisted.internet.task import deferLater
+from twisted.python import log
+
+from txdav.common.datastore.sql import CommonDataStore, v1_schema
+from txdav.base.datastore.subpostgres import PostgresService,\
+ DiagnosticConnectionWrapper
+
+def allInstancesOf(cls):
+ for o in gc.get_referrers(cls):
+ if isinstance(o, cls):
+ yield o
+
+
+
+def dumpConnectionStatus():
+ print '+++ ALL CONNECTIONS +++'
+ for connection in allInstancesOf(DiagnosticConnectionWrapper):
+ print connection.label, connection.state
+ print '--- CONNECTIONS END ---'
+
+class SQLStoreBuilder(object):
+ """
+ Test-fixture-builder which can construct a PostgresStore.
+ """
+ sharedService = None
+ currentTestID = None
+
+ SHARED_DB_PATH = "../_test_sql_db"
+
+ def buildStore(self, testCase, notifierFactory):
+ """
+ Do the necessary work to build a store for a particular test case.
+
+ @return: a L{Deferred} which fires with an L{IDataStore}.
+ """
+ currentTestID = testCase.id()
+ dbRoot = CachingFilePath(self.SHARED_DB_PATH)
+ if self.sharedService is None:
+ ready = Deferred()
+ def getReady(connectionFactory):
+ attachmentRoot = dbRoot.child("attachments")
+ try:
+ attachmentRoot.createDirectory()
+ except OSError:
+ pass
+ try:
+ self.store = CommonDataStore(
+ lambda label=None: connectionFactory(
+ label or currentTestID
+ ),
+ notifierFactory,
+ attachmentRoot
+ )
+ except:
+ ready.errback()
+ raise
+ else:
+ self.cleanDatabase(testCase)
+ ready.callback(self.store)
+ return self.store
+ self.sharedService = PostgresService(
+ dbRoot, getReady, v1_schema, "caldav", resetSchema=True,
+ testMode=True
+ )
+ self.sharedService.startService()
+ def startStopping():
+ log.msg("Starting stopping.")
+ self.sharedService.unpauseMonitor()
+ return self.sharedService.stopService()
+ reactor.addSystemEventTrigger(#@UndefinedVariable
+ "before", "shutdown", startStopping)
+ result = ready
+ else:
+ self.store.notifierFactory = notifierFactory
+ self.cleanDatabase(testCase)
+ result = succeed(self.store)
+
+ def cleanUp():
+ # FIXME: clean up any leaked connections and report them with an
+ # immediate test failure.
+ def stopit():
+ self.sharedService.pauseMonitor()
+ return deferLater(reactor, 0.1, stopit)
+ testCase.addCleanup(cleanUp)
+ return result
+
+
+ def cleanDatabase(self, testCase):
+ cleanupConn = self.store.connectionFactory(
+ "%s schema-cleanup" % (testCase.id(),)
+ )
+ cursor = cleanupConn.cursor()
+ tables = ['INVITE',
+ 'RESOURCE_PROPERTY',
+ 'ATTACHMENT',
+ 'ADDRESSBOOK_OBJECT',
+ 'CALENDAR_OBJECT',
+ 'CALENDAR_BIND',
+ 'ADDRESSBOOK_BIND',
+ 'CALENDAR',
+ 'ADDRESSBOOK',
+ 'CALENDAR_HOME',
+ 'ADDRESSBOOK_HOME',
+ 'NOTIFICATION',
+ 'NOTIFICATION_HOME']
+ for table in tables:
+ try:
+ cursor.execute("delete from "+table)
+ except:
+ log.err()
+ cleanupConn.commit()
+ cleanupConn.close()
Modified: CalendarServer/trunk/txdav/common/inotifications.py
===================================================================
--- CalendarServer/trunk/txdav/common/inotifications.py 2010-08-26 15:34:27 UTC (rev 6191)
+++ CalendarServer/trunk/txdav/common/inotifications.py 2010-08-26 16:47:28 UTC (rev 6192)
@@ -152,7 +152,7 @@
An notification object describes an XML notification.
"""
- def setData(uid, xmltype, xmldata):
+ def setData(uid, xmltype, xmldata, inserting=False):
"""
Rewrite this notification object to match the given C{xmltype} and
C{xmldata}. C{xmldata} must have the same UID as this notification object.
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://lists.macosforge.org/pipermail/calendarserver-changes/attachments/20100826/2aa2ce28/attachment-0001.html>
More information about the calendarserver-changes
mailing list