Boto3的KeyConditionExpression有效,但get_item无效。

huangapple go评论85阅读模式
英文:

Boto3 KeyConditionExpression works and get_item does not

问题

I am new to DynamoDB and finding my way around DocumentDBs versus Relational DBs.

My code is running via AWS Lambda functions with a Python runtime. It works.

  1. def get_item_by_id(table_name: str, partition_key: str, partition_value: str) -> List[dict]:
  2. """
  3. Retrieve items from DynamoDB table based on the partition key and its value.
  4. Args:
  5. table_name: The name of the DynamoDB table.
  6. partition_key: The partition key of the table.
  7. partition_value: The value of the partition key to search for.
  8. Returns:
  9. A list of items matching the partition key and value.
  10. Raises:
  11. boto3.exceptions.ResourceNotFoundException: If the DynamoDB table does not exist.
  12. """
  13. dynamodb = resource('dynamodb')
  14. table = dynamodb.Table(table_name)
  15. response = table.query(
  16. KeyConditionExpression=Key(partition_key).eq(partition_value)
  17. )
  18. items = response['Items']
  19. return items

However, it took me a couple of hours to find from boto3.dynamodb.conditions import Key in the official documentation.

Instead, the official documentation "code examples" refers to a function called get_item():

  1. response = table.get_item(
  2. Key={
  3. 'username': 'janedoe',
  4. 'last_name': 'Doe'
  5. }
  6. )
  7. item = response['Item']
  8. print(item)

Below was my version

  1. def get_item_by_id(table_name: str, item_id: str) -> Optional[Dict[str, Any]]:
  2. """
  3. Retrieve an item from DynamoDB table based on the item ID.
  4. Args:
  5. table_name: The name of the DynamoDB table.
  6. item_id: The ID of the item to retrieve.
  7. Returns:
  8. The retrieved item as a dictionary, or None if the item is not found.
  9. Raises:
  10. boto3.exceptions.ResourceNotFoundException: If the DynamoDB table does not exist.
  11. """
  12. dynamodb = boto3.resource('dynamodb')
  13. table = dynamodb.Table(table_name)
  14. response = table.get_item(Key={'ID': item_id})
  15. item = response.get('Item')
  16. return item

This was never able to return a successful response, this was the error:

> "errorMessage": "An error occurred (ValidationException) when calling the GetItem operation: The provided key element does not match the schema"

Can I ask:

  1. Why does KeyConditionExpression work with identical table names, attribute name and value?
  2. Could I have ever got get_item() working?

Thank you.

英文:

I am new to DynamoDB and finding my way around DocumentDBs versus Relational DBs.

My code is running via AWS Lambda functions with a Python runtime. It works.

  1. def get_item_by_id(table_name: str, partition_key: str, partition_value: str) -> List[dict]:
  2. """
  3. Retrieve items from DynamoDB table based on the partition key and its value.
  4. Args:
  5. table_name: The name of the DynamoDB table.
  6. partition_key: The partition key of the table.
  7. partition_value: The value of the partition key to search for.
  8. Returns:
  9. A list of items matching the partition key and value.
  10. Raises:
  11. boto3.exceptions.ResourceNotFoundException: If the DynamoDB table does not exist.
  12. """
  13. dynamodb = resource('dynamodb')
  14. table = dynamodb.Table(table_name)
  15. response = table.query(
  16. KeyConditionExpression=Key(partition_key).eq(partition_value)
  17. )
  18. items = response['Items']
  19. return items

However, it took me a couple of hours to find from boto3.dynamodb.conditions import Key in the official documentation.

Instead, the official documentation "code examples" refers to a function called get_item():

  1. response = table.get_item(
  2. Key={
  3. 'username': 'janedoe',
  4. 'last_name': 'Doe'
  5. }
  6. )
  7. item = response['Item']
  8. print(item)

Below was my version

  1. def get_item_by_id(table_name: str, item_id: str) -> Optional[Dict[str, Any]]:
  2. """
  3. Retrieve an item from DynamoDB table based on the item ID.
  4. Args:
  5. table_name: The name of the DynamoDB table.
  6. item_id: The ID of the item to retrieve.
  7. Returns:
  8. The retrieved item as a dictionary, or None if the item is not found.
  9. Raises:
  10. boto3.exceptions.ResourceNotFoundException: If the DynamoDB table does not exist.
  11. """
  12. dynamodb = boto3.resource('dynamodb')
  13. table = dynamodb.Table(table_name)
  14. response = table.get_item(Key={'ID': item_id})
  15. item = response.get('Item')
  16. return item

This was never able to return a successful response, this was the error:

> "errorMessage": "An error occurred (ValidationException) when calling the GetItem operation: The provided key element does not match the schema"

Can I ask:

  1. Why does KeyConditionExpression work with identical table names, attribute name and value?
  2. Could I have ever got get_item() working?

Thank you.

答案1

得分: 1

你正在尝试比较两个不同的API。

GetItem

此方法针对表中的单个项目,您必须提供完整的主键,即分区键和排序键。

您之所以看到异常是因为您没有提供排序键的名称或值。

Query

此方法使用KeyConditionExpression并针对多个具有相同分区键的项目。此方法只需要您提供分区键,您可以选择排除排序键,或在排序键上使用条件函数,例如between

文档

我强烈建议您花几个小时仔细阅读DynamoDB文档,这将对您的理解产生巨大的影响,而不是浪费时间编写不起作用的代码。

英文:

You are trying to compare 2 different APIs.

GetItem

This method targets a single item in a table, and you must provide the entire primary key which is partition and sort key.

You are seeing the exception as you do not provide the sort key name or value.

Query

This method uses a KeyConditionExpression and targets multiple items which share the same partition key. This method only requires you to provide the partition key, and you can decide to exclude the sort key, or use a conditional function on the sort key such as between.

Documentation

I would strongly advise that you take a couple of hours and read the DynamoDB documentation thoroughly, it'll make a great difference to you understanding instead of wasting hours writing code which doesn't work.

答案2

得分: 0

从我看到的情况:

KeyConditionExpression=Key(partition_key).eq(partition_value)

在数据类型上更加宽松,而另一方面

response = table.get_item(Key={'ID': item_id})

对于您在DynamoDB中设置的数据方案要求更为严格。如果ID是数值而不是字符串,item_id: str 看起来可疑。参见此处的类型映射

英文:

From what I've seen:

KeyConditionExpression=Key(partition_key).eq(partition_value)

is more relaxed on data types while on the other side

response = table.get_item(Key={'ID': item_id})

is strict on the data scheme you setup with dynamodb. item_id: str looks suspect if the ID is a numeric value instead of a string. See the type mappings here

huangapple
  • 本文由 发表于 2023年5月11日 00:54:58
  • 转载请务必保留本文链接:https://go.coder-hub.com/76220904.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定