Skip to content

Commit f181eda

Browse files
committed
Allow parsing nested JSON responses V3 (#3015)
* Modify helpers so JSON responses can be deserialized correctly * Added explicit contract for responses * Enforce the expectation that status is of type DeployStatus?
1 parent 45ee637 commit f181eda

File tree

3 files changed

+61
-30
lines changed

3 files changed

+61
-30
lines changed

src/Azure.Functions.Cli/Helpers/KuduLiteDeploymentHelpers.cs

+19-30
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
using Azure.Functions.Cli.Arm.Models;
22
using Azure.Functions.Cli.Common;
3+
using Azure.Functions.Cli.Models;
34
using Colors.Net;
45
using Newtonsoft.Json;
56
using System;
@@ -51,61 +52,58 @@ public static async Task<DeployStatus> WaitForRemoteBuild(HttpClient client, Sit
5152

5253
private static async Task<string> GetLatestDeploymentId(HttpClient client, Site functionApp)
5354
{
54-
var json = await InvokeRequest<List<Dictionary<string, string>>>(client, HttpMethod.Get, "/deployments");
55+
var deployments = await InvokeRequest<List<DeploymentResponse>>(client, HttpMethod.Get, "/deployments");
5556

5657
// Automatically ordered by received time
57-
var latestDeployment = json.First();
58-
if (latestDeployment.TryGetValue("status", out string statusString))
58+
var latestDeployment = deployments.First();
59+
DeployStatus? status = latestDeployment.Status;
60+
if (status == DeployStatus.Building || status == DeployStatus.Deploying
61+
|| status == DeployStatus.Success || status == DeployStatus.Failed)
5962
{
60-
DeployStatus status = ConvertToDeployementStatus(statusString);
61-
if (status == DeployStatus.Building || status == DeployStatus.Deploying
62-
|| status == DeployStatus.Success || status == DeployStatus.Failed)
63-
{
64-
return latestDeployment["id"];
65-
}
63+
return latestDeployment.Id;
6664
}
6765
return null;
6866
}
6967

7068
private static async Task<DeployStatus> GetDeploymentStatusById(HttpClient client, Site functionApp, string id)
7169
{
72-
Dictionary<string, string> json = await InvokeRequest<Dictionary<string, string>>(client, HttpMethod.Get, $"/deployments/{id}");
73-
if (!json.TryGetValue("status", out string statusString))
70+
var deploymentInfo = await InvokeRequest<DeploymentResponse>(client, HttpMethod.Get, $"/deployments/{id}");
71+
DeployStatus? status = deploymentInfo.Status;
72+
if (status == null)
7473
{
7574
return DeployStatus.Unknown;
7675
}
77-
78-
return ConvertToDeployementStatus(statusString);
76+
return status.Value;
7977
}
8078

8179
private static async Task<DateTime> DisplayDeploymentLog(HttpClient client, Site functionApp, string id, DateTime lastUpdate, Uri innerUrl = null, StringBuilder innerLogger = null)
8280
{
8381
string logUrl = innerUrl != null ? innerUrl.ToString() : $"/deployments/{id}/log";
8482
StringBuilder sbLogger = innerLogger != null ? innerLogger : new StringBuilder();
8583

86-
var json = await InvokeRequest<List<Dictionary<string, string>>>(client, HttpMethod.Get, logUrl);
87-
var logs = json.Where(dict => DateTime.Parse(dict["log_time"]) > lastUpdate || dict["details_url"] != null);
84+
var deploymentLogs = await InvokeRequest<List<DeploymentLogResponse>>(client, HttpMethod.Get, logUrl);
85+
var newLogs = deploymentLogs.Where(deploymentLog => deploymentLog.LogTime > lastUpdate || !string.IsNullOrEmpty(deploymentLog.DetailsUrlString));
8886
DateTime currentLogDatetime = lastUpdate;
8987

90-
foreach (var log in logs)
88+
foreach (var log in newLogs)
9189
{
9290
// Filter out details_url log
93-
if (DateTime.Parse(log["log_time"]) > lastUpdate)
91+
if (log.LogTime > lastUpdate)
9492
{
95-
sbLogger.AppendLine(log["message"]);
93+
sbLogger.AppendLine(log.Message);
9694
}
9795

9896
// Recursively log details_url from scm/api/deployments/xxx/log endpoint
99-
if (log["details_url"] != null && Uri.TryCreate(log["details_url"], UriKind.Absolute, out Uri detailsUrl))
97+
if (!string.IsNullOrEmpty(log.DetailsUrlString) && Uri.TryCreate(log.DetailsUrlString, UriKind.Absolute, out Uri detailsUrl))
10098
{
10199
DateTime innerLogDatetime = await DisplayDeploymentLog(client, functionApp, id, currentLogDatetime, detailsUrl, sbLogger);
102100
currentLogDatetime = innerLogDatetime > currentLogDatetime ? innerLogDatetime : currentLogDatetime;
103101
}
104102
}
105103

106-
if (logs.LastOrDefault() != null)
104+
if (newLogs.LastOrDefault() != null)
107105
{
108-
DateTime lastLogDatetime = DateTime.Parse(logs.Last()["log_time"]);
106+
DateTime lastLogDatetime = newLogs.Last().LogTime;
109107
currentLogDatetime = lastLogDatetime > currentLogDatetime ? lastLogDatetime : currentLogDatetime;
110108
}
111109

@@ -139,14 +137,5 @@ await RetryHelper.Retry(async () =>
139137
return default(T);
140138
}
141139
}
142-
143-
private static DeployStatus ConvertToDeployementStatus(string statusString)
144-
{
145-
if (Enum.TryParse(statusString, out DeployStatus result))
146-
{
147-
return result;
148-
}
149-
return DeployStatus.Unknown;
150-
}
151140
}
152141
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
2+
// Copyright (c) .NET Foundation. All rights reserved.
3+
// Licensed under the MIT License. See License.txt in the project root for license information.
4+
5+
using System;
6+
using Newtonsoft.Json;
7+
8+
namespace Azure.Functions.Cli.Models
9+
{
10+
[JsonObject]
11+
public class DeploymentLogResponse
12+
{
13+
[JsonProperty("log_time")]
14+
public DateTime LogTime { get; set; }
15+
16+
[JsonProperty("message")]
17+
public string Message { get; set; }
18+
19+
[JsonProperty("details_url")]
20+
public string DetailsUrlString { get; set; }
21+
}
22+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
2+
// Copyright (c) .NET Foundation. All rights reserved.
3+
// Licensed under the MIT License. See License.txt in the project root for license information.
4+
5+
using System;
6+
using Newtonsoft.Json;
7+
using Azure.Functions.Cli.Common;
8+
9+
namespace Azure.Functions.Cli.Models
10+
{
11+
[JsonObject]
12+
public class DeploymentResponse
13+
{
14+
[JsonProperty("id")]
15+
public string Id { get; set; }
16+
17+
[JsonProperty("status")]
18+
public DeployStatus? Status { get; set; }
19+
}
20+
}

0 commit comments

Comments
 (0)