- Download and install Visual Studio 2013
- Download and install the Web Platform Installer, v5.0 or greater
- From the Web Platform Installer, install:
- Windows Azure SDK and related Powershell utilities and command line tools
- Start a new solution (or open an existing one)
- From the NuGet package manager in your solution, ensure that you've installed the 'WindowsAzure.Storage' package, or at least have it in your cache. This is going to be required by the New Project wizard when generating the project.
- Add a new project
- In the 'Add new project' wizard, select the C# projects -> Cloud -> Windows Azure Cloud Service
- Follow this absurdly easy tutorial on implementing an ErrorHandler interceptor (behavior)
- Add logging to your application using the Microsoft Patterns & Practices Enterprise Library Logging Application Block (which can be found here).
- Create a SQL server database in Azure using this tutorial on MSDN. When designing the user roles and authentication, it's recommended that you use the ASP.NET Identity membership design and design your tables accordingly around this.
- Microsoft has provided an extension to the ASP.NET Identity membership framework specifically for EntityFramework. They recommend that you use a code-first model for generating entities.
- Log in to Azure through the management portal: http://manage.windowsazure.com/
- Configure all of your connections and permissions to the database. e.g. you may want to have multiple users: one for read-only operations, another for read-write operations.
- Generate or write your data model. This article will show you how to create a code-first data model with Entity Framework. Ensure that you include users so that you can support proper application authentication and authorization via the ASP.NET Identity membership framework.
- The link above also includes instructions on using code-first migrations for when you update your data model.
- TODO: Elaborate on how to properly set up the database when performing a code-first database design
- Add an IoC container to your WCF service to enable you to easily develop and unit test it. I've chosen to use Ninject because despite a few minutes of inital frustration, it's actually exceedingly easy to integrate with WCF, especially now that there's the Ninject WCF extensions NuGet package. To integration Ninject with your WCF service, perform the following steps:
- Install-Package Ninject.Extensions.Wcf -Version 3.2.1.0 (Ninject and Ninject.Web.Common are installed as dependencies for you automatically).
- Create a NinjectModule descendant to bind your interfaces to concrete implementations. It should look something like the following:
namespace MyService.CloudStorage.Support { using System.Diagnostics.CodeAnalysis; using System.ServiceModel; using Common.Interfaces; using global::Ninject.Modules; using global::Ninject.Syntax; using Services; using NinjectServiceHost = global::Ninject.Extensions.Wcf.NinjectServiceHost; /// <summary> /// A <see cref="NinjectModule"/> descendant for bootstrapping our application /// </summary> public class MyServiceCloudStorageNinjectModule : NinjectModule { /// <inheritdoc/> [SuppressMessage("Microsoft.StyleCop.CSharp.DocumentationRules", "SA1604:ElementDocumentationMustHaveSummary", Justification = "InheritDoc")] public override void Load() { this.Bind<IResolutionRoot>().ToConstant(Kernel); this.Bind<ServiceHost>().To<NinjectServiceHost>(); this.Bind<IMyServiceStorage>().To<MyServiceStorage>(); this.Bind<ILoggingService>().To<MicrosoftEnterpriseLoggingBlockLoggingService>(); } } }
- Create a Global Application Class (global.asax) if one's not already created: Right-click on your project -> Add -> New Item ... and in the window that comes up, go to Visual C# -> Web -> Global Application File
Update your Global class to extend from Ninject.Web.Common.NinjectHttpApplication and override the CreateKernel() method and return a new CustomNinjectModule (as created above). The method should look something like this:
namespace MyService.CloudStorage { using System.Diagnostics.CodeAnalysis; using System.Web; using Ninject; using Ninject.Web.Common; using Support; /// <summary> /// A global <see cref="HttpApplication"/> class for managing the lifecycle /// of the application /// </summary> public class Global : NinjectHttpApplication { /// <inheritdoc/> [SuppressMessage("Microsoft.StyleCop.CSharp.DocumentationRules", "SA1604:ElementDocumentationMustHaveSummary", Justification = "InheritDoc")] [SuppressMessage("Microsoft.StyleCop.CSharp.DocumentationRules", "SA1615:ElementReturnValueMustBeDocumented", Justification = "InheritDoc")] protected override IKernel CreateKernel() { return new StandardKernel(new NinjectSettings(), new MyServiceCloudStorageNinjectModule()); } } }
- Update your .svc file for your service(s) and add an extra XML attribute that looks like this: <%@ ServiceHost Service="HelloNinjectWcf.Service.GreetingService" Factory="Ninject.Extensions.Wcf.NinjectServiceHostFactory" %>
- Implement your business logic
- If you're designing your service properly, you'll need to ensure that communications between clients and your service are secure. Toward that end, you'll need to use SSL to encrypt your communications. There's a number of things involved in this:
- Generate certificates for your server and your client
- Ensure that your IIS server is correctly configured to use 'https' bindings on your site, along with the server-side certificate that you've generated.
- If you're using Windows Store Apps to access a WCF service, you'll need to ensure that you use a CustomBinding correctly configured with the right *BindingElement objects to create an SSL secured, HTTPS-transported binding.
- For the server, in your service's concrete implementation, you'll need to remove any .config file configuration, and add a method specified according to a convention that WCF recognizes that looks like the following:
/// <summary> /// The service certificate store name /// </summary> private const StoreName ServiceCertificateStoreName = StoreName.My; /// <summary> /// The service certificate store location /// </summary> private const StoreLocation ServiceCertificateStoreLocation = StoreLocation.LocalMachine; /// <summary> /// Configures the specified configuration. /// </summary> /// <param name="config">The configuration.</param> /// <remarks> /// A service endpoint configuration method determined by convention. /// <see href="http://msdn.microsoft.com/en-us/library/hh205277(v=vs.110).aspx"/> /// </remarks> public static void Configure(ServiceConfiguration config) { ServiceEndpoint serviceEndpoint = new ServiceEndpoint( ContractDescription.GetContract(typeof(IMyServiceStorage), typeof(MyServiceStorage)), new CustomBinding( new TransportSecurityBindingElement(), new SslStreamSecurityBindingElement { RequireClientCertificate = false }, new TextMessageEncodingBindingElement(MessageVersion.Soap12WSAddressing10, Encoding.UTF8), new HttpsTransportBindingElement() ), new EndpointAddress("https://localhost/MyService.CloudStorage/MyServiceStorage.svc") ); config.AddServiceEndpoint(serviceEndpoint); const string ServiceCertificateThumbprint = "[a 40 digit hexadecimal certificate thumbprint here]"; X509Store certificateStore = new X509Store(ServiceCertificateStoreName, ServiceCertificateStoreLocation); certificateStore.Open(OpenFlags.ReadOnly); X509Certificate2Collection x509Certificate2Collection = certificateStore.Certificates.Find( findType: X509FindType.FindByThumbprint, findValue: ServiceCertificateThumbprint, validOnly: false ); certificateStore.Close(); X509Certificate2 serviceCertificate = x509Certificate2Collection.Cast<X509Certificate2>().FirstOrDefault(); if (serviceCertificate == null) { throw new ConfigurationErrorsException(String.Format("No certificate representing the service with thumbprint {0} could be found in {1} store at {2} location", ServiceCertificateThumbprint, ServiceCertificateStoreName, ServiceCertificateStoreLocation)); } ServiceCredentials serviceCredentials = new ServiceCredentials { IdentityConfiguration = new IdentityConfiguration { // TODO: Change this to authenticate the clients CertificateValidationMode = X509CertificateValidationMode.None }, ServiceCertificate = { Certificate = serviceCertificate }/*, ClientCertificate = { // TODO: Resolve the client certificate Certificate = serviceCertificate }*/ }; config.Description.Behaviors.Add(serviceCredentials); // config.Description.Behaviors.Add(new ServiceMetadataBehavior { HttpGetEnabled = true, HttpsGetEnabled = true }); }
- For the Windows Store App client, you'll need to have a similarly configured counterpart channel factory and binding for connecting to the service:
this.channelFactory = new ChannelFactory<IMyServiceStorageChannel>( binding: new CustomBinding( new TransportSecurityBindingElement(), new SslStreamSecurityBindingElement(), new TextMessageEncodingBindingElement(MessageVersion.Soap12WSAddressing10, Encoding.UTF8), new HttpsTransportBindingElement() ), remoteAddress: new EndpointAddress(serviceUri) ); this.passwordVault = new PasswordVault(); // This IDispatchMessageInspector is a custom addition for our own brand of authentication this.channelFactory.Endpoint.EndpointBehaviors.Add(new ClientAuthenticationDispatchMessageInspector(this.passwordVault));
- For the Windows Store App, you'll also need to have the server's public key (assuming it's not trusted, e.g. a self-signed certificate) added to the app's certificate declarations in the package manifest. There's a video on how to do it here on Channel 9. For the sake of convenience, I'll reproduce the steps here:
- Obtain your server's public key in DER-encoded .cer format.
- Open the Package.appxmanifest file for your App in Visual Studio
- Go to the Declarations tab.
- Under the 'Available Declarations' box, select 'Certificates' and click 'Add' if there's no Certificates declaration already added.
- Select the 'Certificates' declaration in the 'Supported Declarations' box.
- In the 'Certificates' group on the large pane, click 'Add New'.
- In the certificate parameters box that comes up, enter 'Root' in the 'Name' field, and select your public key file. Once you do this, Visual Studio will automatically import it into your project.
- Save the manifest.
- Publish it to Azure Services
- If you've designed your application correctly, you'll be using HTTPS for communicating with your clients. Read Microsoft's guide on MSDN to uploading a certificate with your service.
- Implement your Windows 8.1 client // TODO: Elaborate on this
- Unit test your Windows 8.1 client on your own local machine.
- Visual Studio 2012 / .NET 4.5 added the ability to do asynchronous unit tests to MSTest!
- In order to unit test the Windows 8.1 Metro client against services on your localhost, you'll have to read this. It describes the new security features in Windows 8(.1) and how to explicitly enable your application to communicate through the network loopback interface.
- You can also read this. Bottom line, you have to ensure that you enable a loopback exemption for your unit tests.
- Unit / integration test it against your Azure service
- Ensure that you've created a 'Staging' area in your Azure configuration panel and you're not testing against production! Testing against production is extremely poor practice!
- Publish your App on the Windows Store if you so choose
Wednesday, November 26, 2014
Creating Azure Services with Visual Studio 2013 and Windows 8.1
Subscribe to:
Posts (Atom)