In this exercise, you create a guest book application and execute it in
the local development fabric. For this purpose, you will use the Windows
Azure Tools for Microsoft Visual Studio to create the project using the
Cloud Service project template. These tools extend Visual Studio to
enable the creation, building and running of Windows Azure services. You
will continue to work with this project throughout the remainder of the
lab.
Task 1 – Creating the Visual Studio Project
In this task, you create a new Cloud Service project in Visual Studio.- Open Visual Studio as administrator from Start | All Programs | Microsoft Visual Studio 2010 by right clicking the Microsoft Visual Studio 2010 shortcut and choosing Run as administrator.
- If the User Account Control dialog appears, click Yes.
- From the File menu, choose New and then Project.
- In the New Project dialog, expand the language of your preference (Visual C# or Visual Basic) in the Installed Templates list and select Cloud. Choose the Windows Azure Project template, set the Name of the project to GuestBook, set the location to \Source\Ex1-BuildingYourFirstWindowsAzureApp\[CS|VB], change the solution name to Begin, and ensure that Create directory for solution is checked. Click OK to create the project.
Figure 1Creating a new Windows Azure Cloud Service project (C#)
Figure 2Creating a new Windows Azure Cloud Service project (Visual Basic)Note:Windows Azure supports the .NET Framework 4.0. If you use Visual Studio 2010 to create the project, you can select this version for the target framework and take advantage of its new features. - In the New Windows Azure Project dialog, inside the Roles panel, expand the tab for the language of your choice (Visual C# or Visual Basic), select ASP.NET Web Role from
the list of available roles and click the arrow (>) to add an
instance of this role to the solution. Before closing the dialog, select
the new role in the right panel, click the pencil icon and rename the
role as GuestBook_WebRole. Click OK to create the cloud service solution.
Figure 3Assigning roles to the cloud service project (C#)
Figure 4Assigning roles to the cloud service project (Visual Basic) - In Solution Explorer, review the structure of the created solution.
Figure 5Solution Explorer showing the GuestBook application (C#)
Figure 6Solution Explorer showing the GuestBook application (Visual Basic)Note:The generated solution contains two separate projects. The first project, named GuestBook, holds the configuration for the web and worker roles that compose the cloud application. It includes the service definition file, ServiceDefinition.csdef, which contains metadata needed by the Windows Azure fabric to understand the requirements of your application, such as which roles are used, their trust level, the endpoints exposed by each role, the local storage requirements and the certificates used by the roles. The service definition also establishes configuration settings specific to the application. The service configuration files specify the number of instances to run for each role and sets the value of configuration settings defined in the service definition file. This separation between service definition and configuration allows you to update the settings of a running application by uploading a new service configuration file.You can create many configuration files, each one intended for a specific scenario such as production, development, or QA, and select which to use when publishing the application. By default, Visual Studio creates two files ServiceConfiguration.Local.cscfg and ServiceConfiguration.Cloud.cscfg.The Roles node in the cloud service project enables you to configure what roles the service includes (web, worker or both) as well as which projects to associate with these roles. Adding and configuring roles through the Roles node will update the ServiceDefinition.csdef and ServiceConfiguration.cscfg files.
The second project, named GuestBook_WebRole, is a standard ASP.NET Web Application project template modified for the Windows Azure environment. It contains an additional class that provides the entry point for the web role and contains methods to manage the initialization, starting, and stopping of the role.
Task 2 – Creating a Data Model for Entities in Table Storage
The application stores guest book entries in Windows Azure Table storage. The Table service offers semi-structured storage in the form of tables that contain collections of entities. Entities have a primary key and a set of properties, where a property is a name, typed-value pair.
In addition to the properties required by your model, every entity in Table Storage has two key properties: the PartitionKey and the RowKey. These properties together form the table's primary key and uniquely identify each entity in the table. Entities also have a Timestamp system property, which allows the service to keep track of when an entity was last modified. This field is intended for system use and should not be accessed by the application. The Table Storage client API provides a TableServiceEntity class that defines the necessary properties. Although you can use the TableServiceEntity class as the base class for your entities, this is not required.
The Table service API is compliant with the REST API provided by WCF Data Services (formerly ADO.NET Data Services Framework) allowing you to use the WCF Data Services Client Library (formerly .NET Client Library) to work with data in Table Storage using .NET objects.
The Table service does not enforce any schema for tables making it possible for two entities in the same table to have different sets of properties. Nevertheless, the GuestBook application uses a fixed schema to store its data.
In order to use the WCF Data Services Client Library to access data in table storage, you need to create a context class that derives from TableServiceContext, which itself derives from DataServiceContext in WCF Data Services. The Table Storage API allows applications to create the tables that they use from these context classes. For this to happen, the context class must expose each required table as a property of type IQueryable<SchemaClass>, where SchemaClass is the class that models the entities stored in the table.
In this task, you model the schema of the entities stored by the GuestBook application and create a context class to use WCF Data Services to access the information in table storage. To complete the task, you create an object that can be data bound to data controls in ASP.NET and implements the basic data access operations: read, update, and delete.
- Create a new project for the schema classes. To create the project, in the File menu, point to Add and then select New Project.
- In the Add New Project dialog, expand the language of your choice under the Installed Templates tree view, select the Windows category, and then choose the Class Library project template. Set the name to GuestBook_Data, leave the proposed location inside the solution folder unchanged, and then click OK.
Figure 7Creating a class library for GuestBook entities (C#)
Figure 8Creating a class library for GuestBook entities (Visual Basic) - Delete the default class file generated by the class library template. To do this, right-click Class1.cs (for Visual C# Projects) or Class1.vb (for Visual Basic Projects) and choose Delete. Click OK in the confirmation dialog.
- Add a reference to the .NET Client Library for WCF Data Services in the GuestBook_Data project. In Solution Explorer, right-click the GuestBook_Data project node, select Add Reference, click the .NET tab, select the System.Data.Services.Client component and click OK.
Figure 9Adding a reference to the System.Data.Service.Client component - Repeat the previous step to add a reference to the Windows Azure storage client API assembly, this time choosing the Microsoft.WindowsAzure.StorageClient component instead.
- Before you can store an entity in a table, you must first define its schema. To do this, right-click GuestBook_Data in Solution Explorer, point to Add and select Class. In the Add New Item dialog, set the name to GuestBookEntry.cs (for Visual C# projects) or GuestBookEntry.vb (for Visual Basic projects) and click Add.
Figure 10Adding the GuestBookEntry class (C#)
Figure 11Adding the GuestBookEntry class (Visual Basic) - At the top of the file, insert the following namespace declaration to import the types contained in the Microsoft.WindowsAzure.StorageClient namespace.C#
- using Microsoft.WindowsAzure.StorageClient;
Visual BasicImports Microsoft.WindowsAzure.StorageClient
- If not already opened, open the GuestBookEntry.cs file (for Visual C# projects) or GuestBookEntry.vb file (for Visual Basic projects) and then update the declaration of the GuestBookEntry class to make it public and derive from the TableServiceEntity class.Note:In Visual Basic, the template for a new class already declares the class as Public.C#
- public class GuestBookEntry
- : Microsoft.WindowsAzure.StorageClient.TableServiceEntity
{
}
Visual BasicPublic Class GuestBookEntry Inherits Microsoft.WindowsAzure.StorageClient.TableServiceEntity
End Class
Note:TableServiceEntity is a class found in the Storage Client API. This class defines the PartititionKey, RowKey and TimeStamp system properties required by every entity stored in a Windows Azure table.Together, the PartitionKey and RowKey define the DataServiceKey that uniquely identifies every entity within a table. - Add a default constructor to the GuestBookEntry class that initializes its PartitionKey and RowKey properties.(Code Snippet – Introduction to Windows Azure - Ex1 GuestBookEntry constructor – CS)
C#- public GuestBookEntry()
- {
- PartitionKey = DateTime.UtcNow.ToString("MMddyyyy");
- // Row key allows sorting, so we make sure the rows come back in time order.
- RowKey = string.Format("{0:10}_{1}", DateTime.MaxValue.Ticks - DateTime.Now.Ticks, Guid.NewGuid());
- }
Visual BasicPublic Sub New() PartitionKey = DateTime.UtcNow.ToString("MMddyyyy") ' Row key allows sorting, so we make sure the rows come back in time order. RowKey = String.Format("{0:10}_{1}", DateTime.MaxValue.Ticks - DateTime.Now.Ticks, Guid.NewGuid()) End Sub
Note:To partition the data, the GuestBook application uses the date of the entry as the PartitionKey, which means that there will be a separate partition for each day of guest book entries. In general, you choose the value of the partition key to ensure load balancing of the data across storage nodes. The RowKey is a reverse DateTime field with a GUID appended for uniqueness. Tables within partitions are sorted in RowKey order, so this will sort the tables into the correct order to be shown on the home page, with the newest entry shown at the top. - To complete the definition of the GuestBookEntry class, add properties for Message, GuestName, PhotoUrl, and ThumbnailUrl to hold information about the entry. (Code Snippet – Introduction to Windows Azure - Ex1 Table Schema Properties – CS)
C#- public string Message { get; set; }
- public string GuestName { get; set; }
- public string PhotoUrl { get; set; }
- public string ThumbnailUrl { get; set; }
Visual BasicPublic Property Message As String Public Property GuestName As String Public Property PhotoUrl As String Public Property ThumbnailUrl As String
- Save the GuestBookEntry.cs file (for Visual C# projects) or GuestBookEntry.vb file (for Visual Basic projects).
- Next, you need to create the context class required to access the GuestBook table using WCF Data Services. To do this, in Solution Explorer, right-click the GuestBook_Data project, point to Add and select Class. In the Add New Item dialog, set the Name to GuestBookDataContext.cs (for Visual C# projects) or GuestBookDataContext.vb (for Visual Basic projects) and click Add.
- In the new class file, update the declaration of the new class to make it public and inherit the TableServiceContext class.Note:In Visual Basic, the template for a new class already declares the class as Public.C#
- public class GuestBookDataContext
- : Microsoft.WindowsAzure.StorageClient.TableServiceContext
{
}
Visual BasicPublic Class GuestBookDataContext Inherits Microsoft.WindowsAzure.StorageClient.TableServiceContext
End Class
- Now, add a default constructor to initialize the base class with storage account information.(Code Snippet – Introduction to Windows Azure - Ex1 GuestBookDataContext Class – CS)
C#public class GuestBookDataContext
: Microsoft.WindowsAzure.StorageClient.TableServiceContext
{
- public GuestBookDataContext(string baseAddress, Microsoft.WindowsAzure.StorageCredentials credentials)
- : base(baseAddress, credentials)
- { }
}
Visual BasicPublic Class GuestBookDataContext
Inherits Microsoft.WindowsAzure.StorageClient.TableServiceContext
Public Sub New(ByVal baseAddress As String, ByVal credentials As Microsoft.WindowsAzure.StorageCredentials) MyBase.New(baseAddress, credentials) End Sub
End Class
Note:You can find the TableServiceContext class in the storage client API. This class derives from DataServiceContext in WCF Data Services and manages the credentials required to access your storage account as well as providing support for a retry policy for its operations. - Add a property to the GuestBookDataContext class to expose the GuestBookEntry table. To do this, insert the following (highlighted) code into the class. (Code Snippet – Introduction to Windows Azure - Ex1 GuestBookEntry Property – CS)
C#public class GuestBookDataContext
: Microsoft.WindowsAzure.StorageClient.TableServiceContext
{
...
- public IQueryable<GuestBookEntry> GuestBookEntry
- {
- get
- {
- return this.CreateQuery<GuestBookEntry>("GuestBookEntry");
- }
- }
}
Visual BasicPublic Class GuestBookDataContext
Inherits Microsoft.WindowsAzure.StorageClient.TableServiceContext
...
Public ReadOnly Property GuestBookEntry() As IQueryable(Of GuestBookEntry) Get Return Me.CreateQuery(Of GuestBookEntry)("GuestBookEntry") End Get End Property
End Class
Note:You can use the CreateTablesFromModel method in the CloudTableClient class to create the tables needed by the application. When you supply a DataServiceContext (or TableServiceContext) derived class to this method, it locates any properties that return an IQueryable<T>, where the generic parameter T identifies the class that models the table schema, and creates a table in storage named after the property. - Finally, you need to implement an object that can be bound to data controls in ASP.NET. In Solution Explorer, right-click GuestBook_Data, point to Add, and select Class. In the Add New Item dialog, set the name to GuestBookDataSource.cs (for Visual C# projects) or GuestBookDataSource.vb (for Visual Basic projects) and click Add.
- In the new class file, add the following namespace declarations to import the types contained in the Microsoft.WindowsAzure and Microsoft.WindowsAzure.StorageClient namespaces.C#
- using Microsoft.WindowsAzure;
- using Microsoft.WindowsAzure.StorageClient;
Visual BasicImports Microsoft.WindowsAzure Imports Microsoft.WindowsAzure.StorageClient
- In the GuestBookDataSource class, make the class public and define member fields for the data context and the storage account information, as shown below.Note:In Visual Basic, the template for a new class already declares the class as Public.
C#public class GuestBookDataSource
{
- private static CloudStorageAccount storageAccount;
- private GuestBookDataContext context;
}
Visual BasicPublic Class GuestBookDataSource
Private Shared storageAccount As CloudStorageAccount Private context As GuestBookDataContext
End Class
- Now, add a static
(Shared in Visual Basic) constructor to the data source class as shown
in the following (highlighted) code. This code creates the tables from
the GuestBookDataContext class.(Code Snippet – Introduction to Windows Azure - Ex1 GuestBookDataSource Static Constructor – CS)
C#public class GuestBookDataSource
{
...
- static GuestBookDataSource()
- {
- storageAccount = CloudStorageAccount.FromConfigurationSetting("DataConnectionString");
- CloudTableClient.CreateTablesFromModel(
- typeof(GuestBookDataContext),
- storageAccount.TableEndpoint.AbsoluteUri,
- storageAccount.Credentials);
- }
}
Visual BasicPublic Class GuestBookDataSource
...
Shared Sub New() storageAccount = CloudStorageAccount.FromConfigurationSetting("DataConnectionString") CloudTableClient.CreateTablesFromModel(GetType(GuestBookDataContext), storageAccount.TableEndpoint.AbsoluteUri, storageAccount.Credentials) End Sub
End Class
Note:The static (Shared in Visual Basic) constructor initializes the storage account by reading its settings from the configuration and then uses the CreateTablesFromModel method in the CloudTableClient class to create the tables used by the application from the model defined by the GuestBookDataContext class. By using the static constructor, you ensure that this initialization task is executed only once. - Add a default constructor to the GuestBookDataSource class to initialize the data context class used to access table storage.(Code Snippet – Introduction to Windows Azure - Ex1 GuestBookDataSource Constructor – CS)
C#public class GuestBookDataSource
{
...
- public GuestBookDataSource()
- {
- this.context = new GuestBookDataContext(storageAccount.TableEndpoint.AbsoluteUri, storageAccount.Credentials);
- this.context.RetryPolicy = RetryPolicies.Retry(3, TimeSpan.FromSeconds(1));
- }
}
Visual BasicPublic Class GuestBookDataSource
...
Public Sub New() Me.context = New GuestBookDataContext(storageAccount.TableEndpoint.AbsoluteUri, storageAccount.Credentials) Me.context.RetryPolicy = RetryPolicies.Retry(3, TimeSpan.FromSeconds(1)) End Sub
End Class
- Next, insert the following method to return the contents of the GuestBookEntry table. (Code Snippet – Introduction to Windows Azure - Ex1 GuestBookDataSource Select – CS)
C#public class GuestBookDataSource
{
...
- public IEnumerable<GuestBookEntry> GetGuestBookEntries()
- {
- var results = from g in this.context.GuestBookEntry
- where g.PartitionKey == DateTime.UtcNow.ToString("MMddyyyy")
- select g;
- return results;
- }
}
Visual BasicPublic Class GuestBookDataSource
...
Public Function GetGuestBookEntries() As IEnumerable(Of GuestBookEntry) Dim results = From g In Me.context.GuestBookEntry _ Where g.PartitionKey = DateTime.UtcNow.ToString("MMddyyyy") _ Select g Return results End Function
End Class
Note:The GetGuestBookEntries method retrieves today's guest book entries by constructing a LINQ statement that filters the retrieved information using the current date as the partition key value. The web role uses this method to bind to a data grid and display the guest book. - Now, add the following method to insert new entries into the GuestBookEntry table.(Code Snippet – Introduction to Windows Azure - Ex1 GuestBookDataSource AddGuestBookEntry – CS)
C#public class GuestBookDataSource
{
...
- public void AddGuestBookEntry(GuestBookEntry newItem)
- {
- this.context.AddObject("GuestBookEntry", newItem);
- this.context.SaveChanges();
- }
}
Visual BasicPublic Class GuestBookDataSource
...
Public Sub AddGuestBookEntry(ByVal newItem As GuestBookEntry) Me.context.AddObject("GuestBookEntry", newItem) Me.context.SaveChanges() End Sub
End Class
Note:This method adds a new GuestBookEntry object to the data context and then calls SaveChanges to write the entity to storage. - Finally, add a method to the data source class to update the thumbnail URL property for an entry.(Code Snippet – Introduction to Windows Azure - Ex1 GuestBookDataSource UpdateImageThumbnail – CS)
C#public class GuestBookDataSource
{
...
- public void UpdateImageThumbnail(string partitionKey, string rowKey, string thumbUrl)
- {
- var results = from g in this.context.GuestBookEntry
- where g.PartitionKey == partitionKey && g.RowKey == rowKey
- select g;
- var entry = results.FirstOrDefault<GuestBookEntry>();
- entry.ThumbnailUrl = thumbUrl;
- this.context.UpdateObject(entry);
- this.context.SaveChanges();
- }
}
Visual BasicPublic Class GuestBookDataSource
...
Public Sub UpdateImageThumbnail(ByVal partitionKey As String, ByVal rowKey As String, ByVal thumbUrl As String) Dim results = From g In Me.context.GuestBookEntry _ Where g.PartitionKey = partitionKey AndAlso g.RowKey = rowKey _ Select g Dim entry = results.FirstOrDefault() entry.ThumbnailUrl = thumbUrl Me.context.UpdateObject(entry) Me.context.SaveChanges() End Sub
End Class
Note:The UpdateImageThumbnail method locates an entry using its partition key and row key; it updates the thumbnail URL, notifies the data context of the update, and then saves the changes. - Save the GuestBookDataSource.cs file (for Visual C# projects) or GuestBookDataSource.vb file (for Visual Basic projects).
Task 3 – Creating a Web Role to Display the Guest Book and Process User Input
In this task, you update the web role project that you generated in Task 1, when you created the Windows Azure Cloud Service solution. This involves updating the UI to render the list of guest book entries. For this purpose, you will find a page that has the necessary elements in the Assets folder of this exercise, which you will add to the project. Next, you implement the code necessary to store submitted entries in table storage and images in blob storage. To complete this task, you configure the storage account used by the Web role.
- Add a reference in the web role to the GuestBook_Data project. In Solution Explorer, right-click the GuestBook_WebRole project node and select Add Reference, switch to the Projects tab, select the GuestBook_Data project, and then click then OK.
- The web role template generates a default page. You will replace it with another page that contains the UI of the guest book application. To delete the page, in Solution Explorer, right-click Default.aspx in the GuestBook_WebRole project and select Delete.
- Add the main page and its associated assets to the web role. To do this, right-click GuestBook_WebRole in Solution Explorer, point to Add and select Existing Item. In the Add Existing Item dialog, browse to the Assets folder in \Source\Ex1-BuildingYourFirstWindowsAzureApp for the language of your project (Visual C# or Visual Basic), hold the CTRL key down while you select every file in this folder and click Add.Note:The Assets folder contains five files that you need to add to the project, a Default.aspx file with its code-behind and designer files, a CSS file, and an image file.
- Open the code-behind file for the main page in the GuestBook_WebRole project. To do this, right-click the Default.aspx file in Solution Explorer and select View Code.
- In the code-behind file, insert the following namespace declarations.(Code Snippet– Introduction to Windows Azure - Ex1 Web Role Namespace Declarations – CS)
C#- using System.IO;
- using System.Net;
- using Microsoft.WindowsAzure;
- using Microsoft.WindowsAzure.ServiceRuntime;
- using Microsoft.WindowsAzure.StorageClient;
- using GuestBook_Data;
Visual BasicImports System.IO Imports System.Net Imports Microsoft.WindowsAzure Imports Microsoft.WindowsAzure.ServiceRuntime Imports Microsoft.WindowsAzure.StorageClient Imports GuestBook_Data
- Declare the following member fields in the _Default class.(Code Snippet – Introduction to Windows Azure - Ex1 Web Role Member Fields – CS)
C#public partial class _Default : System.Web.UI.Page
{
- private static bool storageInitialized = false;
- private static object gate = new Object();
- private static CloudBlobClient blobStorage;
...
}
Visual BasicPartial Public Class _Default
Inherits System.Web.UI.Page
Private Shared storageInitialized As Boolean = False Private Shared gate As New Object() Private Shared blobStorage As CloudBlobClient
...
End Class
- Locate the SignButton_Click event handler in the code-behind file and insert the following code.(Code Snippet– Introduction to Windows Azure - Ex1 SignButton_Click – CS)
C#public partial class _Default : System.Web.UI.Page
{
...
protected void SignButton_Click(object sender, EventArgs e)
{
- if (FileUpload1.HasFile)
- {
- InitializeStorage();
- // upload the image to blob storage
- string uniqueBlobName = string.Format("guestbookpics/image_{0}{1}", Guid.NewGuid().ToString(), Path.GetExtension(FileUpload1.FileName));
- CloudBlockBlob blob = blobStorage.GetBlockBlobReference(uniqueBlobName);
- blob.Properties.ContentType = FileUpload1.PostedFile.ContentType;
- blob.UploadFromStream(FileUpload1.FileContent);
- System.Diagnostics.Trace.TraceInformation("Uploaded image '{0}' to blob storage as '{1}'", FileUpload1.FileName, uniqueBlobName);
- // create a new entry in table storage
- GuestBookEntry entry = new GuestBookEntry() { GuestName = NameTextBox.Text, Message = MessageTextBox.Text, PhotoUrl = blob.Uri.ToString(), ThumbnailUrl = blob.Uri.ToString() };
- GuestBookDataSource ds = new GuestBookDataSource();
- ds.AddGuestBookEntry(entry);
- System.Diagnostics.Trace.TraceInformation("Added entry {0}-{1} in table storage for guest '{2}'", entry.PartitionKey, entry.RowKey, entry.GuestName);
- }
- NameTextBox.Text = "";
- MessageTextBox.Text = "";
- DataList1.DataBind();
}
}
Visual BasicPublic Class _Default
Inherits System.Web.UI.Page
...
Protected Sub SignButton_Click(ByVal sender As Object, ByVal e As EventArgs) Handles SignButton.Click
If FileUpload1.HasFile Then InitializeStorage() ' upload the image to blob storage Dim uniqueBlobName As String = String.Format("guestbookpics/image_{0}{1}", Guid.NewGuid().ToString(), Path.GetExtension(FileUpload1.FileName)) Dim blob As CloudBlockBlob = blobStorage.GetBlockBlobReference(uniqueBlobName) blob.Properties.ContentType = FileUpload1.PostedFile.ContentType blob.UploadFromStream(FileUpload1.FileContent) System.Diagnostics.Trace.TraceInformation("Uploaded image '{0}' to blob storage as '{1}'", FileUpload1.FileName, uniqueBlobName) ' create a new entry in table storage Dim entry As New GuestBookEntry() With {.GuestName = NameTextBox.Text, .Message = MessageTextBox.Text, .PhotoUrl = blob.Uri.ToString(), .ThumbnailUrl = blob.Uri.ToString()} Dim ds As New GuestBookDataSource() ds.AddGuestBookEntry(entry) System.Diagnostics.Trace.TraceInformation("Added entry {0}-{1} in table storage for guest '{2}'", entry.PartitionKey, entry.RowKey, entry.GuestName) End If NameTextBox.Text = "" MessageTextBox.Text = "" DataList1.DataBind()
End Sub
End Class
Note:To process a new guest book entry after the user submits the page, the handler first calls the InitializeStorage method to ensure that the blob container used to store images exists and allows public access. You will implement this method shortly. It then obtains a reference to the blob container, generates a unique name and creates a new blob, and then uploads the image submitted by the user into this blob. Notice that the method initializes the ContentType property of the blob from the content type of the file submitted by the user. When the guest book page reads the blob back from storage, the response returns this content type, allowing a page to display the image contained in the blob simply by referring to its URL.
After that, it creates a new GuestBookEntry entity, which is the entity you defined in the previous task, initializes it with the information submitted by the user, and then uses the GuestBookDataSource class to save the entry to table storage using the .NET Client Library for WCF Data Services.
Finally, it data binds the guest book entries list to refresh its contents. - Update the body of the Timer1_Tick method with the code shown below. (Code Snippet– Introduction to Windows Azure - Ex1 Timer1_Tick – CS)
C#public partial class _Default : System.Web.UI.Page
{
...
protected void Timer1_Tick(object sender, EventArgs e)
{
- DataList1.DataBind();
}
}
Visual BasicPublic Class _Default
Inherits System.Web.UI.Page
...
Protected Sub Timer1_Tick(ByVal sender As Object, ByVal e As EventArgs) Handles Timer1.Tick
DataList1.DataBind()
End Sub
End Class
Note:The timer periodically forces the page to refresh the contents of the guest book entries list. - Locate the Page_Load event handler and update its body with the following code to enable the page refresh timer.(Code Snippet – Introduction to Windows Azure - Ex1 Page_Load – CS)
C#public partial class _Default : System.Web.UI.Page
{
...
protected void Page_Load(object sender, EventArgs e)
{
- if (!Page.IsPostBack)
- {
- Timer1.Enabled = true;
- }
}
}
Visual BasicPublic Class _Default
Inherits System.Web.UI.Page
...
Protected Sub Page_Load(ByVal sender As Object, ByVal e As System.EventArgs) Handles Me.Load
If Not (Page.IsPostBack) Then Timer1.Enabled = True End If
End Sub
End Class
- Implement the InitializeStorage method by replacing its body with the following (highlighted) code.(Code Snippet – Introduction to Windows Azure - Ex1 InitializeStorage – CS)
C#public partial class _Default : System.Web.UI.Page
{
...
private void InitializeStorage()
{
- if (storageInitialized)
- {
- return;
- }
- lock (gate)
- {
- if (storageInitialized)
- {
- return;
- }
- try
- {
- // read account configuration settings
- var storageAccount = CloudStorageAccount.FromConfigurationSetting("DataConnectionString");
- // create blob container for images
- blobStorage = storageAccount.CreateCloudBlobClient();
- CloudBlobContainer container = blobStorage.GetContainerReference("guestbookpics");
- container.CreateIfNotExist();
- // configure container for public access
- var permissions = container.GetPermissions();
- permissions.PublicAccess = BlobContainerPublicAccessType.Container;
- container.SetPermissions(permissions);
- }
- catch (WebException)
- {
- throw new WebException("Storage services initialization failure. "
- + "Check your storage account configuration settings. If running locally, "
- + "ensure that the Development Storage service is running.");
- }
- storageInitialized = true;
- }
}
}
Visual BasicPublic Class _Default
Inherits System.Web.UI.Page
...
Private Sub InitializeStorage()
If storageInitialized Then Return End If SyncLock gate If storageInitialized Then Return End If Try ' read account configuration settings Dim storageAccount = CloudStorageAccount.FromConfigurationSetting("DataConnectionString") ' create blob container for images blobStorage = storageAccount.CreateCloudBlobClient() Dim container As CloudBlobContainer = blobStorage.GetContainerReference("guestbookpics") container.CreateIfNotExist() ' configure container for public access Dim permissions = container.GetPermissions() permissions.PublicAccess = BlobContainerPublicAccessType.Container container.SetPermissions(permissions) Catch e1 As WebException Throw New WebException("Storage services initialization failure. " _ & "Check your storage account configuration settings. If running locally, " _ & "ensure that the Development Storage service is running.") End Try storageInitialized = True End SyncLock
End Sub
End Class
Note:The InitializeStorage method first ensures that it executes only once. It reads the storage account settings from the Web role configuration, creates a blob container for the images uploaded with each guest book entry and configures it for public access. - Because the
web role uses Windows Azure storage services, you need to provide your
storage account settings. To create a new setting, in Solution Explorer, expand the Roles node in the GuestBook project, double-click GuestBook_WebRole to open the properties for this role and select the Settings tab. Click Add Setting, type “DataConnectionString” in the Name column, change the Type to ConnectionString, and then click the button labeled with an ellipsis.
Figure 12Configuring the storage account settings - In the Storage Account Connection String dialog, choose the option labeled Use the Windows Azure storage emulator and then click OK.
Figure 13Creating a connection string for the storage emulatorNote:A storage account is a unique endpoint for the Windows Azure Blob, Queue, and Table services. You must create a storage account in the Management Portal to use these services. In this exercise, you use Windows Azure storage emulator, which is included in the Windows Azure SDK development environment to simulate the Blob, Queue, and Table services available in the cloud. If you are building a hosted service that employs storage services or writing any external application that calls storage services, you can test locally against the Windows Azure storage emulator.To use the storage emulator, you set the value of the UseDevelopmentStorage keyword in the connection string for the storage account to true. When you publish your application to Windows Azure, you need to update the connection string to specify storage account settings including your account name and shared key. For example,
<Setting name="DataConnectionString" value="DefaultEndpointsProtocol=https;AccountName=YourAccountName;AccountKey=YourAccountKey" />
where YourAccountName is the name of your Azure Storage account and YourAccountKey is your access key. - Press CTRL + S to save changes to the role configuration.
- Finally, you need to set up the environment for the configuration publisher. In the GuestBook_WebRole project, open the Global.asax.cs file (for Visual C# projects) or the Global.asax.vb file (for Visual Basic projects).
- At the top of the file, insert the following namespace declaration to import the types contained in the Microsoft.WindowsAzure and Microsoft.WindowsAzure.ServiceRuntime namespaces.C#
- using Microsoft.WindowsAzure;
- using Microsoft.WindowsAzure.ServiceRuntime;
Visual BasicImports Microsoft.WindowsAzure Imports Microsoft.WindowsAzure.ServiceRuntime
- Insert the following code into the Application_Start method replacing the default comment.(Code Snippet – Introduction to Windows Azure - Ex1 SetConfigurationSettingPublisher – CS)
C#void Application_Start(object sender, EventArgs e)
{
- Microsoft.WindowsAzure.CloudStorageAccount.SetConfigurationSettingPublisher((configName, configSetter) =>
- {
- configSetter(RoleEnvironment.GetConfigurationSettingValue(configName));
- });
}
Visual BasicSub Application_Start(ByVal sender As Object, ByVal e As EventArgs)
CloudStorageAccount.SetConfigurationSettingPublisher(Function(configName, configSetter) configSetter(RoleEnvironment.GetConfigurationSettingValue(configName)))
End Sub
Task 4 – Queuing Work Items for Background Processing
In preparation for the next exercise, you now update the front-end web role to dispatch work items to an Azure queue for background processing. These work items will remain in the queue until you add a worker role that picks items from the queue and generates thumbnails for each uploaded image.
- Open the code-behind file for the main page of the web role project. To do this, right-click the Default.aspx file in Solution Explorer and select View Code.
- Declare a queue client member by inserting the following (highlighted) declaration into the Default class.(Code Snippet – Introduction to Windows Azure - Ex1 CloudQueueClient member – CS)
C#public partial class _Default : System.Web.UI.Page
{
private static bool storageInitialized = false;
private static object gate = new Object();
private static CloudBlobClient blobStorage;
- private static CloudQueueClient queueStorage;
...
}
Visual BasicPublic Class _Default
Inherits System.Web.UI.Page
Private Shared storageInitialized As Boolean = False
Private Shared gate As New Object()
Private Shared blobStorage As CloudBlobClient
Private Shared queueStorage As CloudQueueClient
...
End Class
- Now,
update the storage initialization code to create the queue, if it does
not exist, and then initialize the queue reference created in the
previous step. To do this, locate the InitializeStorage
method and insert the following (highlighted) code into this method
immediately after the code that configures the blob container for public
access.(Code Snippet – Introduction to Windows Azure - Ex1 Create Queue – CS)
C#public partial class _Default : System.Web.UI.Page
{
...
private void InitializeStorage()
{
...
try
{
...
// configure container for public access
var permissions = container.GetPermissions();
permissions.PublicAccess = BlobContainerPublicAccessType.Container;
container.SetPermissions(permissions);
- // create queue to communicate with worker role
- queueStorage = storageAccount.CreateCloudQueueClient();
- CloudQueue queue = queueStorage.GetQueueReference("guestthumbs");
- queue.CreateIfNotExist();
}
catch (WebException)
{
...
}
Visual BasicPartial Public Class _Default
Inherits System.Web.UI.Page
...
Private Sub InitializeStorage()
...
Try
...
' configure container for public access
Dim permissions = container.GetPermissions()
permissions.PublicAccess = BlobContainerPublicAccessType.Container
container.SetPermissions(permissions)
' create queue to communicate with worker role queueStorage = storageAccount.CreateCloudQueueClient() Dim queue As CloudQueue = queueStorage.GetQueueReference("guestthumbs") queue.CreateIfNotExist()
Catch e1 As WebException
...
End Class
Note:The updated code creates a queue that the web role uses to submit new jobs to the worker role. - Finally, add code to post a work item to the queue. To do this, locate the SignButton_Click event handler and insert the following (highlighted) code immediately after the lines that create a new entry in table storage.(Code Snippet – Introduction to Windows Azure - Ex1 Queueing work items – CS)
C#protected void SignButton_Click(object sender, EventArgs e)
{
if (FileUpload1.HasFile)
{
...
// create a new entry in table storage
GuestBookEntry entry = new GuestBookEntry() { GuestName = NameTextBox.Text, Message = MessageTextBox.Text, PhotoUrl = blob.Uri.ToString(), ThumbnailUrl = blob.Uri.ToString() };
GuestBookDataSource ds = new GuestBookDataSource();
ds.AddGuestBookEntry(entry);
System.Diagnostics.Trace.TraceInformation("Added entry {0}-{1} in table storage for guest '{2}'", entry.PartitionKey, entry.RowKey, entry.GuestName);
- // queue a message to process the image
- var queue = queueStorage.GetQueueReference("guestthumbs");
- var message = new CloudQueueMessage(String.Format("{0},{1},{2}", blob.Uri.ToString(), entry.PartitionKey, entry.RowKey));
- queue.AddMessage(message);
- System.Diagnostics.Trace.TraceInformation("Queued message to process blob '{0}'", uniqueBlobName);
}
NameTextBox.Text = "";
MessageTextBox.Text = "";
DataList1.DataBind();
}
Visual BasicProtected Sub SignButton_Click(ByVal sender As Object, ByVal e As EventArgs) Handles SignButton.Click
If FileUpload1.HasFile Then
...
' create a new entry in table storage
Dim entry As New GuestBookEntry() With {.GuestName = NameTextBox.Text, .Message = MessageTextBox.Text, .PhotoUrl = blob.Uri.ToString(), .ThumbnailUrl = blob.Uri.ToString()}
Dim ds As New GuestBookDataSource()
ds.AddGuestBookEntry(entry)
System.Diagnostics.Trace.TraceInformation("Added entry {0}-{1} in table storage for guest '{2}'", entry.PartitionKey, entry.RowKey, entry.GuestName)
' queue a message to process the image Dim queue = queueStorage.GetQueueReference("guestthumbs") Dim message = New CloudQueueMessage(String.Format("{0},{1},{2}", blob.Uri.ToString(), entry.PartitionKey, entry.RowKey)) queue.AddMessage(message) System.Diagnostics.Trace.TraceInformation("Queued message to process blob '{0}'", uniqueBlobName)
End If
NameTextBox.Text = ""
MessageTextBox.Text = ""
DataList1.DataBind()
End Sub
Note:The updated code obtains a reference to the “guestthumbs” queue. It constructs a new message that consists of a comma-separated string with the name of the blob that contains the image, the partition key, and the row key of the entity that was added. The worker role can easily parse messages with this format. The method then submits the message to the queue.
Verification
The Windows Azure compute emulator, formerly Development Fabric or devfabric, is a simulated environment for developing and testing Windows Azure applications in your machine. In this task, you launch the GuestBook application in the emulator and create one or more guest book entries.
Among the features available in the Windows Azure Tools for Microsoft Visual Studio is a Windows Azure Storage browser that allows you to connect to a storage account and browse the blobs and tables it contains. If you are using this version of Visual Studio, you will use it during this task to examine the storage resources created by the application.
- Press F5 to execute the
service. The service builds and then launches the local Windows Azure
compute emulator. To show the Compute Emulator UI, right-click its icon
located in the system tray and select Show Compute Emulator UI.
Figure 14Showing the Compute Emulator UINote:Note: If it is the first time you run the Windows Azure Emulator, the System will show a Windows Security Alert dialog indicating the Firewall has blocked some features. Click Allow Access to continue.Note:When you use the storage emulator for the first time, it needs to execute a one-time initialization procedure to create the necessary database and tables. If this is the case, wait for the procedure to complete and examine the Development Storage Initialization dialog to ensure that it completes successfully.
Figure 15Storage emulator initialization process - Switch to Internet Explorer to view the GuestBook application.
- Add a new entry to the guest book. To do this, type your name and a message, choose an image to upload from the Pictures\Sample Pictures library, and then click the pencil icon to submit the entry.
Figure 16Windows Azure GuestBook home pageNote:It is a good idea to choose a large hi-resolution image because, once the application is complete, the guestbook service will resize uploaded images.
Figure 17GuestBook application showing an uploaded image in its original sizeIf you are using Visual Studio 2010, you can use the Windows Azure Storage Explorer to view storage resources directly from Visual Studio. This functionality is not available in Visual Studio 2008. - To open the Storage Explorer in Visual Studio 2010, open the View menu, select Server Explorer, and then expand the Windows Azure Storage node.The Windows Azure Storage
node lists the storage accounts that you have currently registered and,
by default, includes an entry for the storage emulator account labeled
as (Development).
Note:Windows Azure Storage Explorer is not available in Visual Studio 2008. - Expand the (Development) node and then the Tables node inside it. Notice that it contains a table named GuestBookEntry created by the application that should contain details for each entry.
Figure 18Viewing tables in the Windows Azure storage emulator - Double-click the GuestBookEntry node in the Windows Azure Storage explorer to show the contents of this table. The GuestBookEntry table contains information for the entry that you created earlier in this task, including its GuestName, Message, PhotoUrl, and ThumbnailUrl properties, as well as the PartitionKey, RowKey, and Timestamp properties common to all table storage entities. Notice that the PhotoUrl and ThumbnailUrl
properties are currently the same. In the next exercise, you will
modify the application to generate image thumbnails and to update the
corresponding URL.
Figure 19Viewing tables using the Windows Azure Tools for Visual Studio - Now, expand the Blobs node in the Windows Azure Storage explorer. Inside this node, you will find an entry for a container named guestbookpics that contains blobs with raw data for the images uploaded by the application.
Figure 20Viewing blobs using the Windows Azure Tools for Visual Studio - Double-click the node for the guestbookpics container to list the blobs it contains. It should include an entry for the image that you uploaded earlier.
Figure 21Viewing the contents of a blob container in Visual Studio - Each
blob in blob storage has an associated content type that Visual Studio
uses to select a suitable viewer for the blob. To display the contents
of the blob, double-click the corresponding entry in the container
listing to display the image.
Figure 22Viewing blob contents in Visual Studio - Press SHIFT + F5 to stop the debugger and shut down the deployment in the development fabric.
source:http://msdn.microsoft.com