tools/code/common/Logger.cs (176 lines of code) (raw):
using Azure.Core.Pipeline;
using Flurl;
using LanguageExt;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text.Json.Nodes;
using System.Text.Json.Serialization;
using System.Threading;
using System.Threading.Tasks;
namespace common;
public sealed record LoggerName : ResourceName, IResourceName<LoggerName>
{
private LoggerName(string value) : base(value) { }
public static LoggerName From(string value) => new(value);
}
public sealed record LoggersUri : ResourceUri
{
public required ManagementServiceUri ServiceUri { get; init; }
private static string PathSegment { get; } = "loggers";
protected override Uri Value => ServiceUri.ToUri().AppendPathSegment(PathSegment).ToUri();
public static LoggersUri From(ManagementServiceUri serviceUri) =>
new() { ServiceUri = serviceUri };
}
public sealed record LoggerUri : ResourceUri
{
public required LoggersUri Parent { get; init; }
public required LoggerName Name { get; init; }
protected override Uri Value => Parent.ToUri().AppendPathSegment(Name.ToString()).ToUri();
public static LoggerUri From(LoggerName name, ManagementServiceUri serviceUri) =>
new()
{
Parent = LoggersUri.From(serviceUri),
Name = name
};
}
public sealed record LoggersDirectory : ResourceDirectory
{
public required ManagementServiceDirectory ServiceDirectory { get; init; }
private static string Name { get; } = "loggers";
protected override DirectoryInfo Value =>
ServiceDirectory.ToDirectoryInfo().GetChildDirectory(Name);
public static LoggersDirectory From(ManagementServiceDirectory serviceDirectory) =>
new() { ServiceDirectory = serviceDirectory };
public static Option<LoggersDirectory> TryParse(DirectoryInfo? directory, ManagementServiceDirectory serviceDirectory) =>
directory is not null &&
directory.Name == Name &&
directory.Parent?.FullName == serviceDirectory.ToDirectoryInfo().FullName
? new LoggersDirectory { ServiceDirectory = serviceDirectory }
: Option<LoggersDirectory>.None;
}
public sealed record LoggerDirectory : ResourceDirectory
{
public required LoggersDirectory Parent { get; init; }
public required LoggerName Name { get; init; }
protected override DirectoryInfo Value =>
Parent.ToDirectoryInfo().GetChildDirectory(Name.ToString());
public static LoggerDirectory From(LoggerName name, ManagementServiceDirectory serviceDirectory) =>
new()
{
Parent = LoggersDirectory.From(serviceDirectory),
Name = name
};
public static Option<LoggerDirectory> TryParse(DirectoryInfo? directory, ManagementServiceDirectory serviceDirectory) =>
from parent in LoggersDirectory.TryParse(directory?.Parent, serviceDirectory)
select new LoggerDirectory
{
Parent = parent,
Name = LoggerName.From(directory!.Name)
};
}
public sealed record LoggerInformationFile : ResourceFile
{
public required LoggerDirectory Parent { get; init; }
private static string Name { get; } = "loggerInformation.json";
protected override FileInfo Value =>
Parent.ToDirectoryInfo().GetChildFile(Name);
public static LoggerInformationFile From(LoggerName name, ManagementServiceDirectory serviceDirectory) =>
new()
{
Parent = new LoggerDirectory
{
Parent = LoggersDirectory.From(serviceDirectory),
Name = name
}
};
public static Option<LoggerInformationFile> TryParse(FileInfo? file, ManagementServiceDirectory serviceDirectory) =>
file is not null && file.Name == Name
? from parent in LoggerDirectory.TryParse(file.Directory, serviceDirectory)
select new LoggerInformationFile { Parent = parent }
: Option<LoggerInformationFile>.None;
}
public sealed record LoggerDto
{
[JsonPropertyName("properties")]
[JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingDefault)]
public required LoggerContract Properties { get; init; }
public record LoggerContract
{
[JsonPropertyName("loggerType")]
[JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingDefault)]
public string? LoggerType { get; init; }
[JsonPropertyName("credentials")]
[JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingDefault)]
public JsonObject? Credentials { get; init; }
[JsonPropertyName("description")]
[JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingDefault)]
public string? Description { get; init; }
[JsonPropertyName("isBuffered")]
[JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingDefault)]
public bool? IsBuffered { get; init; }
[JsonPropertyName("resourceId")]
[JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingDefault)]
public string? ResourceId { get; init; }
}
}
public static class LoggerModule
{
public static async ValueTask DeleteAll(this LoggersUri uri, HttpPipeline pipeline, CancellationToken cancellationToken) =>
await uri.ListNames(pipeline, cancellationToken)
.IterParallel(async name => await LoggerUri.From(name, uri.ServiceUri)
.Delete(pipeline, cancellationToken),
cancellationToken);
public static IAsyncEnumerable<LoggerName> ListNames(this LoggersUri uri, HttpPipeline pipeline, CancellationToken cancellationToken) =>
pipeline.ListJsonObjects(uri.ToUri(), cancellationToken)
.Select(jsonObject => jsonObject.GetStringProperty("name"))
.Select(LoggerName.From);
public static IAsyncEnumerable<(LoggerName Name, LoggerDto Dto)> List(this LoggersUri loggersUri, HttpPipeline pipeline, CancellationToken cancellationToken) =>
loggersUri.ListNames(pipeline, cancellationToken)
.SelectAwait(async name =>
{
var uri = new LoggerUri { Parent = loggersUri, Name = name };
var dto = await uri.GetDto(pipeline, cancellationToken);
return (name, dto);
});
public static async ValueTask<Option<LoggerDto>> TryGetDto(this LoggerUri uri, HttpPipeline pipeline, CancellationToken cancellationToken)
{
var contentOption = await pipeline.GetContentOption(uri.ToUri(), cancellationToken);
return contentOption.Map(content => content.ToObjectFromJson<LoggerDto>());
}
public static async ValueTask<LoggerDto> GetDto(this LoggerUri uri, HttpPipeline pipeline, CancellationToken cancellationToken)
{
var content = await pipeline.GetContent(uri.ToUri(), cancellationToken);
return content.ToObjectFromJson<LoggerDto>();
}
public static async ValueTask Delete(this LoggerUri uri, HttpPipeline pipeline, CancellationToken cancellationToken) =>
await pipeline.DeleteResource(uri.ToUri(), waitForCompletion: true, cancellationToken);
public static async ValueTask PutDto(this LoggerUri uri, LoggerDto dto, HttpPipeline pipeline, CancellationToken cancellationToken)
{
var content = BinaryData.FromObjectAsJson(dto);
await pipeline.PutContent(uri.ToUri(), content, cancellationToken);
}
public static IEnumerable<LoggerDirectory> ListDirectories(ManagementServiceDirectory serviceDirectory)
{
var loggersDirectory = LoggersDirectory.From(serviceDirectory);
return loggersDirectory.ToDirectoryInfo()
.ListDirectories("*")
.Select(directoryInfo => LoggerName.From(directoryInfo.Name))
.Select(name => new LoggerDirectory { Parent = loggersDirectory, Name = name });
}
public static IEnumerable<LoggerInformationFile> ListInformationFiles(ManagementServiceDirectory serviceDirectory) =>
ListDirectories(serviceDirectory)
.Select(directory => new LoggerInformationFile { Parent = directory })
.Where(informationFile => informationFile.ToFileInfo().Exists());
public static async ValueTask WriteDto(this LoggerInformationFile file, LoggerDto dto, CancellationToken cancellationToken)
{
var content = BinaryData.FromObjectAsJson(dto, JsonObjectExtensions.SerializerOptions);
await file.ToFileInfo().OverwriteWithBinaryData(content, cancellationToken);
}
public static async ValueTask<LoggerDto> ReadDto(this LoggerInformationFile file, CancellationToken cancellationToken)
{
var content = await file.ToFileInfo().ReadAsBinaryData(cancellationToken);
return content.ToObjectFromJson<LoggerDto>();
}
}