- Remove use of
ThreadPool
objects. Threads were used to implement async HTTP requests, but were creating massive memory leaks. Async requests should be reimplemented using a real async HTTP request package, so this is just an emergency fix. This also lowers the defaultProtocol.SESSION_POOLSIZE
to 1 because no internal code is running multi-threaded anymore. - All-day calendar items (created as
CalendarItem(is_all_day=True, ...)
) now acceptEWSDate
instances for thestart
andend
values. Similarly, all-day calendar items fetched from the server now returnstart
andend
values asEWSDate
instances. In this case, start and end values are inclusive; a one-day event starts and ends on the sameEWSDate
value. - Add support for
RecurringMasterItemId
andOccurrenceItemId
elements that allow to request the master recurrence from aCalendarItem
occurrence, and to request a specific occurrence from aCalendarItem
master recurrence.CalendarItem.master_recurrence()
andCalendarItem.occurrence(some_occurrence_index)
methods were added to aid this traversal.some_occurrence_index
in the last method specifies which item in the list of occurrences to target;CalendarItem.occurrence(3)
gets the third occurrence in the recurrence. - Change
Contact.birthday
andContact.wedding_anniversary
fromEWSDateTime
toEWSDate
fields. EWS still expects and sends datetime values but has started to reset the time part to 11:59. Dates are a better match for these two fields anyway. - Remove support for
len(some_queryset)
. It had the nasty side-effect of forcinglist(some_queryset)
to run the query twice, once for pre-allocating the list via the result oflen(some_queryset)
, and then once more to fetch the results. All occurrences oflen(some_queryset)
can be replaced withsome_queryset.count()
. Unfortunately, there is no way to keep backwards-compatibility for this feature. - Added
Account.identity
, an attribute to contain extra information for impersonation. SettingAccount.identity.upn
orAccount.identity.sid
removes the need for an AD lookup on every request.upn
will often be the same asprimary_smtp_address
, but it is not guaranteed. If you have access to your organization's AD servers, you can look up these values once and add them to yourAccount
object to improve performance of the following requests. - Added support for CBA authentication
- The
max_wait
argument toFaultTolerance
changed semantics. Previously, it triggered when the delay until the next attempt would exceed this value. It now triggers after the given timespan since the first request attempt. - Fixed a bug when pagination is combined with
max_items
(#710) - Other minor bug fixes
- Removed the legacy autodiscover implementation.
- Added
QuerySet.depth()
to configure item traversal of querysets. Default isShallow
except for theCommonViews
folder where default isAssociated
. - Updating credentials on
Account.protocol
after getting anUnauthorizedError
now works.
- The new Autodiscover implementation added in 2.2.0 is now default. To switch back to the old
implementation, set the environment variable
EXCHANGELIB_AUTODISCOVER_VERSION=legacy
. - Removed support for Python 2
- Added support for specifying a separate retry policy for the autodiscover service endpoint
selection. Set via the
exchangelib.autodiscover.legacy.INITIAL_RETRY_POLICY
module variable for the the old autodiscover implementation, and via theexchangelib.autodiscover.Autodiscovery.INITIAL_RETRY_POLICY
class variable for the new one. - Support the authorization code OAuth 2.0 grant type (see issue #698)
- Removed the
RootOfHierarchy.permission_set
field. It was causing too many failures in the wild. - The full autodiscover response containing all contents of the reponse is now available as
Account.ad_response
. - Added a new Autodiscover implementation that is closer to the specification and easier to debug. To switch
to the new implementation, set the environment variable
EXCHANGELIB_AUTODISCOVER_VERSION=new
. The old one is still the default if the variable is not set, or set toEXCHANGELIB_AUTODISCOVER_VERSION=legacy
. - The
Item.mime_content
field was switched back from a string type to abytes
type. It turns out trying to decode the data was an error (see issue #709).
- Bugfix release.
- Added support for OAuth 2.0 authentication
- Fixed a bug in
RelativeMonthlyPattern
andRelativeYearlyPattern
where theweekdays
field was thought to be a list, but is in fact a single value. Renamed the field toweekday
to reflect the change. - Added support for archiving items to the archive mailbox, if the account has one.
- Added support for getting delegate information on an Account, as
Account.delegates
. - Added support for the
ConvertId
service. Available asProtocol.convert_ids()
.
- Fixed a bug where version 2.x could not open autodiscover cache files generated by version 1.x packages.
-
Item.mime_content
is now a text field instead of a binary field. Encoding and decoding is done automatically. -
The
Item.item_id
,Folder.folder_id
andOccurrence.item_id
fields that were renamed to justid
in 1.12.0, have now been removed. -
The
Persona.persona_id
field was replaced withPersona.id
andPersona.changekey
, to align with theItem
andFolder
classes. -
In addition to bulk deleting via a QuerySet (
qs.delete()
), it is now possible to also bulk send, move and copy items in a QuerySet (viaqs.send()
,qs.move()
andqs.copy()
, respectively). -
SSPI support was added but dependencies are not installed by default since it only works in Win32 environments. Install as
pip install exchangelib[sspi]
to get SSPI support. Install withpip install exchangelib[complete]
to get both Kerberos and SSPI auth. -
The custom
extern_id
field is no longer registered by default. If you require this field, register it manually as part of your setup code on the item types you need:from exchangelib import CalendarItem, Message, Contact, Task from exchangelib.extended_properties import ExternId CalendarItem.register('extern_id', ExternId) Message.register('extern_id', ExternId) Contact.register('extern_id', ExternId) Task.register('extern_id', ExternId)
-
The
ServiceAccount
class has been removed. If you want fault tolerance, set it in aConfiguration
object:from exchangelib import Configuration, Credentials, FaultTolerance c = Credentials('foo', 'bar') config = Configuration(credentials=c, retry_policy=FaultTolerance())
-
It is now possible to use Kerberos and SSPI auth without providing a dummy
Credentials('', '')
object. -
The
has_ssl
argument ofConfiguration
was removed. If you want to connect to a plain HTTP endpoint, pass the full URL in theservice_endpoint
argument. -
We no longer look in
types.xsd
for a hint of which API version the server is running. Instead, we query the service directly, starting with the latest version first.
- Bugfix release.
- Fix bug that left out parts of the folder hierarchy when traversing
account.root
. - Fix bug that did not properly find all attachments if an item has a mix of item and file attachments.
- Add support for reading and writing
PermissionSet
field on folders. - Add support for Exchange 2019 build IDs.
- Add
Protocol.expand_dl()
to get members of a distribution list.
- Lower the session pool size automatically in response to ErrorServerBusy and ErrorTooManyObjectsOpened errors from the server.
- Unusual slicing and indexing (e.g.
inbox.all()[9000]
andinbox.all()[9000:9001]
) is now efficient. - Downloading large attachments is now more memory-efficient. We can now stream the file
content without ever storing the full file content in memory, using the new
Attachment.fp
context manager.
- Add a MAINFEST.in to ensure the LICENSE file gets included + CHANGELOG.md and README.md to sdist tarball
- Renamed
Item.item_id
,Folder.folder_id
andOccurrence.item_id
to justItem.id
,Folder.id
andOccurrence.id
, respectively. This removes redundancy in the naming and provides consistency. For all classes that have an ID, the ID can now be accessed using theid
attribute. Backwards compatibility and deprecation warnings were added. - Support folder traversal without creating a full cache of the folder
hierarchy first, using the
some_folder // 'sub_folder' // 'leaf'
(double-slash) syntax. - Fix a bug in traversal of public and archive folders. These folder hierarchies are now fully supported.
- Fix a bug where the timezone of a calendar item changed when the item was fetched and then saved.
- Kerberos support is now optional and Kerberos dependencies are not
installed by default. Install as
pip install exchangelib[kerberos]
to get Kerberos support.
- Improve back off handling when receiving
ErrorServerBusy
error messages from the server - Fixed bug where
Account.root
and its children would point to the root folder of the connecting account instead of the target account when connecting to other accounts.
- Add experimental Kerberos support. This adds the
pykerberos
package, which needs the following system packages to be installed on Ubuntu/Debian systems:apt-get install build-essential libssl-dev libffi-dev python-dev libkrb5-dev
.
- Bugfix release
- Bugfix release
-
Added
cancel
toCalendarItem
andCancelCalendarItem
class to allow cancelling meetings that were set up -
Added
accept
,decline
andtentatively_accept
toCalendarItem
as wrapper methods -
Added
accept
,decline
andtentatively_accept
toMeetingRequest
to respond to incoming invitations -
Added
BaseMeetingItem
(inheriting fromItem
) being used as base for MeetingCancellation, MeetingMessage, MeetingRequest and MeetingResponse -
Added
AssociatedCalendarItemId
(property),AssociatedCalendarItemIdField
andReferenceItemIdField
-
Added
PostReplyItem
-
Removed
Folder.get_folder_by_name()
which has been deprecated since version1.10.2
. -
Added
Item.copy(to_folder=some_folder)
method which copies an item to the given folder and returns the ID of the new item. -
We now respect the back off value of an
ErrorServerBusy
server error. -
Added support for fetching free/busy availability information ofr a list of accounts.
-
Added
Message.reply()
,Message.reply_all()
, andMessage.forward()
methods. -
The full search API now works on single folders and collections of folders, e.g.
some_folder.glob('foo*').filter()
,some_folder.children.filter()
andsome_folder.walk().filter()
. -
Deprecated
EWSService.CHUNKSIZE
in favor of a per-request chunk_size available onAccount.bulk_foo()
methods. -
Support searching the GAL and other contact folders using
some_contact_folder.people()
. -
Deprecated the
page_size
argument forQuerySet.iterator()
because it was inconsistent with other API methods. You can still set the page size of a queryset like this:qs = a.inbox.filter(...).iterator() qs.page_size = 123 for item in items: print(item)
- Added support for registering extended properties on folders.
- Added support for creating, updating, deleting and emptying folders.
- Added support for getting and setting
Account.oof_settings
using the newOofSettings
class. - Added snake_case named shortcuts to all distinguished folders on
the
Account
model. E.g.Account.search_folders
.
- Bugfix release
- Added support for most item fields. The remaining ones are mentioned in issue #203.
- Added an
exchangelib.util.PrettyXmlHandler
log handler which will pretty-print and highlight XML requests and responses.
- Greatly improved folder navigation. See the 'Folders' section in the README
- Added deprecation warnings for
Account.folders
andFolder.get_folder_by_name()
- Bugfix release
-
Removed the
verify_ssl
argument toAccount
,discover
andConfiguration
. If you need to disable TLS verification, register a customHTTPAdapter
class. A sample adapter class is provided for convenience:from exchangelib.protocol import BaseProtocol, NoVerifyHTTPAdapter BaseProtocol.HTTP_ADAPTER_CLS = NoVerifyHTTPAdapter
- Support new Office365 build numbers
- Added support for the
effective_rights
field on items and folders. - Added support for custom
requests
transport adapters, to allow proxy support, custom TLS validation etc. - Default value for the
affected_task_occurrences
argument toItem.move_to_trash()
,Item.soft_delete()
andItem.delete()
was changed to'AllOccurrences'
as a less surprising default when working with simple tasks. - Added
Task.complete()
helper method to mark tasks as complete.
- Added minimal support for the
PostItem
item type - Added support for the
DistributionList
item type - Added support for receiving naive datetimes from the server. They
will be localized using the new
default_timezone
attribute onAccount
- Added experimental support for recurring calendar items. See examples in issue #37.
- Improved support for
filter()
,.only()
,.order_by()
etc. on indexed properties. It is now possible to specify labels and subfields, e.g..filter(phone_numbers=PhoneNumber(label='CarPhone', phone_number='123'))
.filter(phone_numbers__CarPhone='123')
,.filter(physical_addresses__Home__street='Elm St. 123')
, .only('physical_addresses__Home__street')` etc. - Improved performance of
.order_by()
when sorting on multiple fields. - Implemented QueryString search. You can now filter using an EWS
QueryString, e.g.
filter('subject:XXX')
- Added
EWSTimeZone.localzone()
to get the local timezone - Support
some_folder.get(item_id=..., changekey=...)
as a shortcut to get a single item when you know the ID and changekey. - Support attachments on Exchange 2007
- Fixed XML generation for Exchange 2010 and other picky server versions
- Fixed timezone localization for
EWSTimeZone
created from a static timezone
- Expand support for
ExtendedProperty
to include all possible attributes. This required renaming theproperty_id
attribute toproperty_set_id
. - When using the
Credentials
class,UnauthorizedError
is now raised if the credentials are wrong. - Add a new
version
attribute toConfiguration
, to force the server version if version guessing does not work. Accepts aexchangelib.version.Version
object. - Rework bulk operations
Account.bulk_foo()
andAccount.fetch()
to return some exceptions unraised, if it is deemed the exception does not apply to all items. This means that e.g.fetch()
can return a mix of`Item
andErrorItemNotFound
instances, if only some of the requestedItemId
were valid. Other exceptions will be raised immediately, e.g.ErrorNonExistentMailbox
because the exception applies to all items. It is the responsibility of the caller to check the type of the returned values. - The
Folder
class has new attributestotal_count
,unread_count
andchild_folder_count
, and arefresh()
method to update these values. - The argument to
Account.upload()
was renamed fromupload_data
to justdata
- Support for using a string search expression for
Folder.filter()
was removed. It was a cool idea but using QuerySet chaining andQ
objects is even cooler and provides the same functionality, and more. - Add support for
reminder_due_by
andreminder_minutes_before_start
fields onItem
objects. Submitted by@vikipha
. - Added a new
ServiceAccount
class which is likeCredentials
but does whatis_service_account
did before. If you need fault-tolerane and usedCredentials(..., is_service_account=True)
before, useServiceAccount
now. This also disables fault-tolerance for theCredentials
class, which is in line with what most users expected. - Added an optional
update_fields
attribute tosave()
to specify only some fields to be updated. - Code in in
folders.py
has been split into multiple files, and some classes will have new import locaions. The most commonly used classes have a shortcut in __init__.py - Added support for the
exists
lookup in filters, e.g.my_folder.filter(categories__exists=True|False)
to filter on the existence of that field on items in the folder. - When filtering,
foo__in=value
now requires the value to be a list, andfoo__contains
requires the value to be a list if the field itself is a list, e.g.categories__contains=['a', 'b']
. - Added support for fields and enum entries that are only supported in some EWS versions
- Added a new field
Item.text_body
which is a read-only version of HTML body content, where HTML tags are stripped by the server. Only supported from Exchange 2013 and up. - Added a new choice
WorkingElsewhere
to theCalendarItem.legacy_free_busy_status
enum. Only supported from Exchange 2013 and up.
- Fix completely botched
Message.from
field renaming in 1.8.0 - Improve performance of QuerySet slicing and indexing. For example,
account.inbox.all()[10]
andaccount.inbox.all()[:10]
now only fetch 10 items from the server even thoughaccount.inbox.all()
could contain thousands of messages.
-
Renamed
Message.from
field toMessage.author
.from
is a Python keyword sofrom
could only be accessed asGetattr(my_essage, 'from')
which is just stupid. -
Make
EWSTimeZone
Windows timezone name translation more robust -
Add read-only
Message.message_id
which holds the Internet Message Id -
Memory and speed improvements when sorting querysets using
order_by()
on a single field. -
Allow setting
Mailbox
andAttendee
-type attributes as plain strings, e.g.:calendar_item.organizer = '[email protected]' calendar_item.required_attendees = ['[email protected]', '[email protected]'] message.to_recipients = ['[email protected]', '[email protected]']
- Bugfix release
Account.fetch()
andFolder.fetch()
are now generators. They will do nothing before being evaluated.- Added optional
page_size
attribute toQuerySet.iterator()
to specify the number of items to return per HTTP request for large query results. Defaultpage_size
is 100. - Many minor changes to make queries less greedy and return earlier
- Add Python2 support
-
Implement attachments support. It's now possible to create, delete and get attachments connected to any item type:
from exchangelib.folders import FileAttachment, ItemAttachment # Process attachments on existing items for item in my_folder.all(): for attachment in item.attachments: local_path = os.path.join('/tmp', attachment.name) with open(local_path, 'wb') as f: f.write(attachment.content) print('Saved attachment to', local_path) # Create a new item with an attachment item = Message(...) binary_file_content = 'Hello from unicode æøå'.encode('utf-8') # Or read from file, BytesIO etc. my_file = FileAttachment(name='my_file.txt', content=binary_file_content) item.attach(my_file) my_calendar_item = CalendarItem(...) my_appointment = ItemAttachment(name='my_appointment', item=my_calendar_item) item.attach(my_appointment) item.save() # Add an attachment on an existing item my_other_file = FileAttachment(name='my_other_file.txt', content=binary_file_content) item.attach(my_other_file) # Remove the attachment again item.detach(my_file)
Be aware that adding and deleting attachments from items that are already created in Exchange (items that have an
item_id
) will update thechangekey
of the item. -
Implement
Item.headers
which contains custom Internet message headers. Primarily useful forMessage
objects. Read-only for now.
-
Implement the
Contact.physical_addresses
attribute. This is a list ofexchangelib.folders.PhysicalAddress
items. -
Implement the
CalendarItem.is_all_day
boolean to create all-day appointments. -
Implement
my_folder.export()
andmy_folder.upload()
. Thanks to @SamCB! -
Fixed
Account.folders
for non-distinguished folders -
Added
Folder.get_folder_by_name()
to make it easier to get sub-folders by name. -
Implement
CalendarView
searches asmy_calendar.view(start=..., end=...)
. A view differs from a normalfilter()
in that a view expands recurring items and returns recurring item occurrences that are valid in the time span of the view. -
Persistent storage location for autodiscover cache is now platform independent
-
Implemented custom extended properties. To add support for your own custom property, subclass
exchangelib.folders.ExtendedProperty
and callregister()
on the item class you want to use the extended property with. When you have registered your extended property, you can use it exactly like you would use any other attribute on this item type. If you change your mind, you can remove the extended property again withderegister()
:class LunchMenu(ExtendedProperty): property_id = '12345678-1234-1234-1234-123456781234' property_name = 'Catering from the cafeteria' property_type = 'String' CalendarItem.register('lunch_menu', LunchMenu) item = CalendarItem(..., lunch_menu='Foie gras et consommé de légumes') item.save() CalendarItem.deregister('lunch_menu')
-
Fixed a bug on folder items where an existing HTML body would be converted to text when calling
save()
. When creating or updating an item body, you can use the two new helper classesexchangelib.Body
andexchangelib.HTMLBody
to specify if your body should be saved as HTML or text. E.g.:item = CalendarItem(...) # Plain-text body item.body = Body('Hello UNIX-beard pine user!') # Also plain-text body, works as before item.body = 'Hello UNIX-beard pine user!' # Exchange will see this as an HTML body and display nicely in clients item.body = HTMLBody('<html><body>Hello happy <blink>OWA user!</blink></body></html>') item.save()
-
Fix bug where fetching items from a folder that can contain multiple item types (e.g. the Deleted Items folder) would only return one item type.
-
Added
Item.move(to_folder=...)
that moves an item to another folder, andItem.refresh()
that updates the Item with data from EWS. -
Support reverse sort on individual fields in
order_by()
, e.g.my_folder.all().order_by('subject', '-start')
-
Account.bulk_create()
was added to create items that don't need a folder, e.g.Message.send()
-
Account.fetch()
was added to fetch items without knowing the containing folder. -
Implemented
SendItem
service to send existing messages. -
Folder.bulk_delete()
was moved toAccount.bulk_delete()
-
Folder.bulk_update()
was moved toAccount.bulk_update()
and changed to expect a list of(Item, fieldnames)
tuples where Item is e.g. aMessage
instance andfieldnames
is a list of attributes names that need updating. E.g.:items = [] for i in range(4): item = Message(subject='Test %s' % i) items.append(item) account.sent.bulk_create(items=items) item_changes = [] for i, item in enumerate(items): item.subject = 'Changed subject' % i item_changes.append(item, ['subject']) account.bulk_update(items=item_changes)
- Added the
is_service_account
flag toCredentials
.is_service_account=False
disables the fault-tolerant error handling policy and enables immediate failures. Configuration
now expects a singlecredentials
attribute instead of separateusername
andpassword
attributes.- Added support for distinguished folders
Account.trash
,Account.drafts
,Account.outbox
,Account.sent
andAccount.junk
. - Renamed
Folder.find_items()
toFolder.filter()
- Renamed
Folder.add_items()
toFolder.bulk_create()
- Renamed
Folder.update_items()
toFolder.bulk_update()
- Renamed
Folder.delete_items()
toFolder.bulk_delete()
- Renamed
Folder.get_items()
toFolder.fetch()
- Made various policies for message saving, meeting invitation
sending, conflict resolution, task occurrences and deletion
available on
bulk_create()
,bulk_update()
andbulk_delete()
. - Added convenience methods
Item.save()
,Item.delete()
,Item.soft_delete()
,Item.move_to_trash()
, and methodsMessage.send()
andMessage.send_and_save()
that are specific toMessage
objects. These methods make it easier to create, update and delete single items. - Removed
fetch(.., with_extra=True)
in favor of the more fine-grainedfetch(.., only_fields=[...])
- Added a
QuerySet
class that supports QuerySet-returning methodsfilter()
,exclude()
,only()
,order_by()
,reverse()``values()
andvalues_list()
that all allow for chaining.QuerySet
also has methodsiterator()
,get()
,count()
,exists()
anddelete()
. All these methods behave like their counterparts in Django.
- Use of
my_folder.with_extra_fields = True
to get the extra fields inItem.EXTRA_ITEM_FIELDS
is deprecated (it was a kludge anyway). Instead, usemy_folder.get_items(ids, with_extra=[True, False])
. The default was also changed toTrue
, to avoid head-scratching with newcomers.
-
Simplify
Q
objects andRestriction.from_source()
by using Item attribute names in expressions and kwargs instead of EWS FieldURI values. ChangeFolder.find_items()
to accept either a search expression, or a list ofQ
objects just like Djangofilter()
does. E.g.:ids = account.calendar.find_items( "start < '2016-01-02T03:04:05T' and end > '2016-01-01T03:04:05T' and categories in ('foo', 'bar')", shape=IdOnly ) q1, q2 = (Q(subject__iexact='foo') | Q(subject__contains='bar')), ~Q(subject__startswith='baz') ids = account.calendar.find_items(q1, q2, shape=IdOnly)
-
Complete rewrite of
Folder.find_items()
. The oldstart
,end
,subject
andcategories
args are deprecated in favor of a Django QuerySet filter() syntax. The supported lookup types are__gt
,__lt
,__gte
,__lte
,__range
,__in
,__exact
,__iexact
,__contains
,__icontains
,__contains
,__icontains
,__startswith
,__istartswith
, plus an additional__not
which translates to!=
. Additionally, all fields on the item are now supported inFolder.find_items()
.WARNING: This change is backwards-incompatible! Old uses of
Folder.find_items()
like this:ids = account.calendar.find_items( start=tz.localize(EWSDateTime(year, month, day)), end=tz.localize(EWSDateTime(year, month, day + 1)), categories=['foo', 'bar'], )
must be rewritten like this:
ids = account.calendar.find_items( start__lt=tz.localize(EWSDateTime(year, month, day + 1)), end__gt=tz.localize(EWSDateTime(year, month, day)), categories__contains=['foo', 'bar'], )
failing to do so will most likely result in empty or wrong results.
-
Added a
exchangelib.restrictions.Q
class much like Django Q objects that can be used to create even more complex filtering. Q objects must be passed directly toexchangelib.services.FindItem
.
- Don't require sequence arguments to
Folder.*_items()
methods to supportlen()
(e.g. generators andmap
instances are now supported) - Allow empty sequences as argument to
Folder.*_items()
methods
- Add support for
required_attendees
,optional_attendees
andresources
attribute onfolders.CalendarItem
. These are implemented with a newfolders.Attendee
class.
- Add support for
organizer
attribute onCalendarItem
. Implemented with a newfolders.Mailbox
class.
- Initial import