DynamoDB para noobs como yo
data:image/s3,"s3://crabby-images/31819/31819295cefffdc74b1d368a3369b3d77a47435f" alt=""
10 segundos de fama
- Andrés Santos
- Co-Organizador
-
Senior Software Engineer
- Backend Node + AWS
@dresrok
@EpamAnywhere
@LibertyMutual
@IbaguéJS
Antes de empezar...
- Tipos de Bases de Datos
- Teorema CAP
Tipos de Bases de Datos
data:image/s3,"s3://crabby-images/67259/672596d7aee6331826263512654e30c33d98c1cc" alt=""
data:image/s3,"s3://crabby-images/67259/672596d7aee6331826263512654e30c33d98c1cc" alt=""
Fuente: Graph Databases in Action - Dave Bechberger & Josh Perryman
Teorema CAP
Los sistemas distribuidos no pueden garantizar a la vez que haya Consistencia, Disponibilidad y Tolerancia a Particiones.
data:image/s3,"s3://crabby-images/6cad2/6cad2decf12bf2d3c44d8a8d60f22f903e0e1161" alt=""
Consistency: Todos los nodos ven los mismos datos simultáneamente. En cada lectura se retorna la escritura más reciente.
data:image/s3,"s3://crabby-images/f894e/f894e75e8fdc3904c6e7bd80860088ab0158504d" alt=""
Teorema CAP
Availability: Cada cliente SIEMPRE puede leer y escribir. En el caso de las lecturas el valor puede que no sea el más reciente.
data:image/s3,"s3://crabby-images/6cad2/6cad2decf12bf2d3c44d8a8d60f22f903e0e1161" alt=""
data:image/s3,"s3://crabby-images/78a1c/78a1c77b8364f8cb18b04ad1e82c21457ecc1d36" alt=""
data:image/s3,"s3://crabby-images/78a1c/78a1c77b8364f8cb18b04ad1e82c21457ecc1d36" alt=""
Teorema CAP
Partition Tolerance: El sistema continúa operando a pesar de fallas de en red.
data:image/s3,"s3://crabby-images/6cad2/6cad2decf12bf2d3c44d8a8d60f22f903e0e1161" alt=""
data:image/s3,"s3://crabby-images/f0a17/f0a1760fffc36c660465bdbbee58c1e3a32a9a49" alt=""
data:image/s3,"s3://crabby-images/73700/737006ca59f537bc20fb37fc71310ac89b8cc887" alt=""
Teorema CAP
data:image/s3,"s3://crabby-images/258a1/258a1cb3c035602d488ba024f3020cff35f2eaf1" alt=""
Fuente: Bases de Datos para el Big Data - Marlon Cardenas
Danos el código
data:image/s3,"s3://crabby-images/6b26c/6b26ce2e571eefd143745d1acb73f42e014fef17" alt=""
data:image/s3,"s3://crabby-images/587ab/587abf591ffb481e2d2e86b1321a0d1e6750f18e" alt=""
Oblíguenme
Empecemos...
- ¿Qué es DynamoDB?
- ¿Cómo funciona?
- ¿Cuándo no debe usarse?
- Conceptos Generales
- DynamoDB API
¿Qué es DynamoDB?
- NoSQL rápida y flexible (key-value, schemaless)
- Operaciones de lectura y escritura en ms
- Escalamiento infinito sin degradación del rendimiento
- Totalmente gestionada
- Ofrece API, Autenticación e Infraestructura como código (IaC)
¿Cómo funciona?
data:image/s3,"s3://crabby-images/42d88/42d88dac1cfd912e611c427b531b0a5e935234fa" alt=""
data:image/s3,"s3://crabby-images/ef9d5/ef9d5928788f0e1e92d4e857dc8345676f3265c5" alt=""
data:image/s3,"s3://crabby-images/0c05e/0c05e51b669c7191da8d293ac66b4ae88e078cf8" alt=""
data:image/s3,"s3://crabby-images/544fe/544fe7cc27d4844c802112cef1137abf6c31dafb" alt=""
Primary Key = Partition Key
+ Sort Key
data:image/s3,"s3://crabby-images/bc503/bc50341b1278670d76484cdf6ed5dc461f90db33" alt=""
data:image/s3,"s3://crabby-images/37c8a/37c8a886d80179d8bfbe2f6254d3a8e0d5c466ea" alt=""
data:image/s3,"s3://crabby-images/d8346/d834654fb736fd90bf7f6fa6ee9a3e751c85e94e" alt=""
¿Cuándo no debe usarse?
- Cuando tu aplicación necesite queries avanzados, joins o agrupaciones.
- Para obtener rankings, donde se necesiten funciones de agregación y clasificación
- Cuando se requiere análisis en tiempo real de datos históricos
Conceptos Generales
-
Tablas
-
Atributos*
-
Items
-
Llaves Primarias*
-
Índices**
-
Streams**
-
Time-to-live (TTL)**
Conceptos Generales
data:image/s3,"s3://crabby-images/830bf/830bf92511ea692530fa8ee858130a6cea07aa15" alt=""
Fuente: The DynamoDB Book - Alex DeBrie
Conceptos Generales
Atributos*
-
Scalars: Representan exactamente un valor.
-
string, number, binary, boolean y null
-
-
Documents: Representan una estructura compleja con valores anidados similar a un JSON.
-
list y map
-
-
Sets: Representan grupos de valores escalares.
-
string, number y binary
-
Conceptos Generales
// Scalars
{
"Name": {
"S": "Andrés Santos"
},
"Picture": {
"B": "dGhpcyB0ZXh0IGlzIGJhc2U2NC1lbmNvZGVk"
},
"Age": {
"NULL": true
},
"IsOrganizer": {
"BOOL": true
},
"Attendees": {
"N": "100.0"
}
}
// Documents
{
"Event": {
"M": {
"Name": {
"S": "IbaguéJS - Meetup Octubre"
},
"Date": {
"S": "2022-10-22T10:00:00+0500"
},
"Topics": {
"L": [
{"S": "DynamoDB para noobs como yo"},
{"S": "Redux Toolkit"}
]
}
}
}
}
// Sets
{
"Organizers": {
"SS": ["Diana", "Yeison", "Juan JS", "Jomazao"]
},
"AvgAttendance": {
"NS": ["42.2", "19", "35", "32.14"]
},
"EventImages": {
"BS": ["U3Vubnk=", "UmFpbnk=", "U25vd3k="]
}
}
Conceptos Generales
Llaves Primarias*
-
Simples: Consta de un solo elemento llamado Partition Key (Hash) cuyo valor debe ser único.
-
Compuestas: Consta de dos elementos el Partition Key y el Sort Key (Range), el valor del PK puede repetirse pero la combinación de ambos genera un identificador único.
Conceptos Básicos
Llaves Primarias*
data:image/s3,"s3://crabby-images/2427a/2427aa689ba66ab95582a7893c50f181bc498234" alt=""
data:image/s3,"s3://crabby-images/2427a/2427aa689ba66ab95582a7893c50f181bc498234" alt=""
data:image/s3,"s3://crabby-images/2427a/2427aa689ba66ab95582a7893c50f181bc498234" alt=""
data:image/s3,"s3://crabby-images/2427a/2427aa689ba66ab95582a7893c50f181bc498234" alt=""
data:image/s3,"s3://crabby-images/d4506/d4506868a7ea517fcbd0a45962600a4e401f0141" alt=""
Conceptos Básicos
// Usando SDK
const AWS = require('aws-sdk');
AWS.config.update({region: 'REGION'});
const ddb = new AWS.DynamoDB({apiVersion: '2012-08-10'});
const params = {
AttributeDefinitions: [
{ AttributeName: 'Actor', AttributeType: 'S' },
{ AttributeName: 'Movie', AttributeType: 'S' }
],
KeySchema: [
{ AttributeName: 'Actor', KeyType: 'HASH' },
{ AttributeName: 'Movie', KeyType: 'RANGE' }
],
TableName: 'actors-dev',
};
ddb.createTable(params, function(err, data) {
if (err) {
console.log("Error", err);
} else {
console.log("Table Created", data);
}
});
Creando una tabla
// Usando CDK
import dynamodb = require('aws-cdk-lib/aws-dynamodb');
const dbTable = new dynamodb.Table(this, 'actors-dev', {
tableName: 'actors-dev',
partitionKey: { name: 'Actor', type: 'S' },
sortKey: { name: 'Movie', type: 'S' },
});
Conceptos Básicos
Índices**
-
Local Secondary Index (LSI): Tiene el mismo PK que la tabla base pero un SK diferente.
-
Global Secondary index (GSI): Tanto el PK como el SK son diferentes que la tabla base.
Conceptos Básicos
Streams**
data:image/s3,"s3://crabby-images/92722/92722118c9dd2e4d477d67bc1655971b69a2fdb3" alt=""
Fuente: The DynamoDB Book - Alex DeBrie
Conceptos Básicos
Time-to-live (TTL)**
-
Permite definir un tiempo de expiración para borrar un item de manera automática.
// Usando CDK
import dynamodb = require('aws-cdk-lib/aws-dynamodb');
const dbTable = new dynamodb.Table(this, 'actors-dev', {
tableName: 'actors-dev',
partitionKey: { name: 'Actor', type: 'S' },
sortKey: { name: 'Movie', type: 'S' },
timeToLiveAttribute: 'ExpiresAt', // Unix Timestamp
});
DynamoDB API
- PutItem
- GetItem
- UpdateItem
- DeleteItem
- Query
- Scan
PutItem
data:image/s3,"s3://crabby-images/d4506/d4506868a7ea517fcbd0a45962600a4e401f0141" alt=""
// PutItem
INSERT INTO "actors-dev" (Actor, Movie, Role, ...)
VALUES (Natalie Portman, Star Wars: Attack of the Clones, Padmé Amidala, ...);
import { DynamoDB } from 'aws-sdk';
const params = {
TableName: 'actors-dev',
Item: {
Actor: { S: 'Natalie Portman' },
Movie: { S: 'Star Wars: Attack of the Clones' },
Role: { S: 'Padmé Amidala' },
Year: { N: '2002' },
Genre: { SS: ['Epic', 'Space Opera', 'Fantasy'] },
CharacterAttributes: {
M: {
HairColor: { S: 'brown'},
SkinColor: { S: 'light'},
EyeColor: {S: 'brown'},
}
}
},
};
await new DynamoDB().putItem(params).promise();
data:image/s3,"s3://crabby-images/6d7db/6d7dbb4cb0e0fcc1f4247c6332e04a1d33ae43cb" alt=""
GetItem
data:image/s3,"s3://crabby-images/d4506/d4506868a7ea517fcbd0a45962600a4e401f0141" alt=""
// PK + SK
SELECT * FROM "actors-dev"
WHERE "Actor" = 'Tom Hanks' AND "Movie" = 'Toy Story'
import { DynamoDB } from 'aws-sdk';
const params = {
TableName: 'actors-dev',
Key: {
Actor: { S: 'Tom Hanks' },
Movie: { S: 'Toy Story' },
},
ConsistentRead: true,
};
const result = await new DynamoDB().getItem(params).promise();
result.Item.Role.S // Woody
// PK
SELECT * FROM "actors-dev"
WHERE "ActorID" = 'b261bcc9-3a3f-4ef2-b62f-85311e9f474e'
import { DynamoDB } from 'aws-sdk';
const params = {
TableName: 'actors-dev',
Key: {
ActorID: { S: 'b261bcc9-3a3f-4ef2-b62f-85311e9f474e' }
},
ConsistentRead: true,
};
const result = await new DynamoDB().getItem(params).promise();
result.Item.Role.S // Woody
UpdateItem
import { DynamoDB } from 'aws-sdk';
const params = {
TableName: 'actors-dev',
Key: {
Actor: { S: 'Natalie Portman' },
Movie: { S: 'Star Wars: Attack of the Clones' },
},
UpdateExpression:
'SET #year = :year, #characterAttributes.#eyeColor :eyeColor",
ADD #picture :picture"',
ExpressionAttributeNames: {
'#year': 'Year',
"#characterAttributes": "CharacterAttributes",
"#eyeColor": "EyeColor",
"#picture": "Picture"
},
ExpressionAttributeValues: {
':year': { N: '2002' },
':eyeColor': { S: 'green' },
':picture': { B: 'cGljdHJl=' }
}
}
await new DynamoDB().updateItem(params).promise()
data:image/s3,"s3://crabby-images/13d0e/13d0e58178a94fca40e69b05d504774e4ef73a0d" alt=""
data:image/s3,"s3://crabby-images/13d0e/13d0e58178a94fca40e69b05d504774e4ef73a0d" alt=""
import { DynamoDB } from 'aws-sdk';
const params = {
TableName: 'actors-dev',
Key: {
Actor: { S: 'Tom Hanks' },
Movie: { S: 'Cast Away' },
},
UpdateExpression: 'ADD #awards :awards"',
ExpressionAttributeNames: {
'#awards': 'Awards',
},
ExpressionAttributeValues: {
':awards': {
SS: [
'Golden Globe Award for Best Actor',
'People\'s Choice Award for Favorite Movie Actor'
]
}
}
}
await new DynamoDB().updateItem(params).promise()
import { DynamoDB } from 'aws-sdk';
const params = {
TableName: 'actors-dev',
Key: {
Actor: { S: 'Natalie Portman' },
Movie: { S: 'Star Wars: Attack of the Clones' },
},
UpdateExpression: 'REMOVE #picture :picture"',
ExpressionAttributeNames: {
'#picture': 'Picture',
},
}
await new DynamoDB().updateItem(params).promise()
DeleteItem
data:image/s3,"s3://crabby-images/d4506/d4506868a7ea517fcbd0a45962600a4e401f0141" alt=""
// PK + SK
DELETE FROM "actors-dev"
WHERE "Actor" = 'Tim Allen' AND "Movie" = 'Toy Story'
import { DynamoDB } from 'aws-sdk';
const params = {
TableName: 'actors-dev',
Key: {
Actor: { S: 'Tim Allen' },
Movie: { S: 'Toy Story' },
},
};
await new DynamoDB().deleteItem(params).promise();
// PK
DELETE FROM "actors-dev"
WHERE "ActorID" = '1a0847bc-ae37-4977-ba3d-d400b4109926'
import { DynamoDB } from 'aws-sdk';
const params = {
TableName: 'actors-dev',
Key: {
ActorID: { S: '1a0847bc-ae37-4977-ba3d-d400b4109926' },
};
await new DynamoDB().deleteItem(params).promise();
Query
data:image/s3,"s3://crabby-images/952f8/952f8c45f39646ff6e9d416da000c0a59bef2ef0" alt=""
data:image/s3,"s3://crabby-images/5cbd7/5cbd7cd6cdc6b95dc234c140a7338a7eb252c1ed" alt=""
SELECT * FROM "actors-dev"
WHERE "Actor" LIKE 'Tom Hanks'
LIMIT 2
import { DynamoDB } from 'aws-sdk';
const params = {
TableName: 'actors-dev',
KeyConditionExpression: '#actor = :actor',
ExpressionAttributeNames: {
'#actor': 'Actor'
},
ExpressionAttributeValues: {
':actor': { 'S': 'Tom Hanks' }
},
Limit: 2
}
const result = await new DynamoDB().query(params).promise();
result.Items.forEach((item) => item.Role.S)
data:image/s3,"s3://crabby-images/08b1a/08b1a79b5373be61e5a1cfc6adb26aa21e7b8fb1" alt=""
import { DynamoDB } from 'aws-sdk';
const params = {
TableName: 'actors-dev',
KeyConditionExpression: '#actor = :actor AND #movie BETWEEN :a AND :m',
ExpressionAttributeNames: {
'#actor': 'Actor',
'#movie': 'Movie'
},
ExpressionAttributeValues: {
':actor': { 'S': 'Tom Hanks' },
':a': { 'S': 'A' },
':m': { 'S': 'M' }
},
}
await new DynamoDB().query(params).promise();
result.Items.forEach((item) => item.Role.S)
import { DynamoDB } from 'aws-sdk';
const params = {
TableName: 'actors-dev',
KeyConditionExpression: '#actor = :actor AND begins_with(#movie, :movie)',
ExpressionAttributeNames: {
'#actor': 'Actor',
'#movie': 'Movie'
},
ExpressionAttributeValues: {
':actor': { 'S': 'Tom Hanks' },
':movie': { 'S': 'Cast' }
},
}
await new DynamoDB().query(params).promise();
result.Items.forEach((item) => item.Role.S)
// Key condition expressions for query
a = b
a < b
a <= b
a > b
a >= b
a BETWEEN b AND c
begins_with (a, substr)
Query + GSI
data:image/s3,"s3://crabby-images/b13d2/b13d210ed7d250dcff5bb8fedc9ac1bbc78fb6b9" alt=""
data:image/s3,"s3://crabby-images/14ccd/14ccd10fd3cd0d51c774d872e872581d10460cdb" alt=""
import { DynamoDB } from 'aws-sdk';
const params = {
TableName: 'actors-dev',
IndexName: 'MoviesIndex',
KeyConditionExpression: '#movie = :movie',
ExpressionAttributeNames: {
'#movie': 'Movie'
},
ExpressionAttributeValues: {
':movie': { 'S': 'Toy Story' }
},
}
await new DynamoDB().query(params).promise();
result.Items.forEach((item) => item.Actor.S)
// Usando CDK
import dynamodb = require('aws-cdk-lib/aws-dynamodb');
const dbTable = new dynamodb.Table(this, 'actors-dev', {
tableName: 'actors-dev',
partitionKey: { name: 'Actor', type: 'S' },
sortKey: { name: 'Movie', type: 'S' },
});
dbTable.addGlobalSecondaryIndex({
indexName: 'MoviesIndex',
partitionKey: { name: 'Movie', type: 'S'},
sortKey: { name: 'Actor', type: 'S'},
projectionType: 'ALL',
});
data:image/s3,"s3://crabby-images/1fb4e/1fb4e5b3c846e5b117b62df2c495ddeaea527f57" alt=""
Query + Filter
data:image/s3,"s3://crabby-images/4de0b/4de0b081ab6b6cf9232c98b0b5335334a9e7e9a0" alt=""
import { DynamoDB } from 'aws-sdk';
const params = {
TableName: 'customer-orders-dev',
KeyConditionExpression: '#c = :c AND #ot BETWEEN :start and :end"',
FilterExpression: '#amount > :amount',
ExpressionAttributeNames: {
'#c': 'CustomerId',
'#ot': 'OrderTime',
"#amount": "Amount"
},
ExpressionAttributeValues: {
':c': { 'S': '36ab55a589e4' },
':start': { 'S': '2020-01-10T00:00:00.000000' },
':end': { 'S': '2020-01-20T00:00:00.000000' },
":amount": { "N": "80" }
},
}
await new DynamoDB().query(params).promise();
data:image/s3,"s3://crabby-images/4de0b/4de0b081ab6b6cf9232c98b0b5335334a9e7e9a0" alt=""
data:image/s3,"s3://crabby-images/4de0b/4de0b081ab6b6cf9232c98b0b5335334a9e7e9a0" alt=""
Scan
data:image/s3,"s3://crabby-images/26731/267318b6e54dd1e335eb73f769c61b3fcd882ffc" alt=""
import { DynamoDB } from 'aws-sdk';
const params = {
TableName: 'actors-dev',
FilterExpression: "#genre = :genre",
ExpressionAttributeNames: {
'#genre': 'Genre',
},
ExpressionAttributeValues: {
":genre": { "S": "Drama" }
},
}
await new DynamoDB().scan(params).promise();
data:image/s3,"s3://crabby-images/26731/267318b6e54dd1e335eb73f769c61b3fcd882ffc" alt=""
data:image/s3,"s3://crabby-images/dc92a/dc92a002b75bf7330a1989745b3fbd6d3c62e0d2" alt=""
DynamoDB
By Andrés Santos
DynamoDB
- 167