Exercise - Blob uploads and downloads

Completed

To interact with individual blobs in Blob Storage, use a BlobClient object. You can get a BlobClient by requesting it with the blob's name from the BlobContainerClient in which the blob is located. BlobClient has methods to upload, download, and manage individual blobs in Blob Storage.

Getting a BlobClient object

To get a BlobClient by name, call the GetBlobClient methods on the BlobContainerClient that contains the blob, by using the name of the blob. A BlobClient object allows you to interact with the blob by uploading, downloading, or managing the blob in Blob Storage.

Moving data to and from a blob is a network operation that takes time. The Azure Storage SDK for .NET provides asynchronous implementation of all methods that require network activity. We recommend using these async implementations whenever possible in your application.

We recommend using streams instead of in-memory structures like byte arrays or strings when you're working with large data objects. This approach avoids buffering the full content in memory before sending it to the target. ASP.NET Core supports reading and writing streams from requests and responses.

To get a BlobClient by name, call the getBlobClient methods on the BlobContainerClient that contains the blob, by using the name of the blob. A BlobClient object allows you to interact with the blob by uploading, downloading, or managing the blob in Blob Storage.

We recommend using streams instead of in-memory structures like byte arrays or strings when you're working with large data objects. This approach avoids buffering the full content in memory before sending it to the target.

Create new blobs

To create a new blob, call one of the Upload methods on a reference to a blob that doesn't exist in storage. This approach does two things: creates the blob in storage and uploads the data.

BlobClient blobClient = containerClient.GetBlobClient(name);

var response = blobClient.UploadAsync(fileStream);

To create a new blob, call one of the upload methods on a reference to a blob that doesn't exist in storage. This approach does two things: creates the blob in storage, and uploads the data.

BlobClient blobClient = blobContainerClient.getBlobClient(name);
blobClient.upload(inputStream, contentLength);

Exercise

Finish your app by adding the upload and download code, then deploy it to Azure App Service for testing.

Upload

To upload a blob, you implement the BlobStorage.Save method. First, you get a BlobClient object that represents the blob by calling GetBlobClient on a BlobContainerClient. Then, use the UploadAsync method on the BlobClient to save the Stream of data passed to this method up to Blob Storage.

  • In the editor, in BlobStorage.cs, replace Save with the following code. Use CTRL+S to save your work.

    public Task Save(Stream fileStream, string name)
    {
        BlobServiceClient blobServiceClient = new BlobServiceClient(storageConfig.ConnectionString);
    
        // Get the container (folder) the file will be saved in
        BlobContainerClient containerClient = blobServiceClient.GetBlobContainerClient(storageConfig.FileContainerName);
    
        // Get the Blob Client used to interact with (including create) the blob
        BlobClient blobClient = containerClient.GetBlobClient(name);
    
        // Upload the blob
        return blobClient.UploadAsync(fileStream);
    }
    

    Note

    The stream-based upload code shown here is more efficient than reading the file into a byte array before sending it to Blob Storage. However, the ASP.NET Core IFormFile technique you use to get the file from the client is not a true end-to-end streaming implementation. It is only appropriate for handling uploads of small files.

To upload a blob, you implement the BlobStorage.save method. First, you get a BlobClient object that represents the blob by calling getBlobClient on a BlobContainerClient. Then, use the upload method on the BlobClient to save the InputStream of data passed to this method up to Blob Storage.

  • In the editor, in BlobStorage.java, replace save with the following code.

    public void save(String name, InputStream inputStream, long contentLength) {
        BlobClient blobClient = blobContainerClient.getBlobClient(name);
        blobClient.upload(inputStream, contentLength);
    }
    

Download

To download a file, the OpenReadAsync method on the BlobClient object is returned. This method returns a Stream, which means that your code doesn't need to load all of the bytes from Blob Storage at once. You just need to return a reference to the blob stream, which ASP.NET Core can use to stream the file to the browser.

  • Replace Load with this code and save your work using CTRL + S.

    public Task<Stream> Load(string name)
    {
        BlobServiceClient blobServiceClient = new BlobServiceClient(storageConfig.ConnectionString);
    
        // Get the container the blobs are saved in
        BlobContainerClient containerClient = blobServiceClient.GetBlobContainerClient(storageConfig.FileContainerName);
    
        // Get a client to operate on the blob so we can read it.
        BlobClient blobClient = containerClient.GetBlobClient(name);
    
        return blobClient.OpenReadAsync();
    }
    

To download a file, use the openInputStream method on the BlobClient. This method returns an InputStream, which means that your code doesn't need to load all of the bytes from Blob Storage at once. You just need to return a reference to the blob stream, which IndexBean can use to stream the content to the browser.

Replace read with this code and save your work.

public InputStream read(String name) {
    BlobClient blobClient = blobContainerClient.getBlobClient(name);
    return blobClient.openInputStream();
}

Deploy and run in Azure

Your app is finished. Deploy it and see it work.

  1. Create an App Service app and configure it with app settings for your storage account connection string and container name. Get the storage account's connection string with az storage account show-connection-string, and set the name of the container to be files.

    The app name needs to be globally unique. Choose your own name to fill in <your-unique-app-name>. Use the storage account name you created previously to replace <your-unique-storage-account-name>. Run each of the following commands in order in Azure CLI:

    az appservice plan create \
    --name blob-exercise-plan \
    --resource-group "<rgn>[sandbox resource group name]</rgn>" \
    --sku FREE --location eastus
    
    az webapp create \
    --name <your-unique-app-name> \
    --plan blob-exercise-plan \
    --resource-group "<rgn>[sandbox resource group name]</rgn>"
    
    CONNECTIONSTRING=$(az storage account show-connection-string \
    --name <your-unique-storage-account-name> \
    --resource-group "<rgn>[sandbox resource group name]</rgn>" \
    --output tsv)
    
    az webapp config appsettings set \
    --name <your-unique-app-name> --resource-group "<rgn>[sandbox resource group name]</rgn>" \
    --settings AzureStorageConfig:ConnectionString=$CONNECTIONSTRING AzureStorageConfig:FileContainerName=files
    
  2. Deploy your app. The following commands publish the site to the pub folder, zip it up into site.zip, and deploy the zip to App Service.

    Note

    Make sure your shell is still in the mslearn-store-data-in-azure/store-app-data-with-azure-blob-storage/src/start directory before running the following commands. You can use cd mslearn-store-data-in-azure/store-app-data-with-azure-blob-storage/src/start to change directory to this location.

    dotnet publish -o pub
    cd pub
    zip -r ../site.zip *
    
    az webapp deploy \
    --src-path ../site.zip \
    --resource-group "<rgn>[sandbox resource group name]</rgn>" \
    --name <your-unique-app-name>
    

    To see the running app, in a browser, open https://<your-unique-app-name>.azurewebsites.net. It should look like the following image.

    Screenshot of the FileUploader web app for C#.

  3. Try uploading and downloading some files to test the app. After you upload a few files, to see the blobs in the container, run the following code in the shell. Replace <your-unique-storage-account-name> with the storage account name that you created earlier in the module:

    az storage blob list --account-name <your-unique-storage-account-name> --container-name files --query [].{Name:name} --output table
    

Your app is finished. Deploy it and see it work. Use the Maven Plugin for Azure App Service to create an App Service app, configure it and deploy.

  1. In the editor, open the file pom.xml and add the following plugins under the build xml tag.

    <plugins>
       <plugin>
         <groupId>com.microsoft.azure</groupId>
         <artifactId>azure-webapp-maven-plugin</artifactId>
         <version>2.3.0</version>
         <configuration>
           <schemaVersion>v2</schemaVersion>
           <subscriptionId>${env.AZ_SUBSCRIPTION_ID}</subscriptionId>
           <resourceGroup>${env.AZ_RESOURCE_GROUP}</resourceGroup>
           <appName>${env.AZ_APP_NAME}</appName>
           <pricingTier>${env.AZ_PRICING_TIER}</pricingTier>
           <region>${env.AZ_REGION}</region>
           <runtime>
             <os>Linux</os>
             <javaVersion>Java 11</javaVersion>
             <webContainer>Tomcat 9.0</webContainer>
           </runtime>
           <deployment>
             <resources>
               <resource>
                 <directory>${project.basedir}/target</directory>
                 <includes>
                   <include>*.war</include>
                 </includes>
               </resource>
             </resources>
           </deployment>
                 <appSettings>
                   <property>
                      <name>STORAGE_CONNECTION_STRING</name>
                      <value>${env.AZ_STORAGE_CONNECTION_STRING}</value>
                   </property>
                   <property>
                      <name>STORAGE_CONTAINER_NAME</name>
                      <value>${env.AZ_STORAGE_CONTAINER_NAME}</value>
                   </property>
                </appSettings>
         </configuration>
       </plugin>
       <plugin>  
         <groupId>org.apache.maven.plugins</groupId>  
         <artifactId>maven-war-plugin</artifactId>  
         <version>3.3.2</version>  
       </plugin> 
     </plugins>
    
  2. The following commands prepare environment variables for the Maven Plugin for Azure App Service. Extract the storage account's connection string with az storage account show-connection-string, the subscription ID with az account show, and set region, pricing, container name, and app name. The app name needs to be globally unique. Choose your own name to fill in <your-unique-app-name>.

    export AZ_SUBSCRIPTION_ID=$(az account show --query id --output tsv)
    export AZ_RESOURCE_GROUP="<rgn>[sandbox resource group name]</rgn>"
    export AZ_REGION=eastus
    export AZ_APP_NAME=<your-unique-app-name>
    export AZ_PRICING_TIER=F1
    export AZ_STORAGE_CONNECTION_STRING=$(az storage account show-connection-string --name <your-unique-storage-account-name> --output tsv)
    export AZ_STORAGE_CONTAINER_NAME=files
    

    Tip

    The minimum recommended tier for deploying real world Java applications is any Premium V2 Service Plan.

  3. Deploy your app. The following command builds the app into ROOT.war and deploys the WAR file to App Service. The Maven Plugin for Azure App Service provisions resources at the first deployment attempt.

    Note

    Make sure your shell is still in the mslearn-store-data-in-azure/store-java-ee-application-data-with-azure-blob-storage/start directory before running the following commands. You can use cd mslearn-store-data-in-azure/store-java-ee-application-data-with-azure-blob-storage/start to change directory to this location.

    mvn clean package azure-webapp:deploy
    

    To see the running app, in a browser, open https://<your-unique-app-name>.azurewebsites.net. It should look like the following image.

    Screenshot of the FileUploader web app for Java.

    Tip

    This module uses Maven Plugin for Azure App Service to deploy the app on Tomcat 9 on Azure App Service. To learn about other options see the Further Reading section at the end of this module.

  4. Try uploading and downloading some files to test the app. After you upload a few files, to see the blobs in the container, run the following code in the shell.

    az storage blob list --account-name <your-unique-storage-account-name> --container-name files --query [].{Name:name} --output table