将map [string] DynamoDBAttributeValue解组成一个struct

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了将map [string] DynamoDBAttributeValue解组成一个struct相关的知识,希望对你有一定的参考价值。

我正在尝试使用aws-sdk-go设置AWS-lambda,只要将新的user添加到某个dynamodb表中,就会触发它。

一切都工作得很好,但我找不到解决地图map[string]DynamoDBAttributeValue的方法:

{
    "name": {
        "S" : "John"
    },
    "residence_address": {
        "M": {
            "address": {
                "S": "some place"
            }
        }
    }
}

例如,对于给定的结构,User结构。 Here展示了一个将map[string]*dynamodb.AttributeValue解包到一个给定界面的例子,但我找不到用map[string]DynamoDBAttributeValue做同样事情的方法,即使这些类型似乎符合相同的目的。

map[string]DynamoDBAttributeValue由包装events.DynamoDBEventsgithub.com/aws/aws-lambda-go/events返回。这是我的代码:

package handler

import (
    "context"
    "github.com/aws/aws-lambda-go/events"
    "github.com/aws/aws-sdk-go/service/dynamodb/dynamodbattribute"
    "github.com/aws/aws-sdk-go/service/dynamodb"
)

func HandleDynamoDBRequest(ctx context.Context, e events.DynamoDBEvent) {

    for _, record := range e.Records {

        if record.EventName == "INSERT" {

            // User Struct
            var dynamoUser model.DynamoDBUser

            // Of course this can't be done for incompatible types
            _ := dynamodbattribute.UnmarshalMap(record.Change.NewImage, &dynamoUser)
        }

    }

}

当然,我可以将record.Change.NewImage编组为JSON并将其解组回给定的结构,但是,我必须从后者开始手动初始化dynamoUser属性。

或者我甚至可以编写一个将map[string]DynamoDBAttributeValue解析为map[string]*dynamodb.AttributeValue的函数,如:

func getAttributeValueMapFromDynamoDBStreamRecord(e events.DynamoDBStreamRecord) map[string]*dynamodb.AttributeValue {
    image := e.NewImage
    m := make(map[string]*dynamodb.AttributeValue)
    for k, v := range image {
        if v.DataType() == events.DataTypeString {
            s := v.String()
            m[k] = &dynamodb.AttributeValue{
                S : &s,
            }
        }
        if v.DataType() == events.DataTypeBoolean {
            b := v.Boolean()
            m[k] = &dynamodb.AttributeValue{
                BOOL : &b,
            }
        }
        // . . .
        if v.DataType() == events.DataTypeMap {
            // ?
        }
    }
    return m
}

然后简单地使用dynamodbattribute.UnmarshalMap,但在events.DataTypeMap上这将是一个非常棘手的过程。

有没有办法我可以将events.DynamoDBEvent的DynamoDB记录解组成一个结构,并使用map[string]*dynamodb.AttributeValue显示的类似方法?

答案

我尝试了你提供的功能,我遇到了events.DataTypeList的一些问题,所以我设法编写了以下功能:

// UnmarshalStreamImage converts events.DynamoDBAttributeValue to struct
func UnmarshalStreamImage(attribute map[string]events.DynamoDBAttributeValue, out interface{}) error {

    dbAttrMap := make(map[string]*dynamodb.AttributeValue)

    for k, v := range attribute {

        var dbAttr dynamodb.AttributeValue

        bytes, marshalErr := v.MarshalJSON(); if marshalErr != nil {
            return marshalErr
        }

        json.Unmarshal(bytes, &dbAttr)
        dbAttrMap[k] = &dbAttr
    }

    return dynamodbattribute.UnmarshalMap(dbAttrMap, out)

}
另一答案

目前我设法通过实现我在问题中提到的功能来解决这个问题。

// May take either `map[string]DynamoDBAttributeValue` 
// or `DynamoDBAttributeValue` as input.
func EventStreamToMap(attribute interface{}) map[string]*dynamodb.AttributeValue {

    // Map to be returned
    m := make(map[string]*dynamodb.AttributeValue)

    tmp := make(map[string]events.DynamoDBAttributeValue)

    switch t := attribute.(type) {
    case map[string]events.DynamoDBAttributeValue:
        tmp = t
    case events.DynamoDBAttributeValue:
        tmp = t.Map()
    }

    for k, v := range tmp {
        switch v.DataType() {
        case events.DataTypeString:
            s := v.String()
            m[k] = &dynamodb.AttributeValue{
                S : &s,
            }
        case events.DataTypeBoolean:
            b := v.Boolean()
            m[k] = &dynamodb.AttributeValue{
                BOOL : &b,
            }
        // case events.SomeOtherType:
        //     ...
        case events.DataTypeMap:
            m[k] = &dynamodb.AttributeValue{
                M : EventStreamToMap(v),
            }
        }
    }
    return m
}

但我仍然对其他实现持开放态度。

另一答案

我很沮丧,记录中的NewImage类型不是map [string] * dynamodb.AttributeValue所以我可以使用dynamodbattribute包。

events.DynamoDB AttributeValue的JSON表示似乎与dynamodb.AttributeValue的JSON表示相同。

所以我尝试创建自己的DynamoDBEvent类型并更改了OldImage和NewImage的类型,因此它将被编组到map [string] * dynamodb.AttributeValue而不是map [string] events.DynamoDBAttributeValue

它有点难看但它对我有用。

package main

import (
    "github.com/aws/aws-lambda-go/events"
    "github.com/aws/aws-lambda-go/lambda"
    "github.com/aws/aws-sdk-go/service/dynamodb"
    "github.com/aws/aws-sdk-go/service/dynamodb/dynamodbattribute"
    "fmt"
)

func main() {

    lambda.Start(lambdaHandler)
}

// changed type of event from: events.DynamoDBEvent to DynamoDBEvent (see below)
func lambdaHandler(event DynamoDBEvent) error {

    for _, record := range event.Records {

        change := record.Change
        newImage := change.NewImage // now of type: map[string]*dynamodb.AttributeValue

        var item IdOnly
        err := dynamodbattribute.UnmarshalMap(newImage, &item)
        if err != nil {
            return err
        }

        fmt.Println(item.Id)
    }

    return nil
}

type IdOnly struct {
    Id string `json:"id"`
}

type DynamoDBEvent struct {
    Records []DynamoDBEventRecord `json:"Records"`
}

type DynamoDBEventRecord struct {
    AWSRegion      string                       `json:"awsRegion"`
    Change         DynamoDBStreamRecord         `json:"dynamodb"`
    EventID        string                       `json:"eventID"`
    EventName      string                       `json:"eventName"`
    EventSource    string                       `json:"eventSource"`
    EventVersion   string                       `json:"eventVersion"`
    EventSourceArn string                       `json:"eventSourceARN"`
    UserIdentity   *events.DynamoDBUserIdentity `json:"userIdentity,omitempty"`
}

type DynamoDBStreamRecord struct {
    ApproximateCreationDateTime events.SecondsEpochTime             `json:"ApproximateCreationDateTime,omitempty"`
    // changed to map[string]*dynamodb.AttributeValue
    Keys                        map[string]*dynamodb.AttributeValue `json:"Keys,omitempty"`
    // changed to map[string]*dynamodb.AttributeValue
    NewImage                    map[string]*dynamodb.AttributeValue `json:"NewImage,omitempty"`
    // changed to map[string]*dynamodb.AttributeValue
    OldImage                    map[string]*dynamodb.AttributeValue `json:"OldImage,omitempty"`
    SequenceNumber              string                              `json:"SequenceNumber"`
    SizeBytes                   int64                               `json:"SizeBytes"`
    StreamViewType              string                              `json:"StreamViewType"`
}

以上是关于将map [string] DynamoDBAttributeValue解组成一个struct的主要内容,如果未能解决你的问题,请参考以下文章

将 Map<String,String> 转换为 Map<String,Object>

怎么将map转换成string

如何将mapobject转换成string

如何将map<string list<>>转换成城map<string,object>

Java 8:将 Map<String, List<String>> 转换为 Map<String, String[]> [重复]

将 List[String] 转换为 Map[String,String]