Skip to content

Commit

Permalink
Merge remote-tracking branch 'origin/master' into move-to-botocore-ma…
Browse files Browse the repository at this point in the history
…ke-api-call
  • Loading branch information
ikonst committed Nov 25, 2023
2 parents b08fbb5 + c1d6222 commit aa9174e
Show file tree
Hide file tree
Showing 11 changed files with 236 additions and 39 deletions.
3 changes: 1 addition & 2 deletions docs/index.rst
Original file line number Diff line number Diff line change
Expand Up @@ -46,8 +46,7 @@ Topics
contributing
release_notes
versioning
upgrading_binary
upgrading_unicodeset
upgrading

API docs
========
Expand Down
1 change: 1 addition & 0 deletions docs/indexes.rst
Original file line number Diff line number Diff line change
Expand Up @@ -95,6 +95,7 @@ Local secondary indexes are defined just like global ones, but they inherit from
view = NumberAttribute(range_key=True)
Every local secondary index must meet the following conditions:

- The partition key (hash key) is the same as that of its base table.
- The sort key (range key) consists of exactly one scalar attribute. The range key can be any attribute.
- The sort key (range key) of the base table is projected into the index, where it acts as a non-key attribute.
Expand Down
84 changes: 52 additions & 32 deletions docs/release_notes.rst
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
.. _release_notes:

Release Notes
=============

Expand Down Expand Up @@ -102,8 +104,13 @@ v5.0.3

:date: 2021-02-14

This version has an unintentional breaking change:

* Propagate ``Model.serialize``'s ``null_check`` parameter to nested MapAttributes (:pr:`908`)

Previously null errors (persisting ``None`` into an attribute defined as ``null=False``)
were ignored for attributes in map attributes that were nested in maps or lists. After upgrade,
these will resulted in an :py:class:`~pynamodb.exceptions.AttributeNullError` being raised.

v5.0.2
----------
Expand All @@ -128,22 +135,32 @@ v5.0.0

This is major release and contains breaking changes. Please read the notes below carefully.

**Polymorphism**
Breaking changes
================

This release introduces :ref:`polymorphism` support via :py:class:`DiscriminatorAttribute <pynamodb.attributes.DiscriminatorAttribute>`.
Discriminator values are written to DynamoDB and used during deserialization to instantiate the desired class.
* Python 2 is no longer supported. Python 3.6 or greater is now required.
* :py:class:`~pynamodb.attributes.UnicodeAttribute` and :py:class:`~pynamodb.attributes.BinaryAttribute` now support empty values (:pr:`830`)

**UTCDateTimeAttribute**
In previous versions, assigning an empty value to would be akin to assigning ``None``: if the attribute was defined with ``null=True`` then it would be omitted, otherwise an error would be raised.

The UTCDateTimeAttribute now strictly requires the date string format '%Y-%m-%dT%H:%M:%S.%f%z' to ensure proper ordering.
PynamoDB has always written values with this format but previously would accept reading other formats.
Items written using other formats must be rewritten before upgrading.
As of May 2020, DynamoDB `supports <https://aws.amazon.com/about-aws/whats-new/2020/05/amazon-dynamodb-now-supports-empty-values-for-non-key-string-and-binary-attributes-in-dynamodb-tables/>`_ empty values for String and Binary attributes. This release of PynamoDB starts treating empty values like any other values. If existing code unintentionally assigns empty values to StringAttribute or BinaryAttribute, this may be a breaking change: for example, the code may rely on the fact that in previous versions empty strings would be "read back" as ``None`` values when reloaded from the database.
* :py:class:`~pynamodb.attributes.UTCDateTimeAttribute` now strictly requires the date string format ``'%Y-%m-%dT%H:%M:%S.%f%z'`` to ensure proper ordering.
PynamoDB has always written values with this format but previously would accept reading other formats.
Items written using other formats must be rewritten before upgrading.
* Table backup functionality (``Model.dump[s]`` and ``Model.load[s]``) has been removed.
* ``Model.query`` no longer converts unsupported range key conditions into filter conditions.
* Internal attribute type constants are replaced with their "short" DynamoDB version (:pr:`827`)
* Remove ``ListAttribute.remove_indexes`` (added in v4.3.2) and document usage of remove for list elements (:pr:`838`)
* Remove ``pynamodb.connection.util.pythonic`` (:pr:`753`) and (:pr:`865`)
* Remove ``ModelContextManager`` class (:pr:`861`)

**UnicodeAttribute and BinaryAttribute**
Features
========

In previous versions, assigning an empty value to a :py:class:`UnicodeAttribute <pynamodb.attributes.UnicodeAttribute>` or :py:class:`BinaryAttribute <pynamodb.attributes.BinaryAttribute>` would be akin to assigning ``None``: if the attribute was defined with ``null=True`` then it would be omitted, otherwise an error would be raised.
**Polymorphism**

As of May 2020, DynamoDB `supports <https://aws.amazon.com/about-aws/whats-new/2020/05/amazon-dynamodb-now-supports-empty-values-for-non-key-string-and-binary-attributes-in-dynamodb-tables/>`_ empty values for String and Binary attributes. This release of PynamoDB starts treating empty values like any other values. If existing code unintentionally assigns empty values to StringAttribute or BinaryAttribute, this may be a breaking change: for example, the code may rely on the fact that in previous versions empty strings would be "read back" as ``None`` values when reloaded from the database.
This release introduces :ref:`polymorphism` support via :py:class:`DiscriminatorAttribute <pynamodb.attributes.DiscriminatorAttribute>`.
Discriminator values are written to DynamoDB and used during deserialization to instantiate the desired class.

**Model Serialization**

Expand All @@ -152,22 +169,11 @@ THe ``Model`` class now includes public methods for serializing and deserializin

Other changes in this release:

* Python 2 is no longer supported. Python 3.6 or greater is now required.
* Table backup functionality (``Model.dump[s]`` and ``Model.load[s]``) has been removed.
* ``Model.query`` no longer demotes invalid range key conditions to be filter conditions to avoid surprising behaviors:
where what's intended to be a cheap and fast condition ends up being expensive and slow. Since filter conditions
cannot contain range keys, this had limited utility to begin with, and would sometimes cause confusing
"'filter_condition' cannot contain key attributes" errors.
* Replace the internal attribute type constants with their "short" DynamoDB version (:pr:`827`)
* Typed list attributes can now support any Attribute subclass (:pr:`833`)
* Add support for empty values in Binary and String attributes (:pr:`830`)
* Most API operation methods now accept a ``settings`` argument to customize settings of individual operations.
This currently allow adding or overriding HTTP headers. (:pr:`887`)
* Remove ``ListAttribute.remove_indexes`` (added in v4.3.2) and document usage of remove for list elements (:pr:`838`)
* Add the attribute name to error messages when deserialization fails (:pr:`815`)
* Add the table name to error messages for transactional operations (:pr:`835`)
* Remove ``pynamodb.connection.util.pythonic`` (:pr:`753`) and (:pr:`865`)
* Remove ``ModelContextManager`` class (:pr:`861`)

Contributors to this release:

Expand Down Expand Up @@ -322,20 +328,34 @@ of all attributes including the key attributes.
Support for `Legacy Conditional Parameters <https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/LegacyConditionalParameters.html>`_ has been
removed. See a complete list of affected ``Model`` methods below:

* ``update_item``: removed in favor of ``update``.
* ``rate_limited_scan``: removed in favor of ``scan`` and ``ResultIterator``.
.. list-table::
:widths: 10 90
:header-rows: 1

* - Method
- Changes
* - ``update_item``
- removed in favor of ``update``
* - ``rate_limited_scan``
- removed in favor of ``scan`` and ``ResultIterator``
* - ``delete``
- ``conditional_operator`` and ``**expected_values`` kwargs removed. Use ``condition`` instead.
* - ``update``
- ``attributes``, ``conditional_operator`` and ``**expected_values`` kwargs removed. Use ``actions`` and ``condition`` instead.
* - ``save``
- ``conditional_operator`` and ``**expected_values`` kwargs removed. Use ``condition`` instead.
* - ``count``
- ``**filters`` kwargs removed. Use ``range_key_condition``/``filter_condition`` instead.
* - ``query``
- ``conditional_operator`` and ``**filters`` kwargs removed. Use ``range_key_condition``/``filter_condition`` instead.
* - ``scan``
-
- ``conditional_operator`` and ``**filters`` kwargs removed. Use ``filter_condition`` instead.
- ``allow_rate_limited_scan_without_consumed_capacity`` was removed

+ Relatedly, the ``allow_rate_limited_scan_without_consumed_capacity`` option has been removed.
* ``delete``: ``conditional_operator`` and ``**expected_values`` kwargs removed. Use ``condition`` instead.
* ``update``: ``attributes``, ``conditional_operator`` and ``**expected_values`` kwargs removed. Use ``actions`` and ``condition`` instead.
* ``save``: ``conditional_operator`` and ``**expected_values`` kwargs removed. Use ``condition`` instead.
* ``count``: ``**filters`` kwargs removed. Use ``range_key_condition``/``filter_condition`` instead.
* ``query``: ``conditional_operator`` and ``**filters`` kwargs removed. Use ``range_key_condition``/``filter_condition`` instead.
* ``scan``: ``conditional_operator`` and ``**filters`` kwargs removed. Use ``filter_condition`` instead.

When upgrading, pay special attention to use of ``**filters`` and ``**expected_values``, as you'll need to check for arbitrary names that correspond to
attribute names. Also keep an eye out for kwargs like ``user_id__eq=5`` or ``email__null=True``, which are no longer supported. If you're not already using
``mypy`` to type check your code, it can help you catch cases like these.
attribute names. Also keep an eye out for kwargs like ``user_id__eq=5`` or ``email__null=True``, which are no longer supported. A type check can help you catch cases like these.

New features in this release:

Expand Down
140 changes: 140 additions & 0 deletions docs/upgrading.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,140 @@
Upgrading
=========

This file complements the :ref:`release notes <release_notes>`, focusing on helping safe upgrades of the library
in production scenarios.

PynamoDB 5.x to 6.x
-------------------

BinaryAttribute is no longer double base64-encoded
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

See :ref:`upgrading_binary` for details.

PynamoDB 4.x to 5.x
-------------------

Null checks enforced where they weren't previously
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

Previously null errors (persisting ``None`` into an attribute defined as ``null=False``) were ignored inside **nested** map attributes, e.g.

.. code-block:: python
from pynamodb.models import Model
from pynamodb.attributes import ListAttribute, MapAttribute, UnicodeAttribute
class Employee(MapAttribute):
name = UnicodeAttribute(null=False)
class Team(Model):
employees = ListAttribute(of=Employee)
team = Team()
team.employees = [Employee(name=None)]
team.save() # this will raise now
Now these will resulted in an :py:class:`~pynamodb.exceptions.AttributeNullError` being raised.

This was an unintentional breaking change introduced in 5.0.3.

Empty values are now meaningful
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

:py:class:`~pynamodb.attributes.UnicodeAttribute` and :py:class:`~pynamodb.attributes.BinaryAttribute` now support empty values (:pr:`830`)

In previous versions, assigning an empty value to would be akin to assigning ``None``: if the attribute was defined with ``null=True`` then it would be omitted, otherwise an error would be raised.
DynamoDB `added support <https://aws.amazon.com/about-aws/whats-new/2020/05/amazon-dynamodb-now-supports-empty-values-for-non-key-string-and-binary-attributes-in-dynamodb-tables/>`_ empty values
for String and Binary attributes. This release of PynamoDB starts treating empty values like any other values. If existing code unintentionally assigns empty values to StringAttribute or BinaryAttribute,
this may be a breaking change: for example, the code may rely on the fact that in previous versions empty strings would be "read back" as ``None`` values when reloaded from the database.

No longer parsing date-time strings leniently
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

:py:class:`~pynamodb.attributes.UTCDateTimeAttribute` now strictly requires the date string format ``'%Y-%m-%dT%H:%M:%S.%f%z'`` to ensure proper ordering.
PynamoDB has always written values with this format but previously would accept reading other formats.
Items written using other formats must be rewritten before upgrading.

Removed functionality
~~~~~~~~~~~~~~~~~~~~~

The following changes are breaking but are less likely to go unnoticed:

* Python 2 is no longer supported. Python 3.6 or greater is now required.
* Table backup functionality (``Model.dump[s]`` and ``Model.load[s]``) has been removed.
* ``Model.query`` no longer converts unsupported range key conditions into filter conditions.
* Internal attribute type constants are replaced with their "short" DynamoDB version (:pr:`827`)
* Remove ``ListAttribute.remove_indexes`` (added in v4.3.2) and document usage of remove for list elements (:pr:`838`)
* Remove ``pynamodb.connection.util.pythonic`` (:pr:`753`) and (:pr:`865`)
* Remove ``ModelContextManager`` class (:pr:`861`)

PynamoDB 3.x to 4.x
-------------------

Requests Removal
~~~~~~~~~~~~~~~~

Given that ``botocore`` has moved to using ``urllib3`` directly for making HTTP requests, we'll be doing the same (via ``botocore``). This means the following:

* The ``session_cls`` option is no longer supported.
* The ``request_timeout_seconds`` parameter is no longer supported. ``connect_timeout_seconds`` and ``read_timeout_seconds`` are available instead.

+ Note that the timeouts for connection and read are now ``15`` and ``30`` seconds respectively. This represents a change from the previous ``60`` second combined ``requests`` timeout.
* *Wrapped* exceptions (i.e ``exc.cause``) that were from ``requests.exceptions`` will now be comparable ones from ``botocore.exceptions`` instead.

Key attribute types must match table
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

The previous release would call `DescribeTable` to discover table metadata
and would use the key types as defined in the DynamoDB table. This could obscure
type mismatches e.g. where a table's hash key is a number (`N`) in DynamoDB,
but defined in PynamoDB as a `UnicodeAttribute`.

With this release, we're always using the PynamoDB model's definition
of all attributes including the key attributes.

Deprecation of old APIs
~~~~~~~~~~~~~~~~~~~~~~~

Support for `Legacy Conditional Parameters <https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/LegacyConditionalParameters.html>`_ has been
removed. See a complete list of affected ``Model`` methods below:

.. list-table::
:widths: 10 90
:header-rows: 1

* - Method
- Changes
* - ``update_item``
- removed in favor of ``update``
* - ``rate_limited_scan``
- removed in favor of ``scan`` and ``ResultIterator``
* - ``delete``
- ``conditional_operator`` and ``**expected_values`` kwargs removed. Use ``condition`` instead.
* - ``update``
- ``attributes``, ``conditional_operator`` and ``**expected_values`` kwargs removed. Use ``actions`` and ``condition`` instead.
* - ``save``
- ``conditional_operator`` and ``**expected_values`` kwargs removed. Use ``condition`` instead.
* - ``count``
- ``**filters`` kwargs removed. Use ``range_key_condition``/``filter_condition`` instead.
* - ``query``
- ``conditional_operator`` and ``**filters`` kwargs removed. Use ``range_key_condition``/``filter_condition`` instead.
* - ``scan``
-
- ``conditional_operator`` and ``**filters`` kwargs removed. Use ``filter_condition`` instead.
- ``allow_rate_limited_scan_without_consumed_capacity`` was removed


When upgrading, pay special attention to use of ``**filters`` and ``**expected_values``, as you'll need to check for arbitrary names that correspond to
attribute names. Also keep an eye out for kwargs like ``user_id__eq=5`` or ``email__null=True``, which are no longer supported. A type check can help you catch cases like these.

PynamoDB 2.x to 3.x
--------------------

Changes to UnicodeSetAttribute
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

See :ref:`upgrading_unicodeset` for details.
4 changes: 4 additions & 0 deletions docs/upgrading_binary.rst
Original file line number Diff line number Diff line change
@@ -1,3 +1,7 @@
:orphan:

.. _upgrading_binary:

Upgrading Binary(Set)Attribute
==============================

Expand Down
4 changes: 4 additions & 0 deletions docs/upgrading_unicodeset.rst
Original file line number Diff line number Diff line change
@@ -1,3 +1,7 @@
:orphan:

.. _upgrading_unicodeset:

Upgrading UnicodeSetAttribute
=============================

Expand Down
2 changes: 1 addition & 1 deletion pynamodb/attributes.py
Original file line number Diff line number Diff line change
Expand Up @@ -268,7 +268,7 @@ def prepend(self, other: Iterable) -> '_ListAppend':
def set(
self,
value: Union[_T, 'Attribute[_T]', '_Increment', '_Decrement', '_IfNotExists', '_ListAppend']
) -> 'SetAction':
) -> Union['SetAction', 'RemoveAction']:
return Path(self).set(value)

def remove(self) -> 'RemoveAction':
Expand Down
6 changes: 5 additions & 1 deletion pynamodb/exceptions.py
Original file line number Diff line number Diff line change
@@ -1,13 +1,17 @@
"""
PynamoDB exceptions
"""
import sys
from dataclasses import dataclass
from typing import Any
from typing import Dict
from typing import Iterable
from typing import List
from typing import Optional
from typing_extensions import Literal
if sys.version_info >= (3, 8):
from typing import Literal
else:
from typing_extensions import Literal

import botocore.exceptions

Expand Down
11 changes: 8 additions & 3 deletions pynamodb/expressions/operand.py
Original file line number Diff line number Diff line change
Expand Up @@ -298,9 +298,14 @@ def __getitem__(self, item: Union[int, str]) -> 'Path':
def __or__(self, other):
return _IfNotExists(self, self._to_operand(other))

def set(self, value: Any) -> SetAction:
# Returns an update action that sets this attribute to the given value
return SetAction(self, self._to_operand(value))
def set(self, value: Any) -> Union[SetAction, RemoveAction]:
# Returns an update action that sets this attribute to the given value.
# For attributes that may not be empty (e.g. sets), this may result
# in a remove action.
operand = self._to_operand(value)
if isinstance(operand, Value) and next(iter(operand.value.values())) is None:
return RemoveAction(self)
return SetAction(self, operand)

def remove(self) -> RemoveAction:
# Returns an update action that removes this attribute from the item
Expand Down
4 changes: 4 additions & 0 deletions tests/integration/model_integration_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -102,6 +102,10 @@ class Meta:
for item in TestModel.view_index.query('foo', TestModel.view > 0):
print("Item queried from index: {}".format(item.view))

query_obj.update([TestModel.scores.set([])])
query_obj.refresh()
assert query_obj.scores is None

print(query_obj.update([TestModel.view.add(1)], condition=TestModel.forum.exists()))
TestModel.delete_table()

Expand Down
16 changes: 16 additions & 0 deletions tests/test_expressions.py
Original file line number Diff line number Diff line change
Expand Up @@ -479,6 +479,13 @@ def test_set_action(self):
assert self.placeholder_names == {'foo': '#0'}
assert self.expression_attribute_values == {':0': {'S': 'bar'}}

def test_set_action_as_remove(self):
action = self.set_attribute.set([])
expression = action.serialize(self.placeholder_names, self.expression_attribute_values)
assert expression == "#0"
assert self.placeholder_names == {'foo_set': '#0'}
assert self.expression_attribute_values == {}

def test_set_action_attribute_container(self):
# Simulate initialization from inside an AttributeContainer
my_map_attribute = MapAttribute[str, str](attr_name='foo')
Expand Down Expand Up @@ -620,6 +627,15 @@ def test_update(self):
':2': {'NS': ['1']}
}

def test_update_set_to_empty(self):
update = Update(
self.set_attribute.set([]),
)
expression = update.serialize(self.placeholder_names, self.expression_attribute_values)
assert expression == "REMOVE #0"
assert self.placeholder_names == {'foo_set': '#0'}
assert self.expression_attribute_values == {}

def test_update_skips_empty_clauses(self):
update = Update(self.attribute.remove())
expression = update.serialize(self.placeholder_names, self.expression_attribute_values)
Expand Down

0 comments on commit aa9174e

Please sign in to comment.