Количество атрибутов в ключевой схеме должно соответствовать количеству атрибутов, определенных в определениях атрибутов.

109

Я пытаюсь создать простую таблицу с помощью оболочки javascript DynamoDB и получаю следующее исключение:


    {   
    "message": "The number of attributes in key schema must match the number of attributes defined in attribute definitions.",
    "code": "ValidationException",
    "time": "2015-06-16T10:24:23.319Z",
    "statusCode": 400,
    "retryable": false 
    }

Ниже представлена ​​таблица, которую я пытаюсь создать:


    var params = {
        TableName: 'table_name',
        KeySchema: [ 
            { 
                AttributeName: 'hash_key_attribute_name',
                KeyType: 'HASH',
            },

        ],
        AttributeDefinitions: [ 
            {
                AttributeName: 'hash_key_attribute_name',
                AttributeType: 'S', 
            },
            {
                AttributeName: 'attribute_name_1',
                AttributeType: 'S', 
            }
        ],
        ProvisionedThroughput: { 
            ReadCapacityUnits: 1, 
            WriteCapacityUnits: 1, 
        },


    };
    dynamodb.createTable(params, function(err, data) {
        if (err) print(err); 
        else print(data); 
    });

Однако, если я добавлю второй атрибут в keySchema, он будет работать нормально. Ниже рабочий стол:


    var params = {
        TableName: 'table_name',
        KeySchema: [ 
            { 
                AttributeName: 'hash_key_attribute_name',
                KeyType: 'HASH',
            },
            { 
                AttributeName: 'attribute_name_1', 
                KeyType: 'RANGE', 
            }

        ],
        AttributeDefinitions: [ 
            {
                AttributeName: 'hash_key_attribute_name',
                AttributeType: 'S', 
            },
            {
                AttributeName: 'attribute_name_1',
                AttributeType: 'S', 
            }
        ],
        ProvisionedThroughput: { 
            ReadCapacityUnits: 1, 
            WriteCapacityUnits: 1, 
        },


    };
    dynamodb.createTable(params, function(err, data) {
        if (err) print(err); 
        else print(data); 
    });

Я не хочу добавлять диапазон в схему ключей. Есть идеи, как это исправить?

НАББА
источник
Это происходит только с DynamoDBLocal? Что происходит, когда вы пытаетесь сделать то же самое с реальной службой?
mkobit
У меня еще нет учетной записи AWS, поэтому я не мог проверить ее на реальном сервисе. Я использую последнюю версию DynamoDB local (Dynamodb_local_2015-04-27_1.0).
NAbbas
1
У меня такое же поведение с Dynamodb_local_2016-04-19
Крис
2
Неважно, TL; DR Mingliang говорит само за себя.
Крис

Ответы:

230

DynamoDB не имеет схемы (кроме ключевой схемы)

Иными словами, вам нужно указать схему ключей (имя и тип атрибута) при создании таблицы. Что ж, вам не нужно указывать какие-либо неключевые атрибуты. Позже вы можете поместить элемент с любым атрибутом (конечно, с ключами).

На странице документации , то AttributeDefinitionsопределяется как:

Массив атрибутов, описывающих схему ключей для таблицы и индексов.

Когда вы создаете таблицу, AttributeDefinitionsполе используется только для ключей хеша и / или диапазона. В вашем первом случае есть только хеш-ключ (номер 1), а вы предоставляете 2 AttributeDefinitions. Это основная причина исключения.

TL; DR Не включайте определения неключевых атрибутов в AttributeDefinitions.

Минглян Лю
источник
10
с одним исключением, я считаю, неключевой атрибут должен быть в, AttributeDefinitionsесли этот ключ будет использоваться в качестве ключа hashили rangeв индексе
SRLE
23

Когда вы используете неключевой атрибут в at "AttributeDefinitions", вы должны использовать его как индекс, иначе это противоречит принципу работы DynamoDB. См. Ссылку .

Поэтому не нужно добавлять неключевой атрибут, "AttributeDefinitions"если вы не собираетесь использовать его в качестве индекса или первичного ключа.

var params = {
        TableName: 'table_name',
        KeySchema: [ // The type of of schema.  Must start with a HASH type, with an optional second RANGE.
            { // Required HASH type attribute
                AttributeName: 'UserId',
                KeyType: 'HASH',
            },
            { // Optional RANGE key type for HASH + RANGE tables
                AttributeName: 'RemindTime', 
                KeyType: 'RANGE', 
            }
        ],
        AttributeDefinitions: [ // The names and types of all primary and index key attributes only
            {
                AttributeName: 'UserId',
                AttributeType: 'S', // (S | N | B) for string, number, binary
            },
            {
                AttributeName: 'RemindTime',
                AttributeType: 'S', // (S | N | B) for string, number, binary
            },
            {
                AttributeName: 'AlarmId',
                AttributeType: 'S', // (S | N | B) for string, number, binary
            },
            // ... more attributes ...
        ],
        ProvisionedThroughput: { // required provisioned throughput for the table
            ReadCapacityUnits: 1, 
            WriteCapacityUnits: 1, 
        },
        LocalSecondaryIndexes: [ // optional (list of LocalSecondaryIndex)
            { 
                IndexName: 'index_UserId_AlarmId',
                KeySchema: [ 
                    { // Required HASH type attribute - must match the table's HASH key attribute name
                        AttributeName: 'UserId',
                        KeyType: 'HASH',
                    },
                    { // alternate RANGE key attribute for the secondary index
                        AttributeName: 'AlarmId', 
                        KeyType: 'RANGE', 
                    }
                ],
                Projection: { // required
                    ProjectionType: 'ALL', // (ALL | KEYS_ONLY | INCLUDE)
                },
            },
            // ... more local secondary indexes ...
        ],
    };
    dynamodb.createTable(params, function(err, data) {
        if (err) ppJson(err); // an error occurred
        else ppJson(data); // successful response
    });
Габриэль Ву
источник
2

У меня тоже была эта проблема, и я опубликую здесь, что у меня пошло не так, если это поможет кому-то другому.

В моем у CreateTableRequestменя был пустой массив для GlobalSecondaryIndexes.

CreateTableRequest createTableRequest = new CreateTableRequest
{
  TableName = TableName,
  ProvisionedThroughput = new ProvisionedThroughput { ReadCapacityUnits = 2, WriteCapacityUnits = 2 },
  KeySchema = new List<KeySchemaElement>
  {
     new KeySchemaElement
     {
        AttributeName = "Field1",
        KeyType = KeyType.HASH
     },
     new KeySchemaElement
     {
        AttributeName = "Field2",
        KeyType = KeyType.RANGE
     }
  },
  AttributeDefinitions = new List<AttributeDefinition>()
  {
     new AttributeDefinition
     {
         AttributeName = "Field1", 
         AttributeType = ScalarAttributeType.S
     },
     new AttributeDefinition
     {
        AttributeName = "Field2",
        AttributeType = ScalarAttributeType.S
     }
  },
  //GlobalSecondaryIndexes = new List<GlobalSecondaryIndex>
  //{                            
  //}
};

Комментирование этих строк при создании таблицы решило мою проблему. Так что я полагаю, что список должен быть nullне пустым.

NickBeaugié
источник