Now that we have some Artificial Intelligence (AI) that will analyze the images, we need some code that could be executed every time a new picture is added.
To do this we will use an Azure Function.
The Functions are part of the serverless offering by Microsoft. They provide an efficient way to execute code at scale without being expensive. Learn more about Azure Function here.
Prior to the workshop, you should have installed all the prerequisites. If it's not done now, it's time.
Let's open Visual Studio Code (aka VSCode). From the left menu bar select the Azure icon to open the Azure Function Extension.
To create a Function App, click on the first icon (the folder with a lightning bolt).
-
Create a new local folder and name it HackDemo-Func
-
When prompt for the language, select C#.
-
When Prompt for the project's first function, select Skip for now
-
Finally, select the Open in current window option.
It may take a few seconds while VSCode creates your project. Once it's done, you will have a few files and folders. We are now ready to create our Azure Function.
There are many different types of Azure Function, or many different ways to interact with it. In this workshop what we need is a Function that will get triggered every time a new image is uploaded into our Azure Blob Storage. We need a Blob trigger.
- From the Azure Function extension, click on the second icon (the lightning bolt with a little + sign).
- Select BlobTrigger as the template.
- Enter DogDetector as the function name.
- Enter DogDemo.Function as namespace.
- Enter AzureWebJobsStorage as settings from "local.settings.json"
- The name of our container is "images" so the path that our function will monitor is also images. Enter images.
You should have a warning popup at this time. Since we won't debug it locally, we can skip it for now.
Before we add some code inside the Azure Function let's add some requirement features. Let's first start by adding some packages we will need. Look into the file HackDemo-Func.csproj
. Currently you should have only one <PackageReference>
, referencing the Microsoft.NET.Sdk.Functions
and Microsoft.Azure.WebJobs.Extensions.Storage
.
Let's add a few more.
Open the terminal, from the top menu Terminal or with the shortcut ( Ctrl + ` ). Be sure you are in the DogDemo-FuncApp
folder, and enter the following command.
dotnet add package Newtonsoft.Json
If you look again at the file HackDemo-Func.csproj
, you will see that the package is now referenced.
Repeat the previous command for the package
dotnet add package Microsoft.Azure.CognitiveServices.Vision.ComputerVision
and
dotnet add package Microsoft.Azure.WebJobs.Script.ExtensionsMetadataGenerator
Let's replace all the code inside DogDetector.cs
with the following. (The file is also available here)
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using Microsoft.Azure.CognitiveServices.Vision.ComputerVision;
using Microsoft.Azure.CognitiveServices.Vision.ComputerVision.Models;
using Microsoft.Azure.WebJobs;
using Microsoft.Extensions.Configuration;
using Microsoft.WindowsAzure.Storage.Blob;
using Microsoft.Extensions.Logging;
namespace DogDemo.Function
{
public static class DogDetector
{
private static readonly List<VisualFeatureTypes> Features = new List<VisualFeatureTypes>
{
VisualFeatureTypes.Categories, VisualFeatureTypes.Description,
VisualFeatureTypes.Faces, VisualFeatureTypes.ImageType,
VisualFeatureTypes.Tags
};
// We must provide SAS token in order to have the API read the image located at the provided URL since our container is private
private static SharedAccessBlobPolicy sasConstraints = new SharedAccessBlobPolicy
{
SharedAccessExpiryTime = DateTimeOffset.UtcNow.AddMinutes(10),
Permissions = SharedAccessBlobPermissions.Read | SharedAccessBlobPermissions.List
};
[FunctionName("DogDetector")]
public static async Task Run([BlobTrigger("images/{name}", Connection = "AzureWebJobsStorage")]CloudBlockBlob myBlob, string name, ILogger log)
{
var config = new ConfigurationBuilder()
.AddEnvironmentVariables()
.Build();
log.LogInformation($"Looking image {myBlob.Uri.ToString()}");
var visionAPI = new ComputerVisionClient(new ApiKeyServiceClientCredentials(config["ComputerVision:ApiKey"])) { Endpoint = config["ComputerVision:Endpoint"] };
var path = $"{myBlob.Uri.ToString()}{myBlob.GetSharedAccessSignature(sasConstraints)}";
var results = await visionAPI.AnalyzeImageAsync(path, Features);
if(IsDog(results))
{
log.LogInformation($"It's a Dog");
return;
}
log.LogInformation($"Not a Dog");
await myBlob.DeleteIfExistsAsync();
}
private static bool IsDog(ImageAnalysis image)
{
return image.Categories.Any(x => x.Name == "animal_dog") || image.Tags.Any(x => x.Name == "dog");
}
}
}
You can test to see if your solution compiles by executing the following command in the terminal:
dotnet build
You may have some yellow warning messages about the Newtonsoft.Json version, search for a green "Build succeeded".
All the code is done. It's time to deploy our Azure Function to Azure. Let's return to the Azure Function Extension (from the left menu). This time click on... yes, the third icon (the one with the Up arrow).
- You may need to login to Azure, to authorize VSCode to access to Azure.
- Select your subscription
- Click the Create New Function App in Azure
- You will need to provide a globally unique name. If you read the Learn more section of part 1 you know that a suffix was added to all your resources. You can use it if you want (ex: DogDemo-FuncApp6l3th) or just add your name in the mix (ex: DogDemo-FuncFrank).
- Now a few other resources will be created. It can take 1-2 minutes.
Once the resources are created VSCode will deploy your code. Another minute. It's all done when you see the notifications.
We now need to provide the information to our Function so it can monitor our Azure Storage and access our Vision API.
- Open the file
local.settings.json
- Add two new configurations. Use the Endpoint and Key saved previously.
,"ComputerVision:ApiKey":"60b5aaaaae5445e93ca5bbbbb1268c2e"
,"ComputerVision:Endpoint":"https://eastus.api.cognitive.microsoft.com/"
The only configuration that we need is the connection string to our storage.
-
Go back in the portal.
-
Go into your Resource group.
-
Select your Storage account.
- From the left option select Access keys.
- Copy one of the Connection strings.
-
Back in VSCode, in the file
local.settings.json
, replace the VALUE of setting AzureWebJobsStorage with the Connection string we just took from the portal. -
Save your changes.
The last step is to upload the configuration into Azure. Thanks to the Function extension, it will be easy.
- From the Azure Function extension, expand your subscription (with the key icon).
- Expand the FunctionApp (with the lightning bolt icon)
- Right-click on Application Settings, and select Upload Local Settings
- If prompted, overwrite all.
You have now completed this part of the workshop. **You can continue with Part 4: How to use the Automatic Not a Dog Application.