Microsoft Azure SDK for Rust

legal

Build Status Coverage Status alpha

Crate cratedown cratelastdown

tag release commitssince

GitHub contributors

Introduction

Microsoft Azure expose its technologies via REST API. These APIs are easily consumable from any language (good) but are weakly typed. With this library and its related crate you can exploit the power of Microsoft Azure from Rust in a idiomatic way.

This crate relies heavily on the excellent crate called Hyper. As of this library version 0.4.0 all the methods are future-aware.

NOTE: This repository is under heavy development and is likely to break over time. The current releases will probabily contain bugs. As usual open issues if you find any.

Disclaimer

Although I am a Microsoft employee, this is not a Microsoft endorsed project. It's simply a pet project of mine: I love Rust (who doesn't? :smirk:) and Microsoft Azure technologies so I thought to close the gap between them. It's also a good project for learning Rust. This library relies heavily on Hyper. We use the latest Hyper code so this library is fully async with Futures and Tokio.

Example

You can find examples in the examples folder. Here is a glimpse:

main.rs

```rust extern crate azuresdkfor_rust;

extern crate chrono; extern crate futures; extern crate hyper; extern crate hypertls; extern crate tokio; extern crate tokiocore;

use std::error::Error;

use futures::future::*; use tokio_core::reactor::Core;

use azuresdkfor_rust::cosmos::{AuthorizationToken, TokenType, Client, PartitionKey};

[macro_use]

extern crate serdederive; use azuresdkforrust::cosmos;

[derive(Serialize, Deserialize, Debug)]

struct MySampleStruct<'a> { id: &'a str, astring: &'a str, anumber: u64, a_timestamp: i64, }

const DATABASE: &'static str = "azuresdktestdb"; const COLLECTION: &'static str = "azuresdktc";

fn main() { code().unwrap(); }

// This code will perform these tasks: // 1. Find an Azure Cosmos DB called DATABASE. If it does not exist, create it. // 2. Find an Azure Cosmos collection called COLLECTION in DATABASE. // If it does not exist, create it. // 3. Store an entry in collection COLLECTION of database DATABASE. // 4. Delete everything. // // We will use multiple futures for this hoping to make the code clearer. // There is no need to proceed this way in your code. // You can go crazy with future combinators if you want to :) fn code() -> Result<(), Box> { // Let's get Cosmos account and master key from env variables. // This helps automated testing. let masterkey = std::env::var("COSMOSMASTERKEY").expect("Set env variable COSMOSMASTERKEY first!"); let account = std::env::var("COSMOSACCOUNT").expect("Set env variable COSMOS_ACCOUNT first!");

// First, we create an authorization token. There are two types of tokens, master and resource
// constrained. Please check the Azure documentation for details. You can change tokens
// at will and it's a good practice to raise your privileges only when needed.
let authorization_token = AuthorizationToken::new(account, TokenType::Master, master_key)?;

// We will create a tokio-core reactor which will drive our futures.
let mut core = Core::new()?;

// Next we will create a Cosmos client. You need an authorization_token but you can later
// change it if needed. Notice the client will be tied to your reactor.
let client = Client::new(&core.handle(), authorization_token)?;

// list_databases will give us the databases available in our account. If there is
// an error (for example, the given key is not valid) you will receive a
// specific AzureError. In this example we will look for a specific database
// so we chain a filter operation.
let future = client.list_databases().and_then(|databases| {
    ok(databases.into_iter().find(|db| db.id == DATABASE))
});

// Now we run the future and check the answer. If the requested database
// is not found we create it.
let database = match core.run(future)? {
    Some(db) => db,
    None => core.run(client.create_database(DATABASE))?,
};
println!("database == {:?}", database);

// Now we look for a specific collection. If is not already present
// we will create it. The collection creation is more complex and
// has many options (such as indexing and so on).
let collection = {
    let collections = core.run(client.list_collections(&database))?;

    if let Some(collection) = collections.into_iter().find(|coll| coll.id == COLLECTION) {
        collection
    } else {
        let indexes = cosmos::collection::IncludedPathIndex {
            kind: cosmos::collection::KeyKind::Hash,
            data_type: cosmos::collection::DataType::String,
            precision: Some(3),
        };

        let ip = cosmos::collection::IncludedPath {
            path: "/*".to_owned(),
            indexes: vec![indexes],
        };


        let ip = cosmos::collection::IndexingPolicy {
            automatic: true,
            indexing_mode: cosmos::collection::IndexingMode::Consistent,
            included_paths: vec![ip],
            excluded_paths: vec![],
        };

        let coll = cosmos::collection::Collection::new(COLLECTION, ip);
        // Notice here we specify the expected performance level.
        // Performance levels have price impact. Also, higher
        // performance levels force you to specify an indexing
        // strategy. Consult the documentation for more details.
        core.run(client.create_collection(&database, 400, &coll))?
    }
};

println!("collection = {:?}", collection);

// Now that we have a database and a collection we can insert
// data in them. Let's create a struct. The only constraint
// is that the struct should be Serializable.
let doc = MySampleStruct {
    id: "unique_id1",
    a_string: "Something here",
    a_number: 100,
    a_timestamp: chrono::Utc::now().timestamp(),
};

// Now we store the struct in Azure Cosmos DB.
// Notice how easy it is! :)
// The method create_document will return, upon success,
// the document attributes.
let document_attributes = core.run(client.create_document(
    &database,
    &collection,
    &CreateDocumentOptions::default(),
    &doc,
))?;
println!("document_attributes == {:?}", document_attributes);

// We will perform some cleanup. First we delete the collection...
core.run(client.delete_collection(DATABASE, COLLECTION))?;
println!("collection deleted");

// And then we delete the database.
core.run(client.delete_database(DATABASE))?;
println!("database deleted");

Ok(())

} ```

State of the art

Right now the key framework is in place (authentication, enumerations, parsing and so on). If you want to contribute please do! Methods are added daily so please check the CHANGELOG.md for updates on the progress. Also note that the project is in early stages so the APIs are bound to change at any moment. I will strive to keep things steady but since I'm new to Rust I'm sure I'll have to correct some serious mistake before too long :smile:. I generally build for the latest nightly and leave to Travis to check the retrocompatibility.

Contributing

If you want to contribute please do! No formality required! :wink:. Please note that asking for a pull request you accept to yield your code as per Apache license, version 2.0.

Implemented methods

Storage Container

| Method | URL | | ---- | --- | | Create container | https://msdn.microsoft.com/en-us/library/azure/dd179468.aspx | | List containers | https://msdn.microsoft.com/en-us/library/azure/dd179352.aspx | | Delete container | https://msdn.microsoft.com/en-us/library/azure/dd179408.aspx |

Storage blobs

| Method | URL | | ---- | --- | | List blobs | https://msdn.microsoft.com/en-us/library/azure/dd135734.aspx | | Get blob | https://msdn.microsoft.com/en-us/library/azure/dd179440.aspx | | Put blob | https://msdn.microsoft.com/en-us/library/azure/dd179451.aspx | | Put blob page | https://msdn.microsoft.com/en-us/library/azure/dd179451.aspx | | Clear blob page | https://msdn.microsoft.com/en-us/library/azure/dd179451.aspx | | Put block | https://msdn.microsoft.com/en-us/library/azure/dd135726.aspx | | Lease blob | https://msdn.microsoft.com/library/azure/ee691972.aspx |

Event Hubs

| Method | URL | | ---- | --- | | Send Event | https://msdn.microsoft.com/en-us/library/azure/dn790664.aspx |

Cosmos database

Database

| Method | URL | | ---- | --- | | Create database | https://docs.microsoft.com/en-us/rest/api/documentdb/create-a-database | | List database | https://docs.microsoft.com/en-us/rest/api/documentdb/list-databases | | Get database | https://docs.microsoft.com/en-us/rest/api/documentdb/get-a-database | | Delete database | https://docs.microsoft.com/en-us/rest/api/documentdb/delete-a-database1 |

Collection

| Method | URL | | ---- | --- | | Create collection | https://docs.microsoft.com/en-us/rest/api/documentdb/create-a-collection | | List collections | https://docs.microsoft.com/en-us/rest/api/documentdb/list-collections | | Get collection | https://docs.microsoft.com/en-us/rest/api/documentdb/get-a-collection | | Delete collection | https://docs.microsoft.com/en-us/rest/api/documentdb/delete-a-collection | | Replace collection | https://docs.microsoft.com/en-us/rest/api/documentdb/replace-a-collection |

Document

| Method | URL | | ---- | --- | | Create document | https://docs.microsoft.com/en-us/rest/api/documentdb/create-a-document | | List documents | https://docs.microsoft.com/en-us/rest/api/documentdb/list-documents | | Get document | https://docs.microsoft.com/en-us/rest/api/documentdb/get-a-document | | Query documents | https://docs.microsoft.com/en-us/rest/api/documentdb/query-documents |

Azure tables

| Method | URL | | ---- | --- | | Create table | https://docs.microsoft.com/en-us/rest/api/storageservices/create-table | | Query tables | https://docs.microsoft.com/en-us/rest/api/storageservices/query-tables | | Query entities | https://docs.microsoft.com/en-us/rest/api/storageservices/query-entities | | Insert entity | https://docs.microsoft.com/en-us/rest/api/storageservices/insert-entity | | Update entity | https://docs.microsoft.com/en-us/rest/api/storageservices/update-entity2 | | Delete entity | https://docs.microsoft.com/en-us/rest/api/storageservices/delete-entity1 |

Azure tables entities can be manipulated in batches. The entities are serialized in JSON.

Run E2E test

Linux

```bash export STORAGEACCOUNT= export STORAGEMASTER_KEY=

export AZURESERVICEBUSNAMESPACE= export AZUREEVENTHUBNAME= export AZUREPOLICYNAME= export AZUREPOLICYKEY=

cargo test --features=test_e2e ```

Windows

```bat set STORAGEACCOUNT= set STORAGEMASTER_KEY=

set AZURESERVICEBUSNAMESPACE= set AZUREEVENTHUBNAME= set AZUREPOLICYNAME= set AZUREPOLICYKEY=

cargo test --features=test_e2e ```

License

This project is published under Apache license, version 2.0.