Proof of concept met Azure Batch

Case omschrijving

Net als in mijn vorige blog post heb ik een Proof of Concept gemaakt om grote hoeveelheden XML bestanden te transformeren, het gaat dan 3000 bestanden per keer met een totale grote van 18 gigabyte. Voor dit artikel kan ik wegens privacy redenen niet de echte data gebruiken en heb ik een dataset gebruikt van https://www.kaggle.com/datasets. Het gaat hier om een dataset van landen en de wijnen.

Het doel is om de wijnen uit de XML dump te halen en deze om te zetten naar een JSON formaat en deze bestanden te uploaden in een blob container. We krijgen dus per wijn een JSON bestand in een blob container. Dit alles moet gebeuren op basis Azure Batch met een Azure Function als orchestrator.

Wat is de opzet van de POC

We gaan de XML bestanden transformeren doormiddel van een Azure Batch component en we starten en beheren de Azure Batch doormiddel van een Azure Function.
De Azure Function zal de XML bestanden toevoegen aan een Job in Azure Batch doormiddel van een event grid trigger en via een httptrigger is de voortgang te zien van het batch proces.

Wat is Azure Batch

Azure Batch is plat gezegd eigenlijk een beheer tool voor virtuele machines. Elk van deze machines kan een taak oppakken en dit als input gebruiken voor een commandline applicatie en het resultaat uploaden in bijvoorbeeld een blob container.

Voor de complete omschrijving wat je met Azure Batch kunt doen verwijs ik je graag door naar de Microsoft site https://azure.microsoft.com/nl-nl/services/batch/

Hoe werkt het Proof of Concept

Azure Batch bestaat uit een aantal componenten.

  • Een Task is een opdracht welke uitgevoerd dient te worden op een node.
  • Een Job is een verzameling van tasks. Aan een Job hangt ook een Pool.
  • Een Pool is een verzameling van nodes.
  • Een Node is een virtuele machine welke een van de tasks gaat uitvoeren.

Ik heb 40 bestanden met landen en wijnen. 1 bestand is ongeveer 75 mb groot. Voor testdoeleinden zijn dit dezelfde bestanden met een andere naam. Dit is meer om een gelijkwaardige load op de functie te krijgen als bij de echte POC.

Ik heb een Azure Function welke een event grid trigger heeft welke afgaat op het moment dat er een bestand wordt geupload in de blobcontainer.

1
2
3
4
5
6
7
8
9
10
11
12
[FunctionName("BatchOrchestrator")]
public static async Task Run(
[EventGridTrigger] EventGridEvent eventGridEvent,
ExecutionContext context,
ILogger log)
{
var data = JsonConvert.DeserializeObject<StorageBlobCreatedEventData>(eventGridEvent.Data.ToString());
string name = data.Url.Split('/').Last();

log.LogInformation($"C# Blob trigger function Processed blob\n Name:{name}");
await RunBatch(log, context, name);
}

De bestandsnaam wordt uit de URL gehaald en doorgestuurd naar de methode RunBatch.
De methode RunBatch initialiseert een BatchClient met de credentials welke opgegeven zij in de config.

1
2
3
4
5
6
7
8
9
private static async Task RunBatch(ILogger log, ExecutionContext context, string name)
{
var config = ReadSettings(context);

BatchSharedKeyCredentials credentials = new BatchSharedKeyCredentials(config["BatchUrl"], config["BatchAccount"], config["BatchKey"]);
using (BatchClient batchClient = BatchClient.Open(credentials))
{
}
}

Hierna gaan we kijken of er al een job bestaat in het Azure Batch Account. Als er al een job bestaat en hij is nog actief of wordt aangemaakt voegen we hier een task aan toe. Als er geen job actief is of wordt opgestart dan maken we een Job aan. Dit gebeurt in de CreateJob methode.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
private static async Task RunBatch(ILogger log, ExecutionContext context, string name)
{

var config = ReadSettings(context);

BatchSharedKeyCredentials credentials = new BatchSharedKeyCredentials(config["BatchUrl"], config["BatchAccount"], config["BatchKey"]);
using (BatchClient batchClient = BatchClient.Open(credentials))
{
string jobId = string.Empty;

try
{
batchClient.CustomBehaviors.Add(RetryPolicyProvider.ExponentialRetryProvider(TimeSpan.FromSeconds(5), 3));

if (batchClient.JobOperations.ListJobs().Any())
{
var jobs = batchClient.JobOperations.ListJobs();
CloudJob activeJob = jobs.FirstOrDefault(job => job.State == JobState.Active || job.State == JobState.Enabling);

if (activeJob != null)
{
log.LogDebug("Job still active");
CreateTaskIfNotExists(batchClient, activeJob.Id, name);
}
else
{
jobId = await CreateJob(name, batchClient);
}
}
else
{
jobId = await CreateJob(name, batchClient);
}
}
catch (Exception e)
{
log.LogError(e, e.Message);
if (!string.IsNullOrEmpty(jobId))
{
log.LogDebug($"Deleting job: {jobId}");
await batchClient.JobOperations.DeleteJobAsync(jobId);
}
}
}
}

Een Job heeft een pool met nodes nodig welke het werk uitvoeren. We gaan deze dus eerst aanmaken.

  • We specificeren het besturingssysteem in dit geval staat de waarde 5 voor Windows Server 2016 voor de overige waardes check de Azure Guest OS Releases https://docs.microsoft.com/en-us/azure/cloud-services/cloud-services-guestos-update-matrix#releases
  • We specificeren een virtuele machine size in dit geval een standard_d1_v2 voor de overige ondersteunde machines check https://docs.microsoft.com/en-us/azure/batch/batch-pool-vm-sizes#supported-vm-families-and-sizes
  • We specificeren hoeveel tasks er per node gedraaid mogen worden in dit geval 4. Er zullen dus 4 tasks per keer op de node worden gestart.
  • We specificeren welke applicatie er op de node gedraaid dient te worden. Dit moet een applicatie of script zijn welke via de commandline te draaien is. Deze applicatie kan als een zip bestand worden gupload in de portal.
  • We zetten de lifetime op PoolLifetimeOption.Job dit wil zeggen zodra alle tasks in de Job klaar zijn zal de pool verwijdert worden en zul je dus ook niet meer betalen voor de virtuele machines.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
private static PoolInformation CreatePool()
{
return new PoolInformation()
{
AutoPoolSpecification = new AutoPoolSpecification()
{
AutoPoolIdPrefix = "Wine",
PoolSpecification = new PoolSpecification()
{
CloudServiceConfiguration = new CloudServiceConfiguration("5"),
VirtualMachineSize = "standard_d1_v2",
MaxTasksPerComputeNode = 4,
ApplicationPackageReferences = new List<ApplicationPackageReference>()
{
new ApplicationPackageReference()
{
ApplicationId = "converter"
}
}
},
KeepAlive = false,
PoolLifetimeOption = PoolLifetimeOption.Job
}
};
}

Nu we een Pool hebben kunnen we deze koppelen aan de Job. De CreateJob methode maakt een unieke naam aan voor de job doormiddel van een timestamp te prefixen met “WineConverter”.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
private static async Task<string> CreateJob(string name, BatchClient batchClient)
{
string jobId;

CloudJob activeJob;
jobId = CreateJobId("WineConverter");

//create pool
PoolInformation pool = CreatePool();

// create a job
activeJob = await CreateJobAsync(batchClient, jobId, pool);

//create tasks from the blobs
CreateTaskIfNotExists(batchClient, jobId, name);

await activeJob.RefreshAsync();
activeJob.OnAllTasksComplete = OnAllTasksComplete.TerminateJob;
await activeJob.CommitChangesAsync();

return jobId;
}

We controlleren eerst of de task al is aangemaakt en is toegevoegd aan de job, zoniet dan voegen we hem toe. De naam van de task mag alleen letters en cijfers bevatten en een koppelteken en underscore.

Ook stellen we in welke package de task moet starten met welke argumenten. Hier starten we converter.exe met als argument een naam van de blob (in dit geval een XML bestand met wijn data).

De converter.exe bevat alle logica om de xml te verwerken en het resultaat te uploaden in een Azure blob container.
In de Azure portal kan je een package uploaden welke op de nodes geinstalleerd moeten worden. Meer informatie over hoe packages werken met Azure batch is te vinden op: https://docs.microsoft.com/en-us/azure/batch/batch-application-packages

1
2
3
4
5
6
7
8
9
10
11
12
13
private static void CreateTaskIfNotExists(BatchClient batchClient, string jobId, string blobName)
{
IPagedEnumerable<CloudTask> tasks = batchClient.JobOperations.ListTasks(jobId);
string taskId = $"task_{SanitizeString(blobName)}";

if (tasks.Any(x => x.Id == taskId))
{
Console.WriteLine($"Task with id: {taskId} all ready exists");
return;
}

batchClient.JobOperations.AddTask(jobId, new CloudTask(taskId, $"cmd /c %AZ_BATCH_APP_PACKAGE_CONVERTER%\\converter.exe -name {blobName}"));
}

De volledige code

Zie hieronder de volledige code. Er is een extra function aan toegevoegd met een httptrigger zodra je deze aanroept worden alle jobs met de bijhorende tasks weergegeven en de status van de tasks. Het endpoint is nu niet beveiligd maar dit is omdat het een demo is.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Net.Http;
using System.Threading.Tasks;
using Microsoft.Azure.Batch;
using Microsoft.Azure.Batch.Auth;
using Microsoft.Azure.Batch.Common;
using Microsoft.Azure.WebJobs;
using Microsoft.Azure.WebJobs.Extensions.Http;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.Logging;
using Newtonsoft.Json;
using JobState = Microsoft.Azure.Batch.Common.JobState;

namespace WineConverter
{
public static class BatchOrchestrator
{
[FunctionName("BatchOrchestrator")]
public static async Task Run(
[EventGridTrigger] EventGridEvent eventGridEvent,
ExecutionContext context,
ILogger log)
{
var data = JsonConvert.DeserializeObject<StorageBlobCreatedEventData>(eventGridEvent.Data.ToString());
string name = data.Url.Split('/').Last();

log.LogInformation($"C# Blob trigger function Processed blob\n Name:{name}");
await RunBatch(log, context, name);
}

private static async Task RunBatch(ILogger log, ExecutionContext context, string name)
{

var config = ReadSettings(context);

BatchSharedKeyCredentials credentials = new BatchSharedKeyCredentials(config["BatchUrl"], config["BatchAccount"], config["BatchKey"]);
using (BatchClient batchClient = BatchClient.Open(credentials))
{
string jobId = string.Empty;

try
{
batchClient.CustomBehaviors.Add(RetryPolicyProvider.ExponentialRetryProvider(TimeSpan.FromSeconds(5), 3));

if (batchClient.JobOperations.ListJobs().Any())
{
var jobs = batchClient.JobOperations.ListJobs();
CloudJob activeJob = jobs.FirstOrDefault(job => job.State == JobState.Active || job.State == JobState.Enabling);

if (activeJob != null)
{
log.LogDebug("Job still active");
CreateTaskIfNotExists(batchClient, activeJob.Id, name);
}
else
{
jobId = await CreateJob(name, batchClient);
}
}
else
{
jobId = await CreateJob(name, batchClient);
}
}
catch (Exception e)
{
log.LogError(e, e.Message);
if (!string.IsNullOrEmpty(jobId))
{
log.LogDebug($"Deleting job: {jobId}");
await batchClient.JobOperations.DeleteJobAsync(jobId);
}
}
}
}

[FunctionName("BatchStatus")]
public static string BatchStatus([HttpTrigger(AuthorizationLevel.Anonymous, "get")]HttpRequestMessage req, ExecutionContext context, ILogger log)
{
List<Job> jobList = new List<Job>();

var config = ReadSettings(context);
BatchSharedKeyCredentials credentials = new BatchSharedKeyCredentials(config["BatchUrl"], config["BatchAccount"], config["BatchKey"]);
using (BatchClient batchClient = BatchClient.Open(credentials))
{
IPagedEnumerable<CloudJob> jobs = batchClient.JobOperations.ListJobs();
if (jobs.Any())
{

foreach (CloudJob cloudJob in jobs.ToList())
{
jobList.Add(new Job()
{
Id = cloudJob.Id,
Name = cloudJob.DisplayName,
Status = cloudJob.State.ToString(),
Tasks = cloudJob.ListTasks().Select(task => new JobTask()
{
Id = task.Id,
Name = task.DisplayName,
Status = task.State.ToString()
}).ToList()
});
}
}
}

return JsonConvert.SerializeObject(jobList, Formatting.Indented);
}

private static async Task<string> CreateJob(string name, BatchClient batchClient)
{
string jobId;

CloudJob activeJob;
jobId = CreateJobId("WineConverter");

//create pool
PoolInformation pool = CreatePool();

// create a job
activeJob = await CreateJobAsync(batchClient, jobId, pool);

//create tasks from the blobs
CreateTaskIfNotExists(batchClient, jobId, name);

await activeJob.RefreshAsync();
activeJob.OnAllTasksComplete = OnAllTasksComplete.TerminateJob;
await activeJob.CommitChangesAsync();

return jobId;
}

private static PoolInformation CreatePool()
{
return new PoolInformation()
{
AutoPoolSpecification = new AutoPoolSpecification()
{
AutoPoolIdPrefix = "Wine",
PoolSpecification = new PoolSpecification()
{
CloudServiceConfiguration = new CloudServiceConfiguration("5"),
VirtualMachineSize = "standard_d1_v2",
MaxTasksPerComputeNode = 4,
ApplicationPackageReferences = new List<ApplicationPackageReference>()
{
new ApplicationPackageReference()
{
ApplicationId = "converter"
}
}
},
KeepAlive = false,
PoolLifetimeOption = PoolLifetimeOption.Job
}
};
}

private static async Task<CloudJob> CreateJobAsync(BatchClient batchClient, string jobId, PoolInformation pool)
{
CloudJob unboundJob = batchClient.JobOperations.CreateJob();
unboundJob.Id = jobId;

unboundJob.PoolInformation = pool;
await unboundJob.CommitAsync();

return unboundJob;
}

private static void CreateTaskIfNotExists(BatchClient batchClient, string jobId, string blobName)
{
IPagedEnumerable<CloudTask> tasks = batchClient.JobOperations.ListTasks(jobId);
string taskId = $"task_{SanitizeString(blobName)}";

if (tasks.Any(x => x.Id == taskId))
{
Console.WriteLine($"Task with id: {taskId} all ready exists");
return;
}

batchClient.JobOperations.AddTask(jobId, new CloudTask(taskId, $"cmd /c %AZ_BATCH_APP_PACKAGE_CONVERTER%\\converter.exe -name {blobName}"));
}

private static string CreateJobId(string prefix)
{
return $"{prefix}-{DateTime.Now:yyyyMMdd-HHmmss}";
}

private static string SanitizeString(string text)
{
string pattern = @"[^A-Za-z0-9-_]";
return System.Text.RegularExpressions.Regex.Replace(text, pattern, string.Empty);
}

private static IConfigurationRoot ReadSettings(ExecutionContext context)
{
return new ConfigurationBuilder()
.SetBasePath(context.FunctionAppDirectory)
.AddJsonFile("local.settings.json", optional: true, reloadOnChange: true)
.AddEnvironmentVariables()
.Build();


}
}

public class JobTask
{
public string Id { get; set; }
public string Name { get; set; }
public string Status { get; set; }
}

public class Job
{
public string Id { get; set; }
public string Name { get; set; }
public string Status { get; set; }
public List<JobTask> Tasks { get; set; }
}
}

Conclusie

Azure batch is echt een serieuze keuze als je grote hoeveelheden data moet verwerken en je wilt in controle zijn wat er allemaal gebeurt.
Je kan zowel horizontaal als verticaal schalen en het aantal nodes wat het werk kan doen is standaard 20 maar je kunt een request doen voor meer nodes. Voor de recource limieten check de documentatie https://docs.microsoft.com/en-us/azure/batch/batch-quota-limit.

Ik ken zelf weinig projecten waar ze Azure batch gebruiken maar ik ben echt onder de indruk hoe simpel en krachtig Azure batch is plus je betaald alleen voor de tijd dat de nodes ook echt iets doen dus geen vaste maandelijkse kosten.

Deze POC heeft het qua performance zijn doel wel behaald wat we voor ogen hadden alleen het bevat toch net te veel stappen om het volledig via CI/CD gemakkelijk te deployen.

In een volgende blog zal ik de uiteindelijke oplossing uitwerken met verschillende functie apps op een consumption plan welke aan alle eisen voldoet.