Compartir a través de


Migre su aplicación de Amazon DynamoDB a Azure Cosmos DB

SE APLICA A: NoSQL

Azure Cosmos DB es una base de datos totalmente administrada, escalable y distribuida globalmente. Proporciona acceso de baja latencia garantizado a los datos. Para obtener más información sobre Azure Cosmos DB, vea el artículo de información general. En este artículo se describe cómo migrar la aplicación .NET de DynamoDB a Azure Cosmos DB con unos cambios mínimos en el código.

Diferencias conceptuales

A continuación se muestran las diferencias conceptuales clave entre Azure Cosmos DB y DynamoDB:

DynamoDB Azure Cosmos DB
No aplicable Base de datos
Tabla Colección
Elemento Documento
Atributo Campo
Índice secundario Índice secundario
Clave principal: clave de partición Partition Key
Clave principal: criterio de ordenación No es obligatorio
STREAM ChangeFeed
Unidad de proceso de escritura Unidad de solicitud (flexible, se puede usar para lecturas o escrituras)
Unidad de proceso de lectura Unidad de solicitud (flexible, se puede usar para lecturas o escrituras)
Tablas globales No se requiere. Puede seleccionar la región directamente durante el aprovisionamiento de la cuenta de Azure Cosmos DB (puede cambiar la región más adelante)

Diferencias estructurales

Azure Cosmos DB tiene una estructura JSON más sencilla cuando se compara con la de DynamoDB. En el ejemplo siguiente se muestran las diferencias:

DynamoDB:

El siguiente objeto JSON representa el formato de datos de DynamoDB.

{
TableName: "Music",
KeySchema: [
{ 
  AttributeName: "Artist",
  KeyType: "HASH", //Partition key
},
{ 
  AttributeName: "SongTitle",
  KeyType: "RANGE" //Sort key
}
],
AttributeDefinitions: [
{ 
  AttributeName: "Artist",
  AttributeType: "S"
},
{ 
  AttributeName: "SongTitle",
  AttributeType: "S"
}
],
ProvisionedThroughput: {
  ReadCapacityUnits: 1,
  WriteCapacityUnits: 1
 }
}

Azure Cosmos DB:

El siguiente objeto JSON representa el formato de datos de Azure Cosmos DB.

{
"Artist": "",
"SongTitle": "",
"AlbumTitle": "",
"Year": 9999,
"Price": 0.0,
"Genre": "",
"Tags": ""
}

Migración del código

El ámbito de este artículo es migrar el código de una aplicación a Azure Cosmos DB, que es el aspecto fundamental de la migración de la base de datos. Para ayudarle a reducir la curva de aprendizaje, en las secciones siguientes se incluye una comparación de código en paralelo entre Amazon DynamoDB y el fragmento de código equivalente de Azure Cosmos DB.

Para descargar el código fuente, clone el siguiente repositorio:

git clone https://github.com/Azure-Samples/DynamoDB-to-CosmosDB

Requisitos previos

  • .NET Framework 4.7.2
  • La versión más reciente de Visual Studio con la carga de trabajo de desarrollo de Azure. Puede empezar con el IDE gratuito de Visual Studio Community. Habilite la carga de trabajo de desarrollo de Azure durante la configuración de Visual Studio.
  • Acceso a la cuenta de Azure Cosmos DB for NoSQL
  • Instalación local de Amazon DynamoDB
  • Java 8
  • Ejecución de la versión descargable de Amazon DynamoDB en el puerto 8000 (puede cambiar y configurar el código)

Configuración del código

Instalación del siguiente "paquete NuGet" en el proyecto:

Install-Package Microsoft.Azure.Cosmos

Establecimiento de la conexión

DynamoDB:

En Amazon DynamoDB, se usa el código siguiente para conectarse:

    AmazonDynamoDBConfig addbConfig = new AmazonDynamoDBConfig() ;
        addbConfig.ServiceURL = "endpoint";
        try { aws_dynamodbclient = new AmazonDynamoDBClient( addbConfig ); }

Azure Cosmos DB:

Para conectarse a Azure Cosmos DB, actualice el código para:

client_documentDB = new CosmosClient(
    "<nosql-account-endpoint>",
    tokenCredential
);

Optimización de la conexión en Azure Cosmos DB

Con Azure Cosmos DB, puede usar las siguientes opciones para optimizar la conexión:

  • ConnectionMode: use el modo de conexión directa para conectarse a los nodos de datos del servicio Azure Cosmos DB. Use el modo de puerta de enlace solo para inicializar y almacenar en caché las direcciones lógicas y actualizar en las actualizaciones. Para más información, consulte los modos de conectividad.

  • ApplicationRegion: esta opción se usa para establecer la región de replicación geográfica preferida que se usa para interactuar con Azure Cosmos DB. Para más información, consulte el artículo acerca de la distribución global.

  • ConsistencyLevel: esta opción se usa para invalidar el nivel de coherencia predeterminado. Para más información, consulte el artículo sobre los niveles de coherencia.

  • BulkExecutionMode: esta opción se usa para ejecutar operaciones masivas estableciendo la propiedad AllowBulkExecution en true. Para obtener más información, consulte el artículo sobre la importación en bloque.

    client_cosmosDB = new CosmosClient(" Your connection string ",new CosmosClientOptions()
    { 
      ConnectionMode=ConnectionMode.Direct,
      ApplicationRegion=Regions.EastUS2,
      ConsistencyLevel=ConsistencyLevel.Session,
      AllowBulkExecution=true  
    });
    

Cree el contenedor.

DynamoDB:

Para almacenar los datos en Amazon DynamoDB, primero es preciso crear la tabla. En el proceso de creación de la tabla se define el esquema, el tipo de clave y los atributos, tal como se muestra en el código siguiente:

// movies_key_schema
public static List<KeySchemaElement> movies_key_schema
  = new List<KeySchemaElement>
{
  new KeySchemaElement
  {
    AttributeName = partition_key_name,
    KeyType = "HASH"
  },
  new KeySchemaElement
  {
    AttributeName = sort_key_name,
    KeyType = "RANGE"
  }
};

// key names for the Movies table
public const string partition_key_name = "year";
public const string sort_key_name      = "title";
  public const int readUnits=1, writeUnits=1; 

    // movie_items_attributes
    public static List<AttributeDefinition> movie_items_attributes
  = new List<AttributeDefinition>
{
  new AttributeDefinition
  {
    AttributeName = partition_key_name,
    AttributeType = "N"
  },
  new AttributeDefinition
  {
    AttributeName = sort_key_name,
    AttributeType = "S"
  }

CreateTableRequest  request;
CreateTableResponse response;

// Build the 'CreateTableRequest' structure for the new table
request = new CreateTableRequest
{
  TableName             = table_name,
  AttributeDefinitions  = table_attributes,
  KeySchema             = table_key_schema,
  // Provisioned-throughput settings are always required,
  // although the local test version of DynamoDB ignores them.
  ProvisionedThroughput = new ProvisionedThroughput( readUnits, writeUnits );
};

Azure Cosmos DB:

En Amazon DynamoDB, debe aprovisionar las unidades de proceso de lectura y de escritura. Mientras que en Azure Cosmos DB se especifica el rendimiento como unidades de solicitud (RU/s), que se puede usar para cualquier operación de manera dinámica. Los datos se organizan como base de datos --> contenedor --> elemento. Puede especificar el rendimiento en el nivel de base de datos o en el nivel de colección, o en ambos.

Para crear una base de datos:

await client_cosmosDB.CreateDatabaseIfNotExistsAsync(movies_table_name);

Para crear el contenedor:

await cosmosDatabase.CreateContainerIfNotExistsAsync(new ContainerProperties() { PartitionKeyPath = "/" + partitionKey, Id = new_collection_name }, provisionedThroughput);

Carga de los datos

DynamoDB:

En el código siguiente se muestra cómo cargar los datos en Amazon DynamoDB. MoviesArray consta de una lista de documentos JSON que debe recorrer en iteración y cargar el documento JSON en Amazon DynamoDB:

int n = moviesArray.Count;
for( int i = 0, j = 99; i < n; i++ )
    {
  try
  {
    string itemJson = moviesArray[i].ToString();
    Document doc = Document.FromJson(itemJson);
    Task putItem = moviesTable.PutItemAsync(doc);
    if( i >= j )
    {
      j++;
      Console.Write( "{0,5:#,##0}, ", j );
      if( j % 1000 == 0 )
        Console.Write( "\n " );
      j += 99;
    }
    await putItem;

Azure Cosmos DB:

En Azure Cosmos DB, puede optar por transmitir y escribir con moviesContainer.CreateItemStreamAsync(). Sin embargo, en este ejemplo, el archivo JSON se deserializará en el tipo MovieModel para mostrar la característica de conversión de tipos. El código es multiproceso y usará la arquitectura distribuida de Azure Cosmos DB y acelerará la carga:

List<Task> concurrentTasks = new List<Task>();
for (int i = 0, j = 99; i < n; i++)
{
  try
  {
      MovieModel doc= JsonConvert.DeserializeObject<MovieModel>(moviesArray[i].ToString());
      doc.Id = Guid.NewGuid().ToString();
      concurrentTasks.Add(moviesContainer.CreateItemAsync(doc,new PartitionKey(doc.Year)));
      {
          j++;
          Console.Write("{0,5:#,##0}, ", j);
          if (j % 1000 == 0)
              Console.Write("\n               ");
          j += 99;
      }
      
  }
  catch (Exception ex)
  {
      Console.WriteLine("\n     ERROR: Could not write the movie record #{0:#,##0}, because:\n       {1}",
                          i, ex.Message);
      operationFailed = true;
      break;
  }
}
await Task.WhenAll(concurrentTasks);

Creación de un documento

DynamoDB:

Escribir un nuevo documento en Amazon DynamoDB no tiene seguridad de tipos, en el ejemplo siguiente se usa newItem como tipo de documento:

Task<Document> writeNew = moviesTable.PutItemAsync(newItem, token);
await writeNew;

Azure Cosmos DB:

Azure Cosmos DB proporciona seguridad de tipos a través del modelo de datos. Usamos el modelo de datos denominado "MovieModel":

public class MovieModel
{
    [JsonProperty("id")]
    public string Id { get; set; }
    [JsonProperty("title")]
    public string Title{ get; set; }
    [JsonProperty("year")]
    public int Year { get; set; }
    public MovieModel(string title, int year)
    {
        this.Title = title;
        this.Year = year;
    }
    public MovieModel()
    {

    }
    [JsonProperty("info")]
    public   MovieInfo MovieInfo { get; set; }

    internal string PrintInfo()
    {
        if(this.MovieInfo!=null)
        return            string.Format("\nMovie with title:{1}\n Year: {2}, Actors: {3}\n Directors:{4}\n Rating:{5}\n", this.Id, this.Title, this.Year, String.Join(",",this.MovieInfo.Actors), this.MovieInfo, this.MovieInfo.Rating);
        else
            return string.Format("\nMovie with  title:{0}\n Year: {1}\n",  this.Title, this.Year);
    }
}

En Azure Cosmos DB, newItem será MovieModel:

 MovieModel movieModel = new MovieModel()
            {
                Id = Guid.NewGuid().ToString(),
                Title = "The Big New Movie",
                Year = 2018,
                MovieInfo = new MovieInfo() { Plot = "Nothing happens at all.", Rating = 0 }
            };
    var writeNew= moviesContainer.CreateItemAsync(movieModel, new Microsoft.Azure.Cosmos.PartitionKey(movieModel.Year));
    await writeNew;

Lectura de un documento

DynamoDB:

Para leer en Amazon DynamoDB, debe definir primitivas:

// Create Primitives for the HASH and RANGE portions of the primary key
Primitive hash = new Primitive(year.ToString(), true);
Primitive range = new Primitive(title, false);

  Task<Document> readMovie = moviesTable.GetItemAsync(hash, range, token);
  movie_record = await readMovie;

Azure Cosmos DB:

Sin embargo, con Azure Cosmos DB la consulta es natural (LINQ):

IQueryable<MovieModel> movieQuery = moviesContainer.GetItemLinqQueryable<MovieModel>(true)
                        .Where(f => f.Year == year && f.Title == title);
// The query is executed synchronously here, but can also be executed asynchronously via the IDocumentQuery<T> interface
    foreach (MovieModel movie in movieQuery)
    {
      movie_record_cosmosdb = movie;
    }

La colección de documentos del ejemplo anterior:

  • tendrá seguridad de tipos
  • proporcionará una opción de consulta natural

Actualización de un elemento

DynamoDB: Para actualizar el elemento en Amazon DynamoDB:

updateResponse = await client.UpdateItemAsync( updateRequest );

Azure Cosmos DB:

En Azure Cosmos DB, la actualización se tratará como la operación Upsert, lo que significa insertar el documento si no existe:

await moviesContainer.UpsertItemAsync<MovieModel>(updatedMovieModel);

Eliminar un documento

DynamoDB:

Para eliminar un elemento en Amazon DynamoDB, debe volver a incluir primitivos:

Primitive hash = new Primitive(year.ToString(), true);
      Primitive range = new Primitive(title, false);
      DeleteItemOperationConfig deleteConfig = new DeleteItemOperationConfig( );
      deleteConfig.ConditionalExpression = condition;
      deleteConfig.ReturnValues = ReturnValues.AllOldAttributes;
      
  Task<Document> delItem = table.DeleteItemAsync( hash, range, deleteConfig );
        deletedItem = await delItem;

Azure Cosmos DB:

En Azure Cosmos DB, podemos obtener el documento y eliminarlo de forma asincrónica:

var result= ReadingMovieItem_async_List_CosmosDB("select * from c where c.info.rating>7 AND c.year=2018 AND c.title='The Big New Movie'");
while (result.HasMoreResults)
{
  var resultModel = await result.ReadNextAsync();
  foreach (var movie in resultModel.ToList<MovieModel>())
  {
    await moviesContainer.DeleteItemAsync<MovieModel>(movie.Id, new PartitionKey(movie.Year));
  }
  }

Consulta de documentos

DynamoDB:

En Amazon DynamoDB, se necesitan funciones de API para consultar los datos:

QueryOperationConfig config = new QueryOperationConfig( );
  config.Filter = new QueryFilter( );
  config.Filter.AddCondition( "year", QueryOperator.Equal, new DynamoDBEntry[ ] { 1992 } );
  config.Filter.AddCondition( "title", QueryOperator.Between, new DynamoDBEntry[ ] { "B", "Hzz" } );
  config.AttributesToGet = new List<string> { "year", "title", "info" };
  config.Select = SelectValues.SpecificAttributes;
  search = moviesTable.Query( config ); 

Azure Cosmos DB:

En Azure Cosmos DB, puede realizar proyecciones y filtrar en una consulta SQL simple:

var result = moviesContainer.GetItemQueryIterator<MovieModel>( 
  "select c.Year, c.Title, c.info from c where Year=1998 AND (CONTAINS(Title,'B') OR CONTAINS(Title,'Hzz'))");

En el caso de las operaciones de intervalo, por ejemplo, "between", debe realizar un examen en Amazon DynamoDB:

ScanRequest sRequest = new ScanRequest
{
  TableName = "Movies",
  ExpressionAttributeNames = new Dictionary<string, string>
  {
    { "#yr", "year" }
  },
  ExpressionAttributeValues = new Dictionary<string, AttributeValue>
  {
      { ":y_a", new AttributeValue { N = "1960" } },
      { ":y_z", new AttributeValue { N = "1969" } },
  },
  FilterExpression = "#yr between :y_a and :y_z",
  ProjectionExpression = "#yr, title, info.actors[0], info.directors, info.running_time_secs"
};

ClientScanning_async( sRequest ).Wait( );

En Azure Cosmos DB, puede usar la consulta SQL y una instrucción de una sola línea:

var result = moviesContainer.GetItemQueryIterator<MovieModel>( 
  "select c.title, c.info.actors[0], c.info.directors,c.info.running_time_secs from c where BETWEEN year 1960 AND 1969");

Eliminación de un contenedor

DynamoDB:

Para eliminar la tabla en Amazon DynamoDB, puede especificar:

client.DeleteTableAsync( tableName );

Azure Cosmos DB:

Para eliminar la colección en Azure Cosmos DB, puede especificar:

await moviesContainer.DeleteContainerAsync();

Después, elimine la base de datos si necesita:

await cosmosDatabase.DeleteAsync();

Como puede ver, Azure Cosmos DB admite las consultas naturales (SQL), las operaciones son asincrónicas y mucho más fáciles. Puede migrar fácilmente el código complejo a Azure Cosmos DB, lo que resulta más sencillo después de la migración.

Pasos siguientes