From 846d17f954c5cf190252881e58590b2b45eb3d84 Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Mon, 25 Nov 2024 09:39:15 -0800 Subject: [PATCH 01/24] cleanup --- .../src/CloudMachineExtensions.cs | 10 +-- .../Azure.CloudMachine/src/ClientCache.cs | 4 +- .../src/CloudMachineWorkspace.cs | 36 ++++------- .../src/CoreServices/MessagingServices.cs | 4 +- .../src/CoreServices/StorageFile.cs | 4 +- .../src/CoreServices/StorageServices.cs | 14 ++-- .../src/extensions/ChatTools.cs | 28 ++++---- .../src/extensions/EmbeddingsVectorbase.cs | 10 +-- .../src/extensions/MemoryVectorbaseStore.cs | 16 ++--- .../src/extensions/VectorbaseStore.cs | 16 ++--- .../AzureSdkExtensions/AppServiceFeature.cs | 6 +- .../src/AzureSdkExtensions/KeyVaultFeature.cs | 7 +- .../src/AzureSdkExtensions/OpenAIFeature.cs | 6 +- .../src/AzureSdkExtensions/OpenAIModel.cs | 6 +- .../src/CDKLevel3/AzdHelpers.cs | 2 +- .../src/CDKLevel3/CloudMachineCommands.cs | 11 ++-- .../CDKLevel3/CloudMachineInfrastructure.cs | 64 +++++++++---------- .../CDKLevel3/ServiceBusNamespaceFeature.cs | 2 - .../src/CDKLevel3/StorageFeature.cs | 6 +- .../SystemTopicEventSubscriptionFeature.cs | 6 +- .../src/Core/LoggingPolicy.cs | 10 +-- .../src/Core/RestCLient/RestClient.cs | 9 +-- .../src/Tsp/TypeSpecWriter.cs | 16 ++--- 23 files changed, 134 insertions(+), 159 deletions(-) diff --git a/sdk/cloudmachine/Azure.CloudMachine.Web/src/CloudMachineExtensions.cs b/sdk/cloudmachine/Azure.CloudMachine.Web/src/CloudMachineExtensions.cs index a2c501a68cd2e..bec28911b0ff1 100644 --- a/sdk/cloudmachine/Azure.CloudMachine.Web/src/CloudMachineExtensions.cs +++ b/sdk/cloudmachine/Azure.CloudMachine.Web/src/CloudMachineExtensions.cs @@ -75,7 +75,7 @@ private static RequestDelegate CreateRequestDelegate(T service, MethodInfo im ParameterInfo[] parameters = interfaceMethod!.GetParameters(); object[] implementationArguments = new object[parameters.Length]; - foreach (var parameter in parameters) + foreach (ParameterInfo parameter in parameters) { implementationArguments[0] = await CreateArgumentAsync(parameter, request).ConfigureAwait(false); } @@ -128,13 +128,13 @@ private static async ValueTask CreateArgumentAsync(ParameterInfo paramet if (parameterType == typeof(byte[])) { - var bd = await BinaryData.FromStreamAsync(request.Body).ConfigureAwait(false); + BinaryData bd = await BinaryData.FromStreamAsync(request.Body).ConfigureAwait(false); return bd.ToArray(); } if (parameterType == typeof(BinaryData)) { string? contentType = request.ContentType; - var bd = await BinaryData.FromStreamAsync(request.Body, contentType).ConfigureAwait(false); + BinaryData bd = await BinaryData.FromStreamAsync(request.Body, contentType).ConfigureAwait(false); return bd; } if (parameterType == typeof(string)) @@ -164,10 +164,10 @@ private static async ValueTask CreateArgumentAsync(ParameterInfo paramet // TODO: this is a hack. We should use MRW private static object DeserializeModel(Type modelType, Stream stream) { - var fromJson = modelType.GetMethod("FromJson", BindingFlags.Static); + MethodInfo? fromJson = modelType.GetMethod("FromJson", BindingFlags.Static); if (fromJson == default) throw new InvalidOperationException($"{modelType} does not provide FromJson static method"); - object? deserialized = fromJson.Invoke(null, new object[] { stream }); + object? deserialized = fromJson.Invoke(null, [stream]); if (deserialized == default) throw new InvalidOperationException($"Failed to deserialize {modelType}"); return deserialized; diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/ClientCache.cs b/sdk/cloudmachine/Azure.CloudMachine/src/ClientCache.cs index 921cfcc4ea184..326ef047fbee4 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/ClientCache.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/ClientCache.cs @@ -12,7 +12,7 @@ namespace Azure.Core; /// public class ClientCache { - private readonly Dictionary<(Type, string), object> _clients = new Dictionary<(Type, string), object>(); + private readonly Dictionary<(Type, string), object> _clients = []; /// /// Gets a client from the cache. @@ -23,7 +23,7 @@ public class ClientCache /// public T Get(Func value, string id = default) where T: class { - var client = (typeof(T), id); + (Type, string) client = (typeof(T), id); lock (_clients) { if (_clients.TryGetValue(client, out object cached)) diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs index a0f8ecf1acb61..d24fb88fda98e 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs @@ -70,29 +70,19 @@ public override ClientConnectionOptions GetConnectionOptions(Type clientType, st if (instanceId != null && instanceId.StartsWith("$")) clientId = $"{clientType.FullName}{instanceId}"; - switch (clientId) + return clientId switch { - case "Azure.Security.KeyVault.Secrets.SecretClient": - return new ClientConnectionOptions(new($"https://{Id}.vault.azure.net/"), Credential); - case "Azure.Messaging.ServiceBus.ServiceBusClient": - return new ClientConnectionOptions(new($"https://{Id}.servicebus.windows.net"), Credential); - case "Azure.Messaging.ServiceBus.ServiceBusSender": - return new ClientConnectionOptions(instanceId ?? "cm_servicebus_default_topic"); - case "Azure.Messaging.ServiceBus.ServiceBusProcessor": - return new ClientConnectionOptions("cm_servicebus_default_topic/cm_servicebus_subscription_default"); - case "Azure.Messaging.ServiceBus.ServiceBusProcessor$private": - return new ClientConnectionOptions("cm_servicebus_topic_private/cm_servicebus_subscription_private"); - case "Azure.Storage.Blobs.BlobContainerClient": - return new ClientConnectionOptions(new($"https://{Id}.blob.core.windows.net/{instanceId ?? "default"}"), Credential); - case "Azure.AI.OpenAI.AzureOpenAIClient": - return new ClientConnectionOptions(new($"https://{Id}.openai.azure.com"), Credential); - case "OpenAI.Chat.ChatClient": - return new ClientConnectionOptions($"{Id}_chat"); - case "OpenAI.Embeddings.EmbeddingClient": - return new ClientConnectionOptions($"{Id}_embedding"); - default: - throw new Exception($"unknown client {clientId}"); - } + "Azure.Security.KeyVault.Secrets.SecretClient" => new ClientConnectionOptions(new($"https://{Id}.vault.azure.net/"), Credential), + "Azure.Messaging.ServiceBus.ServiceBusClient" => new ClientConnectionOptions(new($"https://{Id}.servicebus.windows.net"), Credential), + "Azure.Messaging.ServiceBus.ServiceBusSender" => new ClientConnectionOptions(instanceId ?? "cm_servicebus_default_topic"), + "Azure.Messaging.ServiceBus.ServiceBusProcessor" => new ClientConnectionOptions("cm_servicebus_default_topic/cm_servicebus_subscription_default"), + "Azure.Messaging.ServiceBus.ServiceBusProcessor$private" => new ClientConnectionOptions("cm_servicebus_topic_private/cm_servicebus_subscription_private"), + "Azure.Storage.Blobs.BlobContainerClient" => new ClientConnectionOptions(new($"https://{Id}.blob.core.windows.net/{instanceId ?? "default"}"), Credential), + "Azure.AI.OpenAI.AzureOpenAIClient" => new ClientConnectionOptions(new($"https://{Id}.openai.azure.com"), Credential), + "OpenAI.Chat.ChatClient" => new ClientConnectionOptions($"{Id}_chat"), + "OpenAI.Embeddings.EmbeddingClient" => new ClientConnectionOptions($"{Id}_embedding"), + _ => throw new Exception($"unknown client {clientId}"), + }; } /// @@ -118,7 +108,7 @@ internal static string ReadOrCreateCmid() cmid = GenerateCloudMachineId(); using FileStream file = File.OpenWrite(appsettings); - Utf8JsonWriter writer = new Utf8JsonWriter(file); + Utf8JsonWriter writer = new(file); writer.WriteStartObject(); writer.WritePropertyName("CloudMachine"u8); writer.WriteStartObject(); diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/MessagingServices.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/MessagingServices.cs index 7b2b146da7421..64bebd2efe61f 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/MessagingServices.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/MessagingServices.cs @@ -46,8 +46,8 @@ public async Task SendJsonAsync(object serializable) /// public void WhenMessageReceived(Action received) { - var processor = _cm.Messaging.GetServiceBusProcessor(default); - var cm = _cm; + ServiceBusProcessor processor = _cm.Messaging.GetServiceBusProcessor(default); + CloudMachineClient cm = _cm; // TODO: How to unsubscribe? // TODO: Use a subscription filter to ignore Event Grid system events diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageFile.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageFile.cs index e3c27ee57c986..5477f2409d747 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageFile.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageFile.cs @@ -15,7 +15,7 @@ public class StorageFile { private readonly Response _response; - private StorageServices _storage; + private readonly StorageServices _storage; /// /// The path of the file in the storage account. @@ -23,7 +23,7 @@ public class StorageFile public string Path { get; internal set; } /// - /// The requestId for the storage operation that triggered this event + /// The requestId for the storage operation that triggered this event. /// public string RequestId { get; internal set; } diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs index 26fda12e48bce..18c79ef8aa1b2 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs @@ -86,7 +86,7 @@ public async Task UploadJsonAsync(object json, string name = default, bo public string Upload(Stream fileStream, string name = default, string contentType = default, bool overwrite = false) { BlockBlobClient client = GetBlobClient(ref name); - BlobUploadOptions options = CreateUploadOptions(overwrite, contentType); + BlobUploadOptions options = StorageServices.CreateUploadOptions(overwrite, contentType); client.Upload(fileStream, options); return name; @@ -103,7 +103,7 @@ public string Upload(Stream fileStream, string name = default, string contentTyp public async Task UploadAsync(Stream fileStream, string name = default, string contentType = default, bool overwrite = false) { BlockBlobClient client = GetBlobClient(ref name); - BlobUploadOptions options = CreateUploadOptions(overwrite, contentType); + BlobUploadOptions options = StorageServices.CreateUploadOptions(overwrite, contentType); await client.UploadAsync(fileStream, options).ConfigureAwait(false); return name; @@ -117,9 +117,9 @@ private BlockBlobClient GetBlobClient(ref string name) return client; } - private BlobUploadOptions CreateUploadOptions(bool overwrite, string contentType) + private static BlobUploadOptions CreateUploadOptions(bool overwrite, string contentType) { - if (contentType == null) contentType = ContentType.ApplicationOctetStream.ToString(); + contentType ??= ContentType.ApplicationOctetStream.ToString(); BlobUploadOptions options = new() { Conditions = overwrite ? null : new BlobRequestConditions { IfNoneMatch = new ETag("*") }, @@ -206,15 +206,15 @@ public async Task DeleteAsync(string path) private BlobClient GetBlobClientFromPath(string path, string containerName) { - var _blobContainer = GetDefaultContainer(); - var blobPath = ConvertPathToBlobPath(path, _blobContainer); + BlobContainerClient _blobContainer = GetDefaultContainer(); + string blobPath = ConvertPathToBlobPath(path, _blobContainer); if (containerName is null) { return _blobContainer.GetBlobClient(blobPath); } else { - var container = GetContainer(containerName); + BlobContainerClient container = GetContainer(containerName); container.CreateIfNotExists(); return container.GetBlobClient(blobPath); } diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/ChatTools.cs b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/ChatTools.cs index 2564750b6da71..f258962932ea5 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/ChatTools.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/ChatTools.cs @@ -17,8 +17,8 @@ public class ChatTools { private static readonly BinaryData s_noparams = BinaryData.FromString("""{ "type" : "object", "properties" : {} }"""); - private readonly Dictionary _methods = new Dictionary(); - private readonly List _definitions = new List(); + private readonly Dictionary _methods = []; + private readonly List _definitions = []; /// /// Initializes a new instance of the class. @@ -26,7 +26,7 @@ public class ChatTools /// public ChatTools(params Type[] tools) { - foreach (var functionHolder in tools) + foreach (Type functionHolder in tools) Add(functionHolder); } @@ -42,7 +42,7 @@ public ChatTools(params Type[] tools) public static implicit operator ChatCompletionOptions(ChatTools tools) { ChatCompletionOptions options = new(); - foreach (var tool in tools.Definitions) + foreach (ChatTool tool in tools.Definitions) { options.Tools.Add(tool); } @@ -98,10 +98,10 @@ public string Call(ChatToolCall call) if (call.FunctionArguments != null) { var document = JsonDocument.Parse(call.FunctionArguments); - var json = document.RootElement; + JsonElement json = document.RootElement; foreach (JsonProperty argument in json.EnumerateObject()) { - var value = argument.Value; + JsonElement value = argument.Value; switch (value.ValueKind) { case JsonValueKind.String: @@ -151,7 +151,7 @@ public IEnumerable CallAll(IEnumerable toolCalls) protected virtual string GetMethodInfoToDescription(MethodInfo function) { var description = function.Name; - var attribute = function.GetCustomAttribute(); + DescriptionAttribute attribute = function.GetCustomAttribute(); if (attribute != null) { description = attribute.Description; @@ -167,7 +167,7 @@ protected virtual string GetMethodInfoToDescription(MethodInfo function) protected virtual string GetParameterInfoToDescription(ParameterInfo parameter) { var description = parameter.Name; - var attribute = parameter.GetCustomAttribute(); + DescriptionAttribute attribute = parameter.GetCustomAttribute(); if (attribute != null) { description = attribute.Description; @@ -184,9 +184,9 @@ protected virtual string GetMethodInfoToName(MethodInfo function) { var sb = new StringBuilder(); sb.Append(function.Name); - foreach (var parameter in function.GetParameters()) + foreach (ParameterInfo parameter in function.GetParameters()) { - sb.Append($"_{ClrToJsonTypeUtf16(parameter.ParameterType)}"); + sb.Append($"_{ChatTools.ClrToJsonTypeUtf16(parameter.ParameterType)}"); } return sb.ToString(); } @@ -197,7 +197,7 @@ protected virtual string GetMethodInfoToName(MethodInfo function) /// /// /// - protected ReadOnlySpan ClrToJsonTypeUtf8(Type clrType) + protected static ReadOnlySpan ClrToJsonTypeUtf8(Type clrType) { if (clrType == typeof(double)) return "number"u8; @@ -215,7 +215,7 @@ protected ReadOnlySpan ClrToJsonTypeUtf8(Type clrType) /// /// /// - protected string ClrToJsonTypeUtf16(Type clrType) + protected static string ClrToJsonTypeUtf16(Type clrType) { if (clrType == typeof(double)) return "number"; @@ -233,7 +233,7 @@ private BinaryData ParametersToJson(ParameterInfo[] parameters) return s_noparams; var required = new List(); - MemoryStream stream = new MemoryStream(); + MemoryStream stream = new(); var writer = new Utf8JsonWriter(stream); writer.WriteStartObject(); writer.WriteString("type"u8, "object"u8); @@ -241,7 +241,7 @@ private BinaryData ParametersToJson(ParameterInfo[] parameters) foreach (ParameterInfo parameter in parameters) { writer.WriteStartObject(parameter.Name!); - writer.WriteString("type"u8, ClrToJsonTypeUtf8(parameter.ParameterType)); + writer.WriteString("type"u8, ChatTools.ClrToJsonTypeUtf8(parameter.ParameterType)); writer.WriteString("description"u8, GetParameterInfoToDescription(parameter)); writer.WriteEndObject(); if (!parameter.IsOptional || (parameter.HasDefaultValue && parameter.DefaultValue is not null)) diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/EmbeddingsVectorbase.cs b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/EmbeddingsVectorbase.cs index 965a42876ed9b..ab1cf6a63892e 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/EmbeddingsVectorbase.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/EmbeddingsVectorbase.cs @@ -14,7 +14,7 @@ public class EmbeddingsVectorbase { private readonly EmbeddingClient _client; private readonly VectorbaseStore _store; - private readonly List _todo = new List(); + private readonly List _todo = []; private readonly int _chuckSize; /// @@ -73,9 +73,9 @@ private void ProcessToDo() { lock (_todo) { - var embeddings = _client.GenerateEmbeddings(_todo); + OpenAIEmbeddingCollection embeddings = _client.GenerateEmbeddings(_todo); - foreach (var embedding in embeddings.Value) + foreach (OpenAIEmbedding embedding in embeddings) { ReadOnlyMemory vector = embedding.ToFloats(); string item = _todo[(int)embedding.Index]; @@ -87,8 +87,8 @@ private void ProcessToDo() private ReadOnlyMemory GetEmbedding(string fact) { - var embedding = _client.GenerateEmbedding(fact); - return embedding.Value.ToFloats(); + OpenAIEmbedding embedding = _client.GenerateEmbedding(fact); + return embedding.ToFloats(); } private void ChunkFactAndAddToTodo(string text, int chunkSize) diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/MemoryVectorbaseStore.cs b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/MemoryVectorbaseStore.cs index e0e0428766fd8..6dfca9073eb98 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/MemoryVectorbaseStore.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/MemoryVectorbaseStore.cs @@ -8,7 +8,7 @@ namespace Azure.CloudMachine.OpenAI; internal class MemoryVectorbaseStore : VectorbaseStore { - private readonly List _entries = new List(); + private readonly List _entries = []; public override IEnumerable Find(ReadOnlyMemory vector, FindOptions options) { @@ -24,14 +24,14 @@ public override IEnumerable Find(ReadOnlyMemory vector, } distances.Sort(((float D1, int I1) v1, (float D2, int I2) v2) => v1.D1.CompareTo(v2.D2)); - var results = new List(options.MaxEntries); - var top = Math.Min(options.MaxEntries, distances.Count); + List results = new(options.MaxEntries); + int top = Math.Min(options.MaxEntries, distances.Count); for (int i = 0; i < top; i++) { - var distance = distances[i].Distance; + float distance = distances[i].Distance; if (distance > options.Threshold) break; - var index = distances[i].Index; + int index = distances[i].Index; results.Add(_entries[index]); } return results; @@ -41,8 +41,8 @@ public override int Add(VectorbaseEntry entry) { lock (_entries) { - var id = _entries.Count; - var newEntry = new VectorbaseEntry(entry.Vector, entry.Data, id); + int id = _entries.Count; + VectorbaseEntry newEntry = new(entry.Vector, entry.Data, id); _entries.Add(newEntry); return id; } @@ -50,7 +50,7 @@ public override int Add(VectorbaseEntry entry) public override void Add(IReadOnlyList entries) { - foreach (var entry in entries) + foreach (VectorbaseEntry entry in entries) { Add(entry); } diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/VectorbaseStore.cs b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/VectorbaseStore.cs index 6944fe2cf76fb..c25a4c25b39ab 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/VectorbaseStore.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/VectorbaseStore.cs @@ -59,10 +59,6 @@ public static float CosineSimilarity(ReadOnlySpan x, ReadOnlySpan public readonly struct VectorbaseEntry { - private readonly ReadOnlyMemory _vector; - private readonly int? _id; - private readonly BinaryData _data; - /// /// Initializes a new instance of the class. /// @@ -71,23 +67,23 @@ public readonly struct VectorbaseEntry /// public VectorbaseEntry(ReadOnlyMemory vector, BinaryData data, int? id = default) { - _vector = vector; - _data = data; - _id = id; + Vector = vector; + Data = data; + Id = id; } /// /// Gets the data associated with the entry. /// - public BinaryData Data => _data; + public BinaryData Data { get; } /// /// Gets the vector associated with the entry. /// - public ReadOnlyMemory Vector => _vector; + public ReadOnlyMemory Vector { get; } /// /// Gets the id associated with the entry. /// - public int? Id => _id; + public int? Id { get; } } diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs index 4b72fc998eeaa..b64ef5f0dc75a 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs @@ -51,8 +51,8 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf IsHttp20Enabled = true, MinTlsVersion = AppServiceSupportedTlsVersion.Tls1_2, IsWebSocketsEnabled = true, - AppSettings = new() - { + AppSettings = + [ // This is used by the CloudMachineWorkspace to detect that it is running in a deployed App Service. // The ClientId is used to create a ManagedIdentityCredential so that it wires up to our CloudMachine user-assigned identity. new AppServiceNameValuePair @@ -60,7 +60,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf Name = "CLOUDMACHINE_MANAGED_IDENTITY_CLIENT_ID", Value = infrastructure.Identity.ClientId }, - } + ] } }; infrastructure.AddResource(appService); diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs index a7ac7a24c7804..0d8f55bdc6161 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs @@ -16,10 +16,7 @@ public class KeyVaultFeature : CloudMachineFeature public KeyVaultFeature(KeyVaultSku? sku = default) { - if (sku == null) - { - sku = new KeyVaultSku { Name = KeyVaultSkuName.Standard, Family = KeyVaultSkuFamily.A, }; - } + sku ??= new KeyVaultSku { Name = KeyVaultSkuName.Standard, Family = KeyVaultSkuFamily.A, }; Sku = sku; } protected override ProvisionableResource EmitCore(CloudMachineInfrastructure infrastructure) @@ -31,7 +28,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf Properties = new KeyVaultProperties { - Sku = this.Sku, + Sku = Sku, TenantId = BicepFunction.GetSubscription().TenantId, EnabledForDeployment = true, AccessPolicies = [ diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs index 8b0fb82c3d0a5..3b9234a622cd5 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs @@ -10,14 +10,14 @@ namespace Azure.CloudMachine.OpenAI; internal class OpenAIFeature : CloudMachineFeature { - private List _models = new List(); + private readonly List _models = []; public OpenAIFeature() { } protected override ProvisionableResource EmitCore(CloudMachineInfrastructure cloudMachine) { - CognitiveServicesAccount cognitiveServices = CreateOpenAIAccount(cloudMachine); + CognitiveServicesAccount cognitiveServices = OpenAIFeature.CreateOpenAIAccount(cloudMachine); cloudMachine.AddResource(cognitiveServices); RequiredSystemRoles.Add(cognitiveServices, [(CognitiveServicesBuiltInRole.GetBuiltInRoleName(CognitiveServicesBuiltInRole.CognitiveServicesOpenAIContributor) ,CognitiveServicesBuiltInRole.CognitiveServicesOpenAIContributor.ToString())]); @@ -48,7 +48,7 @@ internal void AddModel(OpenAIModel model) _models.Add(model); } - internal CognitiveServicesAccount CreateOpenAIAccount(CloudMachineInfrastructure cm) + internal static CognitiveServicesAccount CreateOpenAIAccount(CloudMachineInfrastructure cm) { return new("openai") { diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModel.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModel.cs index 077927ecf33e2..f2d6272d61a55 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModel.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModel.cs @@ -24,7 +24,7 @@ public OpenAIModel(string model, string modelVersion, AIModelKind kind = AIModel internal OpenAIFeature Account { get; set; } = default!; - private OpenAIFeature GetOrCreateOpenAI(CloudMachineInfrastructure cm) + private static OpenAIFeature GetOrCreateOpenAI(CloudMachineInfrastructure cm) { foreach (OpenAIFeature feature in cm.Features.FindAll()) { @@ -37,7 +37,7 @@ private OpenAIFeature GetOrCreateOpenAI(CloudMachineInfrastructure cm) public override void AddTo(CloudMachineInfrastructure cm) { - OpenAIFeature openAI = GetOrCreateOpenAI(cm); + OpenAIFeature openAI = OpenAIModel.GetOrCreateOpenAI(cm); openAI.AddModel(this); } @@ -51,7 +51,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure cm) }; Debug.Assert(Account != null); - var emitted = Account!.Emitted; + ProvisionableResource emitted = Account!.Emitted; if (emitted == null) { Account.Emit(cm); diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/AzdHelpers.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/AzdHelpers.cs index f2907aaceb68c..950368ffe3f25 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/AzdHelpers.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/AzdHelpers.cs @@ -88,7 +88,7 @@ internal static string ReadOrCreateCmid() cmid = GenerateCloudMachineId(); using FileStream file = File.OpenWrite(appsettings); - Utf8JsonWriter writer = new Utf8JsonWriter(file); + Utf8JsonWriter writer = new(file); writer.WriteStartObject(); writer.WritePropertyName("CloudMachine"u8); writer.WriteStartObject(); diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs index bf59371071c8c..6e9c50a770d3b 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs @@ -1,14 +1,14 @@ // Copyright (c) Microsoft Corporation. All rights reserved. // Licensed under the MIT License. -using System.IO; using System; -using System.ClientModel.TypeSpec; using System.ClientModel.Primitives; +using System.ClientModel.TypeSpec; +using System.IO; +using System.Text.Json; using Azure.AI.OpenAI; using Azure.Core; using Azure.Identity; -using System.Text.Json; namespace Azure.CloudMachine; @@ -46,7 +46,7 @@ public static bool Execute(string[] args, Action? co return false; - bool Handled(bool exitProcessIfHandled) + static bool Handled(bool exitProcessIfHandled) { if (exitProcessIfHandled) Environment.Exit(0); return true; @@ -63,9 +63,8 @@ private static void ListAzureOpenaAIModels(string cmid, string? option) ); string audience = "https://cognitiveservices.azure.com/.default"; TokenCredentialAuthenticationPolicy auth = new(credential, [audience]); - RestClient client = new RestClient(auth); + RestClient client = new(auth); string uri = $"https://{cmid}.openai.azure.com/openai/models?api-version=2024-10-21"; - RequestOptions options = new RequestOptions(); PipelineResponse response = client.Get(uri); if (response.Status != 200) { diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs index 0973d0106f44c..d553857b90968 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs @@ -2,13 +2,13 @@ // Licensed under the MIT License. using System; -using Azure.Provisioning.Authorization; -using Azure.Provisioning.Expressions; -using Azure.Provisioning.Roles; -using Azure.Provisioning.Primitives; using System.Collections.Generic; using Azure.Provisioning; +using Azure.Provisioning.Authorization; using Azure.Provisioning.CloudMachine; +using Azure.Provisioning.Expressions; +using Azure.Provisioning.Primitives; +using Azure.Provisioning.Roles; namespace Azure.CloudMachine; @@ -16,20 +16,19 @@ public class CloudMachineInfrastructure { internal const string SB_PRIVATE_TOPIC = "cm_servicebus_topic_private"; internal const string SB_PRIVATE_SUB = "cm_servicebus_subscription_private"; - private readonly string _cmId; - private Infrastructure _infrastructure = new Infrastructure("cm"); - private List _resources = new(); - public FeatureCollection Features { get; } = new(); - internal List Endpoints { get; } = new(); + private readonly Infrastructure _infrastructure = new("cm"); + private readonly List _resources = []; + internal List Endpoints { get; } = []; + public FeatureCollection Features { get; } = new(); public UserAssignedIdentity Identity { get; private set; } - public string Id => _cmId; + public string Id { get; private set; } /// /// The common principalId parameter. /// - public ProvisioningParameter PrincipalIdParameter => new ProvisioningParameter("principalId", typeof(string)); + public ProvisioningParameter PrincipalIdParameter => new("principalId", typeof(string)); ///// ///// The common principalType parameter. @@ -43,27 +42,27 @@ public class CloudMachineInfrastructure public CloudMachineInfrastructure(string cmId) { - _cmId = cmId; + Id = cmId; // setup CM identity Identity = new UserAssignedIdentity("cm_identity"); - Identity.Name = _cmId; - _infrastructure.Add(new ProvisioningOutput($"cm_managed_identity_id", typeof(string)) { Value = Identity.Id }); + Identity.Name = Id; + _infrastructure.Add(new ProvisioningOutput("cm_managed_identity_id", typeof(string)) { Value = Identity.Id }); // Add core features - var storage = new StorageFeature(_cmId); + var storage = new StorageFeature(Id); Features.Add(storage); - var sbNamespace = new ServiceBusNamespaceFeature(_cmId); + var sbNamespace = new ServiceBusNamespaceFeature(Id); Features.Add(sbNamespace); - var sbTopic_private = new ServiceBusTopicFeature("cm_servicebus_topic_private", sbNamespace); - Features.Add(sbTopic_private); - var sbTopic_default = new ServiceBusTopicFeature("cm_servicebus_default_topic", sbNamespace); - Features.Add(sbTopic_default); - Features.Add(new ServiceBusSubscriptionFeature("cm_servicebus_subscription_private", sbTopic_private)); - Features.Add(new ServiceBusSubscriptionFeature("cm_servicebus_subscription_default", sbTopic_default)); - var systemTopic = new EventGridSystemTopicFeature(_cmId, storage); + var sbTopicPrivate = new ServiceBusTopicFeature("cm_servicebus_topic_private", sbNamespace); + Features.Add(sbTopicPrivate); + var sbTopicDefault = new ServiceBusTopicFeature("cm_servicebus_default_topic", sbNamespace); + Features.Add(sbTopicDefault); + Features.Add(new ServiceBusSubscriptionFeature("cm_servicebus_subscription_private", sbTopicPrivate)); + Features.Add(new ServiceBusSubscriptionFeature("cm_servicebus_subscription_default", sbTopicDefault)); + var systemTopic = new EventGridSystemTopicFeature(Id, storage); Features.Add(systemTopic); - Features.Add(new SystemTopicEventSubscriptionFeature("cm_eventgrid_subscription_blob", systemTopic, sbTopic_private, sbNamespace)); + Features.Add(new SystemTopicEventSubscriptionFeature("cm_eventgrid_subscription_blob", systemTopic, sbTopicPrivate, sbNamespace)); } public void AddResource(NamedProvisionableConstruct resource) @@ -85,8 +84,7 @@ public void AddEndpoints() public ProvisioningPlan Build(ProvisioningBuildOptions? context = null) { - if (context == null) - context = new ProvisioningBuildOptions(); + context ??= new ProvisioningBuildOptions(); Features.Emit(this); @@ -127,28 +125,28 @@ public override IEnumerable ResolveResources(IEnumerable userPrincipal in userPrincipals) { - yield return new RoleAssignment($"{resource.BicepIdentifier}_{userPrincipal.Value.ToString().Replace('-', '_')}_{role.RoleName}") + yield return new RoleAssignment($"{resource.BicepIdentifier}_{userPrincipal.Value.ToString().Replace('-', '_')}_{RoleName}") { - Name = BicepFunction.CreateGuid(resource.BicepIdentifier, userPrincipal, BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", role.RoleId)), + Name = BicepFunction.CreateGuid(resource.BicepIdentifier, userPrincipal, BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", RoleId)), Scope = new IdentifierExpression(resource.BicepIdentifier), PrincipalType = RoleManagementPrincipalType.User, - RoleDefinitionId = BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", role.RoleId), + RoleDefinitionId = BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", RoleId), PrincipalId = userPrincipal }; } foreach (UserAssignedIdentity identity in managedIdentities) { - yield return new RoleAssignment($"{resource.BicepIdentifier}_{identity.BicepIdentifier}_{role.RoleName}") + yield return new RoleAssignment($"{resource.BicepIdentifier}_{identity.BicepIdentifier}_{RoleName}") { - Name = BicepFunction.CreateGuid(resource.BicepIdentifier, identity.Id, BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", role.RoleId)), + Name = BicepFunction.CreateGuid(resource.BicepIdentifier, identity.Id, BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", RoleId)), Scope = new IdentifierExpression(resource.BicepIdentifier), PrincipalType = RoleManagementPrincipalType.ServicePrincipal, - RoleDefinitionId = BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", role.RoleId), + RoleDefinitionId = BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", RoleId), PrincipalId = identity.PrincipalId }; } diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs index 441fd1be69022..395094679aab9 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs @@ -1,9 +1,7 @@ // Copyright (c) Microsoft Corporation. All rights reserved. // Licensed under the MIT License. -using Azure.Provisioning.Authorization; using Azure.Provisioning.CloudMachine; -using Azure.Provisioning.Expressions; using Azure.Provisioning.Primitives; using Azure.Provisioning.ServiceBus; diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs index b7f67141bc1f1..45411ace4e854 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs @@ -14,9 +14,9 @@ namespace Azure.CloudMachine; public class StorageFeature : CloudMachineFeature { - private List _containerNames; - private StorageSkuName _skuName; - private string _name; + private readonly List _containerNames; + private readonly StorageSkuName _skuName; + private readonly string _name; public StorageFeature(string accountName, StorageSkuName sku = StorageSkuName.StandardLrs, IEnumerable? containerNames = null) { diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs index 85faafb2435d8..386aac7790423 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs @@ -14,9 +14,9 @@ public class SystemTopicEventSubscriptionFeature(string name, EventGridSystemTop { protected override ProvisionableResource EmitCore(CloudMachineInfrastructure infrastructure) { - var serviceBusNamespace = ValidateIsOfType(parentNamespace); + ServiceBusNamespace serviceBusNamespace = ValidateIsOfType(parentNamespace); - var role = ServiceBusBuiltInRole.AzureServiceBusDataSender; + ServiceBusBuiltInRole role = ServiceBusBuiltInRole.AzureServiceBusDataSender; var roleAssignment = new RoleAssignment($"cm_servicebus_{ValidateIsOfType(parent).Name.Value}_role") { Name = BicepFunction.CreateGuid(serviceBusNamespace.Id, infrastructure.Identity.Id, BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", role.ToString())), @@ -26,7 +26,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf PrincipalId = infrastructure.Identity.PrincipalId, }; - var systemTopic = ValidateIsOfType(parent); + SystemTopic systemTopic = ValidateIsOfType(parent); var subscription = new SystemTopicEventSubscription("cm_eventgrid_subscription_blob", "2022-06-15") { Name = name, diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/LoggingPolicy.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/LoggingPolicy.cs index fd05bf879bda3..12fefe1e4aa27 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/LoggingPolicy.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/LoggingPolicy.cs @@ -13,7 +13,7 @@ internal class LoggingPolicy : PipelinePolicy { public LoggingPolicy() {} - public List AllowedHeaders { get; } = new List(["Content-Type", "Accept", "User-Agent", "x-ms-client-request-id"]); + public List AllowedHeaders { get; } = ["Content-Type", "Accept", "User-Agent", "x-ms-client-request-id"]; public override void Process(PipelineMessage message, IReadOnlyList pipeline, int currentIndex) { LogRequest(message); @@ -45,7 +45,7 @@ protected virtual void LogResponse(PipelineMessage message) } protected virtual string FormatRequestLog(PipelineMessage message) { - StringBuilder logMessage = new StringBuilder(); + StringBuilder logMessage = new(); FormatRequestLine(message, logMessage); FormatHeaders(message, logMessage); FormatContent(message, logMessage); @@ -53,9 +53,9 @@ protected virtual string FormatRequestLog(PipelineMessage message) { } protected virtual string FormatResponseLog(PipelineMessage message) { - StringBuilder logMessage = new StringBuilder(); + StringBuilder logMessage = new(); PipelineResponse response = message.Response!; - logMessage.Append(response.Status.ToString()); + logMessage.Append(response.Status); logMessage.Append(' '); logMessage.AppendLine(response.ReasonPhrase); FormatHeaders(message, logMessage); @@ -72,7 +72,7 @@ protected virtual void FormatRequestLine(PipelineMessage message, StringBuilder } protected virtual void FormatHeaders(PipelineMessage message, StringBuilder logMessage) { - foreach (var header in message.Request.Headers) + foreach (KeyValuePair header in message.Request.Headers) { if (AllowedHeaders.Contains(header.Key)) { diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/RestClient.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/RestClient.cs index b56813f79dfc3..9a7343f0b41b4 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/RestClient.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/RestClient.cs @@ -9,11 +9,9 @@ namespace Azure { public class RestClient { - private static readonly RestClient _shared = new RestClient(); - private readonly ClientPipeline _pipeline; - public static RestClient Shared => _shared; + public static RestClient Shared { get; } = new RestClient(); public RestClient() : this(default(RestClientOptions)) { @@ -25,15 +23,14 @@ public RestClient(PipelinePolicy auth) : this(CreateOptions(auth)) private static RestClientOptions CreateOptions(PipelinePolicy auth) { - RestClientOptions options = new RestClientOptions(); + RestClientOptions options = new(); options.AddPolicy(auth, PipelinePosition.PerTry); return options; } private RestClient(RestClientOptions? options = default) { - if (options == null) - options = new RestClientOptions(); + options ??= new RestClientOptions(); _pipeline = ClientPipeline.Create(options); } diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Tsp/TypeSpecWriter.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Tsp/TypeSpecWriter.cs index f8fe852c9a0b1..2b1c6af5713b5 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Tsp/TypeSpecWriter.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Tsp/TypeSpecWriter.cs @@ -34,8 +34,8 @@ public static void WriteServer(Stream output, Type service) writer.WriteLine(); writer.WriteLine($"@client interface {name}Client {{"); - HashSet models = new HashSet(); - foreach (var method in service.GetMethods()) + HashSet models = []; + foreach (MethodInfo method in service.GetMethods()) { writer.Write(" "); WriteOperation(writer, method, models); @@ -44,7 +44,7 @@ public static void WriteServer(Stream output, Type service) writer.WriteLine(); writer.Flush(); - foreach (var model in models) + foreach (Type model in models) { WriteModel(output, model); } @@ -69,7 +69,7 @@ public static void WriteModel(Stream output, Type model) } public static void WriteModel(Stream output) { - var model = typeof(T); + Type model = typeof(T); WriteModel(output, model); } @@ -77,7 +77,7 @@ private static void WriteClassModel(StreamWriter writer, Type model) { writer.WriteLine($"model {model.Name} {{"); - foreach (var property in model.GetProperties()) + foreach (PropertyInfo property in model.GetProperties()) { WriteClassModelProperty(writer, property); } @@ -109,7 +109,7 @@ private static void WriteOperation(StreamWriter writer, MethodInfo method, HashS writer.Write($"{httpVerb} @route(\"{ToCamel(methodName)}\") {methodName}("); bool first = true; - foreach (var parameter in method.GetParameters()) + foreach (ParameterInfo parameter in method.GetParameters()) { Type parameterType = parameter.ParameterType; @@ -139,7 +139,7 @@ private static void WriteOperation(StreamWriter writer, MethodInfo method, HashS writer.WriteLine(") : {"); writer.WriteLine($" @statusCode statusCode: 200;"); - var returnType = method.ReturnType; + Type returnType = method.ReturnType; if (returnType == typeof(Task)) returnType = typeof(void); else if (returnType.IsGenericType && returnType.GetGenericTypeDefinition() == typeof(Task<>)) returnType = returnType.GetGenericArguments()[0]; @@ -197,7 +197,7 @@ private static bool IsModel(this Type type) private static string ToCamel(this string text) { - return $"{Char.ToLower(text[0])}{text.Substring(1)}"; + return $"{char.ToLower(text[0])}{text.Substring(1)}"; } private static string ToTspType(this Type type) { From 1ca450bc79acee36787434616e55057c94ad4105 Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Mon, 25 Nov 2024 14:37:30 -0800 Subject: [PATCH 02/24] more cleanup --- .../src/ClientConnectionOptions.cs | 92 +++++++++++++++++++ .../Azure.CloudMachine/src/ClientWorkspace.cs | 86 ----------------- .../src/CoreServices/StorageFile.cs | 7 ++ .../src/CoreServices/StorageServices.cs | 12 +-- .../src/extensions/AzureOpenAIExtensions.cs | 6 +- .../src/extensions/ChatTools.cs | 45 ++------- .../tests/CloudMachineTests.cs | 6 +- .../AzureSdkExtensions/AppServiceFeature.cs | 8 +- .../src/AzureSdkExtensions/OpenAIFeature.cs | 8 +- .../{OpenAIModel.cs => OpenAIModelFeature.cs} | 7 +- .../src/CDKLevel3/CloudMachineFeature.cs | 6 +- .../CDKLevel3/EventGridSystemTopicFeature.cs | 6 +- .../src/CDKLevel3/FeatureCollection.cs | 2 + .../ServiceBusSubscriptionFeature.cs | 2 +- .../src/CDKLevel3/ServiceBusTopicFeature.cs | 2 +- .../SystemTopicEventSubscriptionFeature.cs | 8 +- .../tests/CloudMachineTests.cs | 14 +-- 17 files changed, 146 insertions(+), 171 deletions(-) create mode 100644 sdk/cloudmachine/Azure.CloudMachine/src/ClientConnectionOptions.cs rename sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/{OpenAIModel.cs => OpenAIModelFeature.cs} (91%) diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/ClientConnectionOptions.cs b/sdk/cloudmachine/Azure.CloudMachine/src/ClientConnectionOptions.cs new file mode 100644 index 0000000000000..f2afd6cba5076 --- /dev/null +++ b/sdk/cloudmachine/Azure.CloudMachine/src/ClientConnectionOptions.cs @@ -0,0 +1,92 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +using System; + +namespace Azure.Core; + +/// +/// Represents the connection options for a client. +/// +public readonly struct ClientConnectionOptions +{ + /// + /// Initializes a new instance of the struct with the specified endpoint and API key. + /// + /// The endpoint URI. + /// The API key credential. + public ClientConnectionOptions(Uri endpoint, string apiKey) + { + Endpoint = endpoint; + ApiKeyCredential = apiKey; + ConnectionKind = ClientConnectionKind.ApiKey; + } + + /// + /// Initializes a new instance of the struct with the specified endpoint and token credential. + /// + /// The endpoint URI. + /// The token credential. + public ClientConnectionOptions(Uri endpoint, TokenCredential credential) + { + Endpoint = endpoint; + TokenCredential = credential; + ConnectionKind = ClientConnectionKind.EntraId; + } + + /// + /// Initializes a new instance of the struct with the specified subclient ID. + /// + /// The subclient ID. + public ClientConnectionOptions(string subclientId) + { + Id = subclientId; + ConnectionKind = ClientConnectionKind.OutOfBand; + } + + /// + /// Gets the kind of connection used by the client. + /// + public ClientConnectionKind ConnectionKind { get; } + + /// + /// Gets the endpoint URI. + /// + public Uri Endpoint { get; } + + /// + /// Gets the subclient ID. + /// + public string Id { get; } + + /// + /// Gets the API key credential. + /// + public string ApiKeyCredential { get; } + + /// + /// Gets the token credential. + /// + public TokenCredential TokenCredential { get; } +} + +/// +/// Specifies the kind of connection used by the client. +/// +public enum ClientConnectionKind +{ + /// + /// Represents a connection using Entra ID. + /// + EntraId, + + /// + /// Represents a connection using an API key. + /// + ApiKey, + + /// + /// Represents a connection using an out-of-band method. + /// + OutOfBand +} diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/ClientWorkspace.cs b/sdk/cloudmachine/Azure.CloudMachine/src/ClientWorkspace.cs index 52ebda1f4216a..e441cbd53a58b 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/ClientWorkspace.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/ClientWorkspace.cs @@ -26,89 +26,3 @@ public abstract class ClientWorkspace [EditorBrowsable(EditorBrowsableState.Never)] public ClientCache Subclients { get; } = new ClientCache(); } - -/// -/// Represents the connection options for a client. -/// -public readonly struct ClientConnectionOptions -{ - /// - /// Initializes a new instance of the struct with the specified endpoint and API key. - /// - /// The endpoint URI. - /// The API key credential. - public ClientConnectionOptions(Uri endpoint, string apiKey) - { - Endpoint = endpoint; - ApiKeyCredential = apiKey; - ConnectionKind = ClientConnectionKind.ApiKey; - } - - /// - /// Initializes a new instance of the struct with the specified endpoint and token credential. - /// - /// The endpoint URI. - /// The token credential. - public ClientConnectionOptions(Uri endpoint, TokenCredential credential) - { - Endpoint = endpoint; - TokenCredential = credential; - ConnectionKind = ClientConnectionKind.EntraId; - } - - /// - /// Initializes a new instance of the struct with the specified subclient ID. - /// - /// The subclient ID. - public ClientConnectionOptions(string subclientId) - { - Id = subclientId; - ConnectionKind = ClientConnectionKind.OutOfBand; - } - - /// - /// Gets the kind of connection used by the client. - /// - public ClientConnectionKind ConnectionKind { get; } - - /// - /// Gets the endpoint URI. - /// - public Uri Endpoint { get; } - - /// - /// Gets the subclient ID. - /// - public string Id { get; } - - /// - /// Gets the API key credential. - /// - public string ApiKeyCredential { get; } - - /// - /// Gets the token credential. - /// - public TokenCredential TokenCredential { get; } -} - -/// -/// Specifies the kind of connection used by the client. -/// -public enum ClientConnectionKind -{ - /// - /// Represents a connection using Entra ID. - /// - EntraId, - - /// - /// Represents a connection using an API key. - /// - ApiKey, - - /// - /// Represents a connection using an out-of-band method. - /// - OutOfBand -} diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageFile.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageFile.cs index 5477f2409d747..ba36c0533efce 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageFile.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageFile.cs @@ -46,6 +46,13 @@ public class StorageFile public BinaryData Download() => _storage.Download(Path); + /// + /// Downloads the file from the storage account. + /// + /// + public async Task DownloadAsync() + => await _storage.DownloadAsync(Path).ConfigureAwait(false); + // public async Task DownloadAsync() // => await _storage.DownloadBlobAsync(Path).ConfigureAwait(false); diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs index 18c79ef8aa1b2..75a7992bef1f5 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs @@ -52,26 +52,26 @@ private BlobContainerClient GetContainer(string containerName) /// /// Uploads a JSON object to the storage account. /// - /// + /// /// /// /// - public string UploadJson(object json, string name = default, bool overwrite = false) + public string UploadJson(object serializable, string name = default, bool overwrite = false) { - BinaryData data = BinaryData.FromObjectAsJson(json); + BinaryData data = BinaryData.FromObjectAsJson(serializable); return Upload(data, name, overwrite); } /// /// Uploads a JSON object to the storage account. /// - /// + /// /// /// /// - public async Task UploadJsonAsync(object json, string name = default, bool overwrite = false) + public async Task UploadJsonAsync(object serializable, string name = default, bool overwrite = false) { - BinaryData data = BinaryData.FromObjectAsJson(json); + BinaryData data = BinaryData.FromObjectAsJson(serializable); return await UploadAsync(data, name, overwrite).ConfigureAwait(false); } diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/AzureOpenAIExtensions.cs b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/AzureOpenAIExtensions.cs index 7a66c81f9d644..7068300733b39 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/AzureOpenAIExtensions.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/AzureOpenAIExtensions.cs @@ -60,12 +60,12 @@ public static string AsText(this ChatCompletion completion) /// /// returns full text of all parts. /// - /// + /// /// - public static string AsText(this ChatMessageContent completion) + public static string AsText(this ChatMessageContent content) { StringBuilder sb = new(); - foreach (ChatMessageContentPart part in completion) + foreach (ChatMessageContentPart part in content) { switch (part.Kind) { diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/ChatTools.cs b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/ChatTools.cs index f258962932ea5..be51567ab0f6b 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/ChatTools.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/ChatTools.cs @@ -68,7 +68,7 @@ public void Add(Type functions) public void Add(MethodInfo function) { var name = function.Name; - var chatTool = ChatTool.CreateFunctionTool(name, GetMethodInfoToDescription(function), ParametersToJson(function.GetParameters())); + var chatTool = ChatTool.CreateFunctionTool(name, MethodInfoToDescription(function), ChatTools.ParametersToJson(function.GetParameters())); _definitions.Add(chatTool); _methods[name] = function; } @@ -122,7 +122,7 @@ public string Call(ChatToolCall call) } } var name = call.FunctionName; - var result = Call(name, arguments.ToArray()); + var result = Call(name, [.. arguments]); return result; } @@ -143,12 +143,7 @@ public IEnumerable CallAll(IEnumerable toolCalls) return messages; } - /// - /// Gets the description of a . - /// - /// - /// - protected virtual string GetMethodInfoToDescription(MethodInfo function) + private static string MethodInfoToDescription(MethodInfo function) { var description = function.Name; DescriptionAttribute attribute = function.GetCustomAttribute(); @@ -159,12 +154,7 @@ protected virtual string GetMethodInfoToDescription(MethodInfo function) return description; } - /// - /// Gets the description of a . - /// - /// - /// - protected virtual string GetParameterInfoToDescription(ParameterInfo parameter) + private static string ParameterInfoToDescription(ParameterInfo parameter) { var description = parameter.Name; DescriptionAttribute attribute = parameter.GetCustomAttribute(); @@ -175,12 +165,7 @@ protected virtual string GetParameterInfoToDescription(ParameterInfo parameter) return description; } - /// - /// Gets the name of a . - /// - /// - /// - protected virtual string GetMethodInfoToName(MethodInfo function) + private static string GetMethodInfoToName(MethodInfo function) { var sb = new StringBuilder(); sb.Append(function.Name); @@ -191,13 +176,7 @@ protected virtual string GetMethodInfoToName(MethodInfo function) return sb.ToString(); } - /// - /// Converts a CLR type to a JSON Utf8 type. - /// - /// - /// - /// - protected static ReadOnlySpan ClrToJsonTypeUtf8(Type clrType) + private static ReadOnlySpan ClrToJsonTypeUtf8(Type clrType) { if (clrType == typeof(double)) return "number"u8; @@ -209,13 +188,7 @@ protected static ReadOnlySpan ClrToJsonTypeUtf8(Type clrType) throw new NotImplementedException(); } - /// - /// Converts a CLR type to a JSON Utf16 type. - /// - /// - /// - /// - protected static string ClrToJsonTypeUtf16(Type clrType) + private static string ClrToJsonTypeUtf16(Type clrType) { if (clrType == typeof(double)) return "number"; @@ -227,7 +200,7 @@ protected static string ClrToJsonTypeUtf16(Type clrType) throw new NotImplementedException(); } - private BinaryData ParametersToJson(ParameterInfo[] parameters) + private static BinaryData ParametersToJson(ParameterInfo[] parameters) { if (parameters.Length == 0) return s_noparams; @@ -242,7 +215,7 @@ private BinaryData ParametersToJson(ParameterInfo[] parameters) { writer.WriteStartObject(parameter.Name!); writer.WriteString("type"u8, ChatTools.ClrToJsonTypeUtf8(parameter.ParameterType)); - writer.WriteString("description"u8, GetParameterInfoToDescription(parameter)); + writer.WriteString("description"u8, ChatTools.ParameterInfoToDescription(parameter)); writer.WriteEndObject(); if (!parameter.IsOptional || (parameter.HasDefaultValue && parameter.DefaultValue is not null)) required.Add(parameter.Name!); diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs b/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs index 1b24bcfbccb4c..a6c04b1d89577 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs @@ -27,8 +27,8 @@ public void Configuration() CloudMachineCommands.Execute(["-bicep"], (infrastructure) => { infrastructure.AddFeature(new KeyVaultFeature()); - infrastructure.AddFeature(new OpenAIModel("gpt-35-turbo", "0125")); - infrastructure.AddFeature(new OpenAIModel("text-embedding-ada-002", "2", AIModelKind.Embedding)); + infrastructure.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); + infrastructure.AddFeature(new OpenAIModelFeature("text-embedding-ada-002", "2", AIModelKind.Embedding)); }, exitProcessIfHandled: false); CloudMachineWorkspace cm = new(); @@ -72,7 +72,7 @@ public void OpenAI(string[] args) { if (CloudMachineCommands.Execute(args, (infrastructure) => { - infrastructure.AddFeature(new OpenAIModel("gpt-35-turbo", "0125")); + infrastructure.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); }, exitProcessIfHandled: false)) return; CloudMachineWorkspace cm = new(); diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs index b64ef5f0dc75a..def1791387c5b 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs @@ -13,13 +13,9 @@ public class AppServiceFeature : CloudMachineFeature { public AppServiceSkuDescription Sku { get; set; } - public AppServiceFeature(AppServiceSkuDescription? sku = default) + public AppServiceFeature() { - if (sku == default) - { - sku = new AppServiceSkuDescription { Tier = "Free", Name = "F1" }; - } - Sku = sku; + Sku = new AppServiceSkuDescription { Tier = "Free", Name = "F1" }; } protected override ProvisionableResource EmitCore(CloudMachineInfrastructure infrastructure) diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs index 3b9234a622cd5..055ce85209919 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs @@ -10,7 +10,7 @@ namespace Azure.CloudMachine.OpenAI; internal class OpenAIFeature : CloudMachineFeature { - private readonly List _models = []; + private readonly List _models = []; public OpenAIFeature() { } @@ -24,8 +24,8 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure clo Emitted = cognitiveServices; - OpenAIModel? previous = null; - foreach (OpenAIModel model in _models) + OpenAIModelFeature? previous = null; + foreach (OpenAIModelFeature model in _models) { model.Emit(cloudMachine); if (previous != null) @@ -38,7 +38,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure clo return cognitiveServices; } - internal void AddModel(OpenAIModel model) + internal void AddModel(OpenAIModelFeature model) { if (model.Account != null) { diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModel.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs similarity index 91% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModel.cs rename to sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs index f2d6272d61a55..776a3fff9ff66 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModel.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs @@ -2,7 +2,6 @@ // Licensed under the MIT License. using System; -using System.Collections.Generic; using System.Diagnostics; using Azure.Provisioning.CloudMachine; using Azure.Provisioning.CognitiveServices; @@ -10,9 +9,9 @@ namespace Azure.CloudMachine.OpenAI; -public class OpenAIModel : CloudMachineFeature +public class OpenAIModelFeature : CloudMachineFeature { - public OpenAIModel(string model, string modelVersion, AIModelKind kind = AIModelKind.Chat) { + public OpenAIModelFeature(string model, string modelVersion, AIModelKind kind = AIModelKind.Chat) { Kind = kind; Model = model; ModelVersion = modelVersion; @@ -37,7 +36,7 @@ private static OpenAIFeature GetOrCreateOpenAI(CloudMachineInfrastructure cm) public override void AddTo(CloudMachineInfrastructure cm) { - OpenAIFeature openAI = OpenAIModel.GetOrCreateOpenAI(cm); + OpenAIFeature openAI = OpenAIModelFeature.GetOrCreateOpenAI(cm); openAI.AddModel(this); } diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs index ac23c3c3d5198..aa45ed532596b 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs @@ -30,10 +30,10 @@ public void Emit(CloudMachineInfrastructure cm) protected internal Dictionary RequiredSystemRoles { get; } = []; - protected static T ValidateIsOfType(CloudMachineFeature resource) + protected static T EnsureEmits(CloudMachineFeature feature) { - if (resource.Emitted is T typed) + if (feature.Emitted is T typed) return typed; - throw new ArgumentException($"Expected resource of type {typeof(T).Name}, but got {resource.GetType().Name}"); + throw new ArgumentException($"Expected resource of type {typeof(T).Name}, but got {feature.GetType().Name}"); } } diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs index d2f21333eee43..9daf87b7ae7e1 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs @@ -10,20 +10,20 @@ namespace Azure.CloudMachine; -public class EventGridSystemTopicFeature(string name, CloudMachineFeature source) : CloudMachineFeature +public class EventGridSystemTopicFeature(string topicName, CloudMachineFeature source) : CloudMachineFeature { protected override ProvisionableResource EmitCore(CloudMachineInfrastructure infrastructure) { var topic = new SystemTopic("cm_eventgrid_topic", "2022-06-15") { TopicType = "Microsoft.Storage.StorageAccounts", - Source = ValidateIsOfType(source).Id, + Source = EnsureEmits(source).Id, Identity = new() { ManagedServiceIdentityType = ManagedServiceIdentityType.UserAssigned, UserAssignedIdentities = { { BicepFunction.Interpolate($"{infrastructure.Identity.Id}").Compile().ToString(), new UserAssignedIdentityDetails() } } }, - Name = name + Name = topicName }; infrastructure.AddResource(topic); diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/FeatureCollection.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/FeatureCollection.cs index 50cdbfadbbff8..204a3045c720f 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/FeatureCollection.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/FeatureCollection.cs @@ -20,6 +20,8 @@ public class FeatureCollection .SelectMany(d => d) .ToDictionary(kvp => kvp.Key, kvp => kvp.Value); + internal FeatureCollection() { } + public IEnumerable FindAll() where T : CloudMachineFeature { for (int i = 0; i < _count; i++) diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs index b60042fe63245..ad6b360fc57e8 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs @@ -15,7 +15,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf var subscription = new ServiceBusSubscription(name, "2021-11-01") { Name = name, - Parent = ValidateIsOfType(parent), + Parent = EnsureEmits(parent), IsClientAffine = false, LockDuration = TimeSpan.FromSeconds(30), RequiresSession = false, diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs index 707b316d9389f..e3abb6b9c3d3a 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs @@ -15,7 +15,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf var topic = new ServiceBusTopic(name, "2021-11-01") { Name = name, - Parent = ValidateIsOfType(parent), + Parent = EnsureEmits(parent), MaxMessageSizeInKilobytes = 256, DefaultMessageTimeToLive = TimeSpan.FromDays(14), RequiresDuplicateDetection = false, diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs index 386aac7790423..5da011f35ee34 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs @@ -14,10 +14,10 @@ public class SystemTopicEventSubscriptionFeature(string name, EventGridSystemTop { protected override ProvisionableResource EmitCore(CloudMachineInfrastructure infrastructure) { - ServiceBusNamespace serviceBusNamespace = ValidateIsOfType(parentNamespace); + ServiceBusNamespace serviceBusNamespace = EnsureEmits(parentNamespace); ServiceBusBuiltInRole role = ServiceBusBuiltInRole.AzureServiceBusDataSender; - var roleAssignment = new RoleAssignment($"cm_servicebus_{ValidateIsOfType(parent).Name.Value}_role") + var roleAssignment = new RoleAssignment($"cm_servicebus_{EnsureEmits(parent).Name.Value}_role") { Name = BicepFunction.CreateGuid(serviceBusNamespace.Id, infrastructure.Identity.Id, BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", role.ToString())), Scope = new IdentifierExpression(serviceBusNamespace.BicepIdentifier), @@ -26,7 +26,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf PrincipalId = infrastructure.Identity.PrincipalId, }; - SystemTopic systemTopic = ValidateIsOfType(parent); + SystemTopic systemTopic = EnsureEmits(parent); var subscription = new SystemTopicEventSubscription("cm_eventgrid_subscription_blob", "2022-06-15") { Name = name, @@ -40,7 +40,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf }, Destination = new ServiceBusTopicEventSubscriptionDestination { - ResourceId = ValidateIsOfType(destination).Id + ResourceId = EnsureEmits(destination).Id } }, Filter = new EventSubscriptionFilter diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/tests/CloudMachineTests.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/tests/CloudMachineTests.cs index be895b522e5c3..d0e96c617051d 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/tests/CloudMachineTests.cs +++ b/sdk/provisioning/Azure.Provisioning.CloudMachine/tests/CloudMachineTests.cs @@ -18,21 +18,13 @@ public class CloudMachineTests [Test] public void GenerateBicep() { - CloudMachineCommands.Execute(["-bicep"], (CloudMachineInfrastructure infrastructure) => - { - infrastructure.AddFeature(new KeyVaultFeature()); - infrastructure.AddFeature(new OpenAIModel("gpt-35-turbo", "0125")); - infrastructure.AddFeature(new OpenAIModel("text-embedding-ada-002", "2", AIModelKind.Embedding)); - infrastructure.AddFeature(new AppServiceFeature()); - }, exitProcessIfHandled: false); - CloudMachineInfrastructure infra = new("cm0c420d2f21084cd"); infra.AddFeature(new KeyVaultFeature()); - infra.AddFeature(new OpenAIModel("gpt-35-turbo", "0125")); - infra.AddFeature(new OpenAIModel("text-embedding-ada-002", "2", AIModelKind.Embedding)); + infra.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); + infra.AddFeature(new OpenAIModelFeature("text-embedding-ada-002", "2", AIModelKind.Embedding)); infra.AddFeature(new AppServiceFeature()); - string actualBicep = infra!.Build().Compile().FirstOrDefault().Value; + string actualBicep = infra.Build().Compile().FirstOrDefault().Value; string expectedBicep = File.ReadAllText(Path.Combine(TestContext.CurrentContext.TestDirectory, "Data", "GenerateBicep.bicep")).Replace("\r\n", Environment.NewLine); Assert.AreEqual(expectedBicep, actualBicep); } From 0f422d9166be50fcd4054ed736edd25d54f6160a Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Mon, 25 Nov 2024 16:21:52 -0800 Subject: [PATCH 03/24] moved CM provisioning projects to CM folder --- .../src/Azure.CloudMachine.All.csproj | 2 +- .../src/Core/LoggingPolicy.cs | 0 .../RestCLient/RestCallFailedException.cs | 8 + .../src/Core/RestCLient/RestClient.cs | 131 +++++++++++++++ .../src/Core/RestCLient/RestClientOptions.cs | 3 + .../TokenCredentialAuthenticationPolicy.cs | 28 +++- .../src/GlobalSuppressions.cs | 7 + .../tests/Azure.CloudMachine.Tests.csproj | 2 +- .../tests/CloudMachineTests.cs | 24 ++- .../Azure.CloudMachine.sln | 14 +- .../CHANGELOG.md | 0 .../Azure.Provisioning.CloudMachine/README.md | 0 .../Azure.Provisioning.CloudMachine.net8.0.cs | 0 ...rovisioning.CloudMachine.netstandard2.0.cs | 0 .../Azure.Provisioning.CloudMachine.csproj | 3 +- .../AzureSdkExtensions/AppServiceFeature.cs | 0 .../src/AzureSdkExtensions/KeyVaultFeature.cs | 0 .../src/AzureSdkExtensions/OpenAIFeature.cs | 0 .../AzureSdkExtensions/OpenAIModelFeature.cs | 0 .../src/CDKLevel3/AppConfigHelpers.cs | 103 ++++++++++++ .../src/CDKLevel3/Azd.cs | 82 ++++++++++ .../src/CDKLevel3/CloudMachineCommands.cs | 22 ++- .../src/CDKLevel3/CloudMachineFeature.cs | 0 .../CDKLevel3/CloudMachineInfrastructure.cs | 10 +- .../CDKLevel3/EventGridSystemTopicFeature.cs | 0 .../src/CDKLevel3/FeatureCollection.cs | 0 .../CDKLevel3/ServiceBusNamespaceFeature.cs | 0 .../ServiceBusSubscriptionFeature.cs | 0 .../src/CDKLevel3/ServiceBusTopicFeature.cs | 0 .../src/CDKLevel3/StorageFeature.cs | 0 .../SystemTopicEventSubscriptionFeature.cs | 0 .../src/GlobalSuppressions.cs | 0 .../src/Properties/AssemblyInfo.cs | 0 .../src/Tsp/TypeSpecWriter.cs | 0 ...ure.Provisioning.CloudMachine.Tests.csproj | 9 +- .../tests/CloudMachineTests.cs | 0 .../tests/Data/GenerateBicep.bicep | 0 .../tests/TdkTests.cs | 3 +- .../tests/appsettings.json | 0 .../src/CDKLevel3/AzdHelpers.cs | 153 ------------------ .../src/Core/RestCLient/RestClient.cs | 79 --------- 41 files changed, 417 insertions(+), 266 deletions(-) rename sdk/{provisioning/Azure.Provisioning.CloudMachine => cloudmachine/Azure.CloudMachine}/src/Core/LoggingPolicy.cs (100%) rename sdk/{provisioning/Azure.Provisioning.CloudMachine => cloudmachine/Azure.CloudMachine}/src/Core/RestCLient/RestCallFailedException.cs (56%) create mode 100644 sdk/cloudmachine/Azure.CloudMachine/src/Core/RestCLient/RestClient.cs rename sdk/{provisioning/Azure.Provisioning.CloudMachine => cloudmachine/Azure.CloudMachine}/src/Core/RestCLient/RestClientOptions.cs (75%) rename sdk/{provisioning/Azure.Provisioning.CloudMachine => cloudmachine/Azure.CloudMachine}/src/Core/RestCLient/TokenCredentialAuthenticationPolicy.cs (76%) create mode 100644 sdk/cloudmachine/Azure.CloudMachine/src/GlobalSuppressions.cs rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/Azure.CloudMachine.sln (75%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/CHANGELOG.md (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/README.md (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/Azure.Provisioning.CloudMachine.csproj (89%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs (100%) create mode 100644 sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/AppConfigHelpers.cs create mode 100644 sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/Azd.cs rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs (90%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs (96%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/CDKLevel3/FeatureCollection.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/GlobalSuppressions.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/Properties/AssemblyInfo.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/src/Tsp/TypeSpecWriter.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/tests/Azure.Provisioning.CloudMachine.Tests.csproj (53%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/tests/CloudMachineTests.cs (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/tests/Data/GenerateBicep.bicep (100%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/tests/TdkTests.cs (95%) rename sdk/{provisioning => cloudmachine}/Azure.Provisioning.CloudMachine/tests/appsettings.json (100%) delete mode 100644 sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/AzdHelpers.cs delete mode 100644 sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/RestClient.cs diff --git a/sdk/cloudmachine/Azure.CloudMachine.All/src/Azure.CloudMachine.All.csproj b/sdk/cloudmachine/Azure.CloudMachine.All/src/Azure.CloudMachine.All.csproj index 406fe8cbb1add..38cea841c6953 100644 --- a/sdk/cloudmachine/Azure.CloudMachine.All/src/Azure.CloudMachine.All.csproj +++ b/sdk/cloudmachine/Azure.CloudMachine.All/src/Azure.CloudMachine.All.csproj @@ -10,7 +10,7 @@ - + diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/LoggingPolicy.cs b/sdk/cloudmachine/Azure.CloudMachine/src/Core/LoggingPolicy.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/LoggingPolicy.cs rename to sdk/cloudmachine/Azure.CloudMachine/src/Core/LoggingPolicy.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/RestCallFailedException.cs b/sdk/cloudmachine/Azure.CloudMachine/src/Core/RestCLient/RestCallFailedException.cs similarity index 56% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/RestCallFailedException.cs rename to sdk/cloudmachine/Azure.CloudMachine/src/Core/RestCLient/RestCallFailedException.cs index 333a2934a308b..22d7a0c98243c 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/RestCallFailedException.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/Core/RestCLient/RestCallFailedException.cs @@ -6,8 +6,16 @@ namespace Azure { + /// + /// Exception thrown when a REST call fails. + /// public class RestCallFailedException : Exception { + /// + /// Initializes a new instance of the class. + /// + /// + /// public RestCallFailedException(string message, PipelineResponse response) : base(message, new System.ClientModel.ClientResultException(response)) { } } diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/Core/RestCLient/RestClient.cs b/sdk/cloudmachine/Azure.CloudMachine/src/Core/RestCLient/RestClient.cs new file mode 100644 index 0000000000000..648044c383817 --- /dev/null +++ b/sdk/cloudmachine/Azure.CloudMachine/src/Core/RestCLient/RestClient.cs @@ -0,0 +1,131 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +using System; +using System.ClientModel.Primitives; +using System.ClientModel; + +namespace Azure +{ + /// + /// A simple REST client that sends HTTP requests and receives responses. + /// + public class RestClient + { + private readonly ClientPipeline _pipeline; + + /// + /// A shared instance of the RestClient class. + /// + public static RestClient Shared { get; } = new RestClient(); + + /// + /// Initializes a new instance of the RestClient class. + /// + public RestClient() : this(default(RestClientOptions)) + { + } + + /// + /// Initializes a new instance of the RestClient class with the specified authentication policy. + /// + /// + public RestClient(PipelinePolicy auth) : this(CreateOptions(auth)) + { + } + + private static RestClientOptions CreateOptions(PipelinePolicy auth) + { + RestClientOptions options = new(); + options.AddPolicy(auth, PipelinePosition.PerTry); + return options; + } + + private RestClient(RestClientOptions options = default) + { + options ??= new RestClientOptions(); + _pipeline = ClientPipeline.Create(options); + } + + /// + /// Sends a GET request to the specified URI. + /// + /// + /// + /// + public PipelineResponse Get(string uri, RequestOptions options = default) + { + PipelineMessage message = Create("GET", new Uri(uri)); + return Send(message, options); + } + + /// + /// Sends a POST request to the specified URI. + /// + /// + /// + /// + /// + public PipelineResponse Post(string uri, BinaryContent content, RequestOptions options = default) + { + PipelineMessage message = Create("POST", new Uri(uri)); + message.Request.Content = content; + return Send(message, options); + } + + /// + /// Sends a PUT request to the specified URI. + /// + /// + /// + /// + /// + public PipelineResponse Put(string uri, BinaryContent content, RequestOptions options = default) + { + PipelineMessage message = Create("PUT", new Uri(uri)); + message.Request.Content = content; + return Send(message, options); + } + + /// + /// Sends a DELETE request to the specified URI. + /// + /// + /// + /// + /// + public PipelineResponse Patch(string uri, BinaryContent content, RequestOptions options = default) + { + PipelineMessage message = Create("PATCH", new Uri(uri)); + message.Request.Content = content; + return Send(message, options); + } + + /// + /// Sends a request with the specified message. + /// + /// + /// + /// + public PipelineResponse Send(PipelineMessage message, RequestOptions options = default) + { + if (options != default) message.Apply(options); + _pipeline.Send(message); + return message.Response!; + } + + /// + /// Creates a new PipelineMessage with the specified method and URI. + /// + /// + /// + /// + public PipelineMessage Create(string method, Uri uri) + { + PipelineMessage message = _pipeline.CreateMessage(); + message.Request.Method = method; + message.Request.Uri = uri; + return message; + } + } +} diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/RestClientOptions.cs b/sdk/cloudmachine/Azure.CloudMachine/src/Core/RestCLient/RestClientOptions.cs similarity index 75% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/RestClientOptions.cs rename to sdk/cloudmachine/Azure.CloudMachine/src/Core/RestCLient/RestClientOptions.cs index c5013124286cc..daa6df9758e40 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/RestClientOptions.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/Core/RestCLient/RestClientOptions.cs @@ -5,6 +5,9 @@ namespace Azure { + /// + /// Options for the REST client. + /// public class RestClientOptions : ClientPipelineOptions { } diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/TokenCredentialAuthenticationPolicy.cs b/sdk/cloudmachine/Azure.CloudMachine/src/Core/RestCLient/TokenCredentialAuthenticationPolicy.cs similarity index 76% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/TokenCredentialAuthenticationPolicy.cs rename to sdk/cloudmachine/Azure.CloudMachine/src/Core/RestCLient/TokenCredentialAuthenticationPolicy.cs index 1ec1f690bee8e..3cb1ae80ccd14 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/TokenCredentialAuthenticationPolicy.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/Core/RestCLient/TokenCredentialAuthenticationPolicy.cs @@ -11,13 +11,23 @@ namespace Azure.AI.OpenAI; -internal partial class TokenCredentialAuthenticationPolicy : PipelinePolicy +/// +/// TokenCredentialAuthenticationPolicy is a pipeline policy that authenticates requests using a TokenCredential. +/// +public partial class TokenCredentialAuthenticationPolicy : PipelinePolicy { private readonly TokenCredential _credential; private readonly string[] _scopes; private readonly TimeSpan _refreshOffset; private AccessToken? _currentToken; + /// + /// Creates a new TokenCredentialAuthenticationPolicy instance. + /// + /// + /// + /// + /// public TokenCredentialAuthenticationPolicy(TokenCredential credential, IEnumerable scopes, TimeSpan? refreshOffset = null) { if (credential is null) throw new ArgumentNullException(nameof(credential)); @@ -28,6 +38,13 @@ public TokenCredentialAuthenticationPolicy(TokenCredential credential, IEnumerab _refreshOffset = refreshOffset ?? s_defaultRefreshOffset; } + /// + /// executes the policy. + /// + /// + /// + /// + public override void Process(PipelineMessage message, IReadOnlyList pipeline, int currentIndex) { if (message?.Request is not null) @@ -46,6 +63,13 @@ public override void Process(PipelineMessage message, IReadOnlyList + /// executes the policy asynchronously. + /// + /// + /// + /// + /// public override async ValueTask ProcessAsync(PipelineMessage message, IReadOnlyList pipeline, int currentIndex) { if (message?.Request is not null) @@ -75,7 +99,7 @@ private bool IsTokenFresh() private TokenRequestContext CreateRequestContext(PipelineRequest request) { - if (request.Headers.TryGetValue("x-ms-client-request-id", out string? messageClientId)) + if (request.Headers.TryGetValue("x-ms-client-request-id", out string messageClientId)) { return new TokenRequestContext(_scopes, messageClientId); } diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/GlobalSuppressions.cs b/sdk/cloudmachine/Azure.CloudMachine/src/GlobalSuppressions.cs new file mode 100644 index 0000000000000..57ea0ca077129 --- /dev/null +++ b/sdk/cloudmachine/Azure.CloudMachine/src/GlobalSuppressions.cs @@ -0,0 +1,7 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +using System.Diagnostics.CodeAnalysis; + +[assembly: SuppressMessage("Usage", "AZC0007:DO provide a minimal constructor that takes only the parameters required to connect to the service.", Justification = "", Scope = "member", Target = "~M:Azure.RestClient.#ctor")] +[assembly: SuppressMessage("Usage", "AZC0007:DO provide a minimal constructor that takes only the parameters required to connect to the service.", Justification = "", Scope = "member", Target = "~M:Azure.RestClient.#ctor(System.ClientModel.Primitives.PipelinePolicy)")] diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/Azure.CloudMachine.Tests.csproj b/sdk/cloudmachine/Azure.CloudMachine/tests/Azure.CloudMachine.Tests.csproj index 7743315c7873e..72514fb231b79 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/tests/Azure.CloudMachine.Tests.csproj +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/Azure.CloudMachine.Tests.csproj @@ -8,7 +8,7 @@ - + diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs b/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs index a6c04b1d89577..fca6c0648a22f 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs @@ -15,12 +15,34 @@ using Microsoft.Extensions.Primitives; using System.Linq; using System.IO; -using Azure.Provisioning; namespace Azure.CloudMachine.Tests; public class CloudMachineTests { + [Test] + public void TwoAppsOpenAI() + { + CloudMachineInfrastructure infra = new(); + infra.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); + //if (args.Contains("-azd")) Azd.Init(infra); + + //var configuration = infra.Configuration; + + CloudMachineClient client = new(); + ChatClient chat = client.GetOpenAIChatClient(); + } + + //[Test] + //public void SingleAppOpenAI(string[] args) + //{ + // CloudMachineClient client = new(); + // client.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); + // if (args.Contains("-azd")) Azd.Init(client); + + // ChatClient chat = client.GetOpenAIChatClient(); + //} + [Test] public void Configuration() { diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/Azure.CloudMachine.sln b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/Azure.CloudMachine.sln similarity index 75% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/Azure.CloudMachine.sln rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/Azure.CloudMachine.sln index d2a38b3fc057e..0cc453bfe0e2a 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/Azure.CloudMachine.sln +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/Azure.CloudMachine.sln @@ -7,15 +7,13 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.Provisioning.CloudMac EndProject Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.Provisioning.CloudMachine.Tests", "tests\Azure.Provisioning.CloudMachine.Tests.csproj", "{46DCEF27-4157-4FB6-A283-B8484EC45665}" EndProject -Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.CloudMachine", "..\..\cloudmachine\Azure.CloudMachine\src\Azure.CloudMachine.csproj", "{AE9C0E9B-27D9-4220-98DA-0CB6A7936277}" +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.CloudMachine", "..\Azure.CloudMachine\src\Azure.CloudMachine.csproj", "{AE9C0E9B-27D9-4220-98DA-0CB6A7936277}" EndProject -Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.CloudMachine.Tests", "..\..\cloudmachine\Azure.CloudMachine\tests\Azure.CloudMachine.Tests.csproj", "{246D5F77-0151-40C5-8644-26A468A57CDC}" +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.CloudMachine.Tests", "..\Azure.CloudMachine\tests\Azure.CloudMachine.Tests.csproj", "{246D5F77-0151-40C5-8644-26A468A57CDC}" EndProject -Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.Provisioning.Deployment", "..\Azure.Provisioning.Deployment\src\Azure.Provisioning.Deployment.csproj", "{4562F8C1-9FE3-4B68-BBB2-D052B49C915A}" +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.CloudMachine.Web", "..\Azure.CloudMachine.Web\src\Azure.CloudMachine.Web.csproj", "{DD18D434-4665-43CA-97C0-4D63894EE23C}" EndProject -Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.CloudMachine.Web", "..\..\cloudmachine\Azure.CloudMachine.Web\src\Azure.CloudMachine.Web.csproj", "{DD18D434-4665-43CA-97C0-4D63894EE23C}" -EndProject -Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.CloudMachine.All", "..\..\cloudmachine\Azure.CloudMachine.All\src\Azure.CloudMachine.All.csproj", "{4E651FF9-EF43-4302-B77B-83C08D8FD117}" +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.CloudMachine.All", "..\Azure.CloudMachine.All\src\Azure.CloudMachine.All.csproj", "{4E651FF9-EF43-4302-B77B-83C08D8FD117}" EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution @@ -39,10 +37,6 @@ Global {246D5F77-0151-40C5-8644-26A468A57CDC}.Debug|Any CPU.Build.0 = Debug|Any CPU {246D5F77-0151-40C5-8644-26A468A57CDC}.Release|Any CPU.ActiveCfg = Release|Any CPU {246D5F77-0151-40C5-8644-26A468A57CDC}.Release|Any CPU.Build.0 = Release|Any CPU - {4562F8C1-9FE3-4B68-BBB2-D052B49C915A}.Debug|Any CPU.ActiveCfg = Debug|Any CPU - {4562F8C1-9FE3-4B68-BBB2-D052B49C915A}.Debug|Any CPU.Build.0 = Debug|Any CPU - {4562F8C1-9FE3-4B68-BBB2-D052B49C915A}.Release|Any CPU.ActiveCfg = Release|Any CPU - {4562F8C1-9FE3-4B68-BBB2-D052B49C915A}.Release|Any CPU.Build.0 = Release|Any CPU {DD18D434-4665-43CA-97C0-4D63894EE23C}.Debug|Any CPU.ActiveCfg = Debug|Any CPU {DD18D434-4665-43CA-97C0-4D63894EE23C}.Debug|Any CPU.Build.0 = Debug|Any CPU {DD18D434-4665-43CA-97C0-4D63894EE23C}.Release|Any CPU.ActiveCfg = Release|Any CPU diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/CHANGELOG.md b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/CHANGELOG.md similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/CHANGELOG.md rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/CHANGELOG.md diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/README.md b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/README.md similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/README.md rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/README.md diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Azure.Provisioning.CloudMachine.csproj b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Azure.Provisioning.CloudMachine.csproj similarity index 89% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/Azure.Provisioning.CloudMachine.csproj rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Azure.Provisioning.CloudMachine.csproj index ff39f04075f17..89a9b23ce3fc3 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Azure.Provisioning.CloudMachine.csproj +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Azure.Provisioning.CloudMachine.csproj @@ -5,12 +5,13 @@ 1.0.0-beta.1 $(RequiredTargetFrameworks) 12 - + enable CS1591;AZC0007 + diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/AppConfigHelpers.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/AppConfigHelpers.cs new file mode 100644 index 0000000000000..4c1e97e052baf --- /dev/null +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/AppConfigHelpers.cs @@ -0,0 +1,103 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +using System.IO; +using System.Text.Json.Nodes; +using System.Text.Json; +using System; + +namespace Azure.CloudMachine; + +internal static class AppConfigHelpers +{ + public static string GenerateCloudMachineId() + { + var guid = Guid.NewGuid(); + var guidString = guid.ToString("N"); + var cnId = "cm" + guidString.Substring(0, 15); // we can increase it to 20, but the template name cannot be that long + return cnId; + } + + internal static string ReadOrCreateCmid() + { + string appsettingsPath = Path.Combine(".", "appsettings.json"); + + if (TryReadCmId(appsettingsPath, out string? cmid)) return cmid!; + + cmid = GenerateCloudMachineId(); + + if (!File.Exists(appsettingsPath)) + { + CreateAppConfig(appsettingsPath, cmid); + } + else + { + AddCloudMachineId(appsettingsPath, cmid); + } + + return cmid; + } + + private static void CreateAppConfig(string appsettingsPath, string? cmid) + { + FileStream file = File.OpenWrite(appsettingsPath); + Utf8JsonWriter writer = new(file); + writer.WriteStartObject(); + writer.WritePropertyName("CloudMachine"u8); + writer.WriteStartObject(); + writer.WriteString("ID"u8, cmid); + writer.WriteEndObject(); + writer.WriteEndObject(); + writer.Flush(); + } + + private static bool TryReadCmId(string appsettings, out string? cmid) + { + using FileStream json = File.OpenRead(appsettings); + using JsonDocument jd = JsonDocument.Parse(json); + JsonElement je = jd.RootElement; + if (je.TryGetProperty("CloudMachine"u8, out JsonElement cm)) + { + if (!cm.TryGetProperty("ID"u8, out JsonElement id)) + { + throw new NotImplementedException(); + } + cmid = id.GetString(); + if (cmid == null) throw new InvalidOperationException($"CloudMachine:ID in {appsettings} is invalid"); + return true; + } + else + { + cmid = null; + return false; + } + } + + private static void AddCloudMachineId(string appsettings, string cmid) + { + FileStream json = File.OpenRead(appsettings); + + JsonNode? root = JsonNode.Parse(json); + json.Close(); + if (root is null || root is not JsonObject obj) + throw new InvalidOperationException("Existing appsettings.json is not a valid JSON object"); + + var cmProperties = new JsonObject(); + cmid = GenerateCloudMachineId(); + cmProperties.Add("ID", cmid); + obj.Add("CloudMachine", cmProperties); + + using FileStream file = File.OpenWrite(appsettings); + JsonWriterOptions writerOptions = new() + { + Indented = true, + }; + Utf8JsonWriter writer = new(file, writerOptions); + JsonSerializerOptions options = new() + { + WriteIndented = true, + }; + root.WriteTo(writer, options); + writer.Flush(); + } +} diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/Azd.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/Azd.cs new file mode 100644 index 0000000000000..094b3d101bde2 --- /dev/null +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/Azd.cs @@ -0,0 +1,82 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +using System.IO; +using Azure.Provisioning; +using Azure.Provisioning.Primitives; +using Azure.Provisioning.Resources; +using Azure.Provisioning.Expressions; + +namespace Azure.CloudMachine; + +/// +/// Azure Developer CLI helpers. +/// +#pragma warning disable AZC0012 // Avoid single word type names +public static class Azd +#pragma warning restore AZC0012 // Avoid single word type names +{ + private const string MainBicepName = "main"; + private const string ResourceGroupVersion = "2024-03-01"; + + public static void Init(CloudMachineInfrastructure infra, string? infraDirectory = default) + { + if (infraDirectory == default) infraDirectory = Path.Combine(".", "infra"); + + Directory.CreateDirectory(infraDirectory); + + infra.Build().Save(infraDirectory); + var cmid = AppConfigHelpers.ReadOrCreateCmid(); + + // main.bicep + var location = new ProvisioningParameter("location", typeof(string)); + var principalId = new ProvisioningParameter("principalId", typeof(string)); + + ResourceGroup rg = new(nameof(rg), ResourceGroupVersion) + { + Name = cmid, + Location = location + }; + + Infrastructure mainBicep = new("main") + { + TargetScope = DeploymentScope.Subscription + }; + ModuleImport import = new("cm", $"cm.bicep") + { + Name = "cm", + Scope = new IdentifierExpression(rg.BicepIdentifier) + }; + import.Parameters.Add(nameof(location), location); + import.Parameters.Add(nameof(principalId), principalId); + + mainBicep.Add(rg); + mainBicep.Add(import); + mainBicep.Add(location); + mainBicep.Add(principalId); + mainBicep.Build().Save(infraDirectory); + + WriteMainParametersFile(infraDirectory); + } + private static void WriteMainParametersFile(string infraDirectory) + { + File.WriteAllText(Path.Combine(infraDirectory, $"{MainBicepName}.parameters.json"), + """ + { + "$schema": "https://schema.management.azure.com/schemas/2019-04-01/deploymentParameters.json#", + "contentVersion": "1.0.0.0", + "parameters": { + "environmentName": { + "value": "${AZURE_ENV_NAME}" + }, + "location" : { + "value" : "${AZURE_LOCATION}" + }, + "principalId": { + "value": "${AZURE_PRINCIPAL_ID}" + } + } + } + """); + } +} diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs similarity index 90% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs index 6e9c50a770d3b..f388f9d6fca32 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs @@ -16,9 +16,10 @@ public class CloudMachineCommands { public static bool Execute(string[] args, Action? configure = default, bool exitProcessIfHandled = true) { - if (args.Length < 1) return false; + if (args.Length < 1) + return false; - string cmid = AzdHelpers.ReadOrCreateCmid(); + string cmid = AppConfigHelpers.ReadOrCreateCmid(); CloudMachineInfrastructure cmi = new(cmid); if (configure != default) { @@ -27,7 +28,7 @@ public static bool Execute(string[] args, Action? co if (args[0] == "-bicep") { - GenerateBicep(cmi); + Azd.Init(cmi); return Handled(exitProcessIfHandled); } @@ -48,7 +49,8 @@ public static bool Execute(string[] args, Action? co static bool Handled(bool exitProcessIfHandled) { - if (exitProcessIfHandled) Environment.Exit(0); + if (exitProcessIfHandled) + Environment.Exit(0); return true; } } @@ -86,7 +88,8 @@ private static void ListAzureOpenaAIModels(string cmid, string? option) { scenario = "chat"; } - if (caps.GetProperty("embeddings"u8).GetBoolean() == true) { + if (caps.GetProperty("embeddings"u8).GetBoolean() == true) + { scenario = "embeddings"; } if (caps.GetProperty("inference"u8).GetBoolean() == false) @@ -94,7 +97,8 @@ private static void ListAzureOpenaAIModels(string cmid, string? option) scenario = default; // inference==false means model cannot be deployed } - if (scenario == default) continue; + if (scenario == default) + continue; if (scenario != default) { @@ -129,10 +133,4 @@ private static void GenerateTsp(CloudMachineInfrastructure cmi) TypeSpecWriter.WriteServer(stream, endpoints); } } - - private static void GenerateBicep(CloudMachineInfrastructure cmi) - { - string infraDirectory = Path.Combine(".", "infra"); - AzdHelpers.Init(infraDirectory, cmi); - } } diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs similarity index 96% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs index d553857b90968..14ff5484d2fc2 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs @@ -3,6 +3,7 @@ using System; using System.Collections.Generic; +using Azure.Core; using Azure.Provisioning; using Azure.Provisioning.Authorization; using Azure.Provisioning.CloudMachine; @@ -17,6 +18,7 @@ public class CloudMachineInfrastructure internal const string SB_PRIVATE_TOPIC = "cm_servicebus_topic_private"; internal const string SB_PRIVATE_SUB = "cm_servicebus_subscription_private"; + private readonly List _connections = []; private readonly Infrastructure _infrastructure = new("cm"); private readonly List _resources = []; internal List Endpoints { get; } = []; @@ -40,8 +42,12 @@ public class CloudMachineInfrastructure ///// //public ProvisioningParameter PrincipalNameParameter => new BicepParameter("principalName", typeof(string)); - public CloudMachineInfrastructure(string cmId) + public CloudMachineInfrastructure(string? cmId = default) { + if (cmId == default) + { + cmId = AppConfigHelpers.ReadOrCreateCmid(); + } Id = cmId; // setup CM identity @@ -82,7 +88,7 @@ public void AddEndpoints() Endpoints.Add(endpointsType); } - public ProvisioningPlan Build(ProvisioningBuildOptions? context = null) + public ProvisioningPlan Build(ProvisioningBuildOptions? context = default) { context ??= new ProvisioningBuildOptions(); diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/FeatureCollection.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/FeatureCollection.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/FeatureCollection.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/FeatureCollection.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/GlobalSuppressions.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/GlobalSuppressions.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/GlobalSuppressions.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/GlobalSuppressions.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Properties/AssemblyInfo.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Properties/AssemblyInfo.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/Properties/AssemblyInfo.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Properties/AssemblyInfo.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Tsp/TypeSpecWriter.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tsp/TypeSpecWriter.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/src/Tsp/TypeSpecWriter.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tsp/TypeSpecWriter.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/tests/Azure.Provisioning.CloudMachine.Tests.csproj b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Azure.Provisioning.CloudMachine.Tests.csproj similarity index 53% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/tests/Azure.Provisioning.CloudMachine.Tests.csproj rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Azure.Provisioning.CloudMachine.Tests.csproj index 0a891c2b442f0..44a160bb6cac6 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/tests/Azure.Provisioning.CloudMachine.Tests.csproj +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Azure.Provisioning.CloudMachine.Tests.csproj @@ -1,11 +1,16 @@  + 12 + - + + + + - + diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/tests/CloudMachineTests.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/CloudMachineTests.cs similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/tests/CloudMachineTests.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/CloudMachineTests.cs diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/tests/Data/GenerateBicep.bicep b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Data/GenerateBicep.bicep similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/tests/Data/GenerateBicep.bicep rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Data/GenerateBicep.bicep diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/tests/TdkTests.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/TdkTests.cs similarity index 95% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/tests/TdkTests.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/TdkTests.cs index bc0d0c4959551..8f9bd3641bbee 100644 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/tests/TdkTests.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/TdkTests.cs @@ -7,7 +7,6 @@ using System.ClientModel.TypeSpec; using System.IO; using System.Threading.Tasks; -using Microsoft.AspNetCore.Http; using NUnit.Framework; namespace Azure.CloudMachine.Tests; @@ -59,7 +58,7 @@ @get @route("send") Send(@query message: string) : { internal interface IAssistantService { [HttpPut] - Task UploadAsync(HttpRequest document); + Task UploadAsync(byte[] document); Task SendAsync([FromQuery] string message); } diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/tests/appsettings.json b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/appsettings.json similarity index 100% rename from sdk/provisioning/Azure.Provisioning.CloudMachine/tests/appsettings.json rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/appsettings.json diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/AzdHelpers.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/AzdHelpers.cs deleted file mode 100644 index 950368ffe3f25..0000000000000 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/CDKLevel3/AzdHelpers.cs +++ /dev/null @@ -1,153 +0,0 @@ -// Copyright (c) Microsoft Corporation. All rights reserved. -// Licensed under the MIT License. - -using System.IO; -using Azure.Provisioning; -using Azure.Provisioning.CloudMachine; -using Azure.Provisioning.Primitives; -using Azure.Provisioning.Resources; -using Azure.Provisioning.Expressions; -using System.Text.Json.Nodes; -using System.Text.Json; -using System; - -namespace Azure.CloudMachine; - -internal static class AzdHelpers -{ - private const string MainBicepName = "main"; - private const string ResourceGroupVersion = "2024-03-01"; - - internal static void Init(string infraDirectory, CloudMachineInfrastructure cmi) - { - Directory.CreateDirectory(infraDirectory); - - cmi.Build().Save(infraDirectory); - var cmid = ReadOrCreateCmid(); - - // main.bicep - var location = new ProvisioningParameter("location", typeof(string)); - var principalId = new ProvisioningParameter("principalId", typeof(string)); - - ResourceGroup rg = new(nameof(rg), ResourceGroupVersion) - { - Name = cmid, - Location = location - }; - - Infrastructure mainBicep = new("main") - { - TargetScope = DeploymentScope.Subscription - }; - ModuleImport import = new("cm", $"cm.bicep") - { - Name = "cm", - Scope = new IdentifierExpression(rg.BicepIdentifier) - }; - import.Parameters.Add(nameof(location), location); - import.Parameters.Add(nameof(principalId), principalId); - - mainBicep.Add(rg); - mainBicep.Add(import); - mainBicep.Add(location); - mainBicep.Add(principalId); - mainBicep.Build().Save(infraDirectory); - - WriteMainParametersFile(infraDirectory); - } - - private static void WriteMainParametersFile(string infraDirectory) - { - File.WriteAllText(Path.Combine(infraDirectory, $"{MainBicepName}.parameters.json"), - """ - { - "$schema": "https://schema.management.azure.com/schemas/2019-04-01/deploymentParameters.json#", - "contentVersion": "1.0.0.0", - "parameters": { - "environmentName": { - "value": "${AZURE_ENV_NAME}" - }, - "location" : { - "value" : "${AZURE_LOCATION}" - }, - "principalId": { - "value": "${AZURE_PRINCIPAL_ID}" - } - } - } - """); - } - - internal static string ReadOrCreateCmid() - { - string appsettings = Path.Combine(".", "appsettings.json"); - - string? cmid; - if (!File.Exists(appsettings)) - { - cmid = GenerateCloudMachineId(); - - using FileStream file = File.OpenWrite(appsettings); - Utf8JsonWriter writer = new(file); - writer.WriteStartObject(); - writer.WritePropertyName("CloudMachine"u8); - writer.WriteStartObject(); - writer.WriteString("ID"u8, cmid); - writer.WriteEndObject(); - writer.WriteEndObject(); - writer.Flush(); - return cmid; - } - - using FileStream json = File.OpenRead(appsettings); - using JsonDocument jd = JsonDocument.Parse(json); - JsonElement je = jd.RootElement; - // attempt to read CM configuration from existing configuration file - if (je.TryGetProperty("CloudMachine"u8, out JsonElement cm)) - { - if (!cm.TryGetProperty("ID"u8, out JsonElement id)) - { - throw new NotImplementedException(); - } - cmid = id.GetString(); - if (cmid == null) - throw new NotImplementedException(); - return cmid; - } - else - { // add CM configuration to existing file - json.Seek(0, SeekOrigin.Begin); - JsonNode? root = JsonNode.Parse(json); - json.Close(); - if (root is null || root is not JsonObject obj) throw new InvalidOperationException("Existing appsettings.json is not a valid JSON object"); - - var cmProperties = new JsonObject(); - cmid = GenerateCloudMachineId(); - cmProperties.Add("ID", cmid); - obj.Add("CloudMachine", cmProperties); - - using FileStream file = File.OpenWrite(appsettings); - JsonWriterOptions writerOptions = new() - { - Indented = true, - }; - Utf8JsonWriter writer = new(file, writerOptions); - JsonSerializerOptions options = new() - { - WriteIndented = true, - }; - root.WriteTo(writer, options); - writer.Flush(); - } - - return cmid; - - static string GenerateCloudMachineId() - { - var guid = Guid.NewGuid(); - var guidString = guid.ToString("N"); - var cnId = "cm" + guidString.Substring(0, 15); // we can increase it to 20, but the template name cannot be that long - return cnId; - } - } -} diff --git a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/RestClient.cs b/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/RestClient.cs deleted file mode 100644 index 9a7343f0b41b4..0000000000000 --- a/sdk/provisioning/Azure.Provisioning.CloudMachine/src/Core/RestCLient/RestClient.cs +++ /dev/null @@ -1,79 +0,0 @@ -// Copyright (c) Microsoft Corporation. All rights reserved. -// Licensed under the MIT License. - -using System; -using System.ClientModel.Primitives; -using System.ClientModel; - -namespace Azure -{ - public class RestClient - { - private readonly ClientPipeline _pipeline; - - public static RestClient Shared { get; } = new RestClient(); - - public RestClient() : this(default(RestClientOptions)) - { - } - - public RestClient(PipelinePolicy auth) : this(CreateOptions(auth)) - { - } - - private static RestClientOptions CreateOptions(PipelinePolicy auth) - { - RestClientOptions options = new(); - options.AddPolicy(auth, PipelinePosition.PerTry); - return options; - } - - private RestClient(RestClientOptions? options = default) - { - options ??= new RestClientOptions(); - _pipeline = ClientPipeline.Create(options); - } - - public PipelineResponse Get(string uri, RequestOptions? options = default) - { - PipelineMessage message = Create("GET", new Uri(uri)); - return Send(message, options); - } - - public PipelineResponse Post(string uri, BinaryContent content, RequestOptions? options = default) - { - PipelineMessage message = Create("POST", new Uri(uri)); - message.Request.Content = content; - return Send(message, options); - } - - public PipelineResponse Put(string uri, BinaryContent content, RequestOptions? options = default) - { - PipelineMessage message = Create("PUT", new Uri(uri)); - message.Request.Content = content; - return Send(message, options); - } - - public PipelineResponse Patch(string uri, BinaryContent content, RequestOptions? options = default) - { - PipelineMessage message = Create("PATCH", new Uri(uri)); - message.Request.Content = content; - return Send(message, options); - } - - public PipelineResponse Send(PipelineMessage message, RequestOptions? options = default) - { - if (options != default) message.Apply(options); - _pipeline.Send(message); - return message.Response!; - } - - public PipelineMessage Create(string method, Uri uri) - { - PipelineMessage message = _pipeline.CreateMessage(); - message.Request.Method = method; - message.Request.Uri = uri; - return message; - } - } -} From 75eea3519ed1b1d1e360f8f16877c9ad95eea670 Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Mon, 25 Nov 2024 17:09:09 -0800 Subject: [PATCH 04/24] added the ability to provide extension connections --- .../src/CloudMachineClient.cs | 6 ++-- .../src/CloudMachineWorkspace.cs | 36 ++++++++++++++++--- .../tests/CloudMachineTests.cs | 4 +-- .../src/AzureSdkExtensions/OpenAIFeature.cs | 2 +- .../AzureSdkExtensions/OpenAIModelFeature.cs | 19 +++++++++- .../src/CDKLevel3/AppConfigHelpers.cs | 5 +++ .../CDKLevel3/CloudMachineInfrastructure.cs | 16 +++------ 7 files changed, 66 insertions(+), 22 deletions(-) diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineClient.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineClient.cs index 1d223f539be13..cde11e3c01b9e 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineClient.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineClient.cs @@ -1,6 +1,7 @@ // Copyright (c) Microsoft Corporation. All rights reserved. // Licensed under the MIT License. +using System.Collections.Generic; using Azure.Core; using Microsoft.Extensions.Configuration; @@ -25,9 +26,10 @@ protected CloudMachineClient() /// /// The token credential. /// The configuration settings. - public CloudMachineClient(TokenCredential credential = default, IConfiguration configuration = default) + /// + public CloudMachineClient(TokenCredential credential = default, IConfiguration configuration = default, Dictionary connections = default) #pragma warning restore AZC0007 // DO provide a minimal constructor that takes only the parameters required to connect to the service. - : base(credential, configuration) + : base(credential, configuration, connections) { Messaging = new MessagingServices(this); Storage = new StorageServices(this); diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs index d24fb88fda98e..e63b4fb9237e7 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs @@ -2,6 +2,7 @@ // Licensed under the MIT License. using System; +using System.Collections.Generic; using System.ComponentModel; using System.Diagnostics.CodeAnalysis; using System.IO; @@ -19,6 +20,7 @@ namespace Azure.CloudMachine; public class CloudMachineWorkspace : ClientWorkspace { private TokenCredential Credential { get; } + private Dictionary Connections { get; } /// /// The cloud machine ID. @@ -31,9 +33,10 @@ public class CloudMachineWorkspace : ClientWorkspace /// /// /// + /// /// [SuppressMessage("Usage", "AZC0007:DO provide a minimal constructor that takes only the parameters required to connect to the service.", Justification = "")] - public CloudMachineWorkspace(TokenCredential credential = default, IConfiguration configuration = default) + public CloudMachineWorkspace(TokenCredential credential = default, IConfiguration configuration = default, Dictionary connections = default) { if (credential != default) { @@ -49,6 +52,15 @@ public CloudMachineWorkspace(TokenCredential credential = default, IConfiguratio }; } + if (connections != default) + { + Connections = connections; + } + else + { + Connections = new(); + } + Id = configuration switch { null => ReadOrCreateCmid(), @@ -78,10 +90,24 @@ public override ClientConnectionOptions GetConnectionOptions(Type clientType, st "Azure.Messaging.ServiceBus.ServiceBusProcessor" => new ClientConnectionOptions("cm_servicebus_default_topic/cm_servicebus_subscription_default"), "Azure.Messaging.ServiceBus.ServiceBusProcessor$private" => new ClientConnectionOptions("cm_servicebus_topic_private/cm_servicebus_subscription_private"), "Azure.Storage.Blobs.BlobContainerClient" => new ClientConnectionOptions(new($"https://{Id}.blob.core.windows.net/{instanceId ?? "default"}"), Credential), - "Azure.AI.OpenAI.AzureOpenAIClient" => new ClientConnectionOptions(new($"https://{Id}.openai.azure.com"), Credential), - "OpenAI.Chat.ChatClient" => new ClientConnectionOptions($"{Id}_chat"), - "OpenAI.Embeddings.EmbeddingClient" => new ClientConnectionOptions($"{Id}_embedding"), - _ => throw new Exception($"unknown client {clientId}"), + _ => GetExtensionConnection(clientId) + }; + + ClientConnectionOptions GetExtensionConnection(string clientId) + { + if (Connections.TryGetValue(clientId, out object connection)) + { + if (connection is Uri uri) + { + return new ClientConnectionOptions(uri, Credential); + } + if (connection is string str) + { + return new ClientConnectionOptions(str); + } + throw new NotImplementedException(); + } + throw new Exception($"unknown client {clientId}"); }; } diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs b/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs index fca6c0648a22f..302da7a4757d7 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs @@ -27,9 +27,9 @@ public void TwoAppsOpenAI() infra.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); //if (args.Contains("-azd")) Azd.Init(infra); - //var configuration = infra.Configuration; + var connections = infra.Connections; - CloudMachineClient client = new(); + CloudMachineClient client = new(connections: connections); ChatClient chat = client.GetOpenAIChatClient(); } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs index 055ce85209919..68251847f06b6 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs @@ -17,7 +17,7 @@ public OpenAIFeature() protected override ProvisionableResource EmitCore(CloudMachineInfrastructure cloudMachine) { - CognitiveServicesAccount cognitiveServices = OpenAIFeature.CreateOpenAIAccount(cloudMachine); + CognitiveServicesAccount cognitiveServices = CreateOpenAIAccount(cloudMachine); cloudMachine.AddResource(cognitiveServices); RequiredSystemRoles.Add(cognitiveServices, [(CognitiveServicesBuiltInRole.GetBuiltInRoleName(CognitiveServicesBuiltInRole.CognitiveServicesOpenAIContributor) ,CognitiveServicesBuiltInRole.CognitiveServicesOpenAIContributor.ToString())]); diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs index 776a3fff9ff66..44f1c73332f96 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs @@ -3,8 +3,10 @@ using System; using System.Diagnostics; +using Azure.Core; using Azure.Provisioning.CloudMachine; using Azure.Provisioning.CognitiveServices; +using Azure.Provisioning.Expressions; using Azure.Provisioning.Primitives; namespace Azure.CloudMachine.OpenAI; @@ -31,13 +33,28 @@ private static OpenAIFeature GetOrCreateOpenAI(CloudMachineInfrastructure cm) } var openAI = new OpenAIFeature(); cm.AddFeature(openAI); + cm.Connections.Add("Azure.AI.OpenAI.AzureOpenAIClient", new Uri($"https://{cm.Id}.openai.azure.com")); return openAI; } public override void AddTo(CloudMachineInfrastructure cm) { - OpenAIFeature openAI = OpenAIModelFeature.GetOrCreateOpenAI(cm); + OpenAIFeature openAI = GetOrCreateOpenAI(cm); + openAI.AddModel(this); + + // add connections + switch (Kind) + { + case AIModelKind.Chat: + cm.Connections.Add("OpenAI.Chat.ChatClient", $"{cm.Id}_chat"); + break; + case AIModelKind.Embedding: + cm.Connections.Add("OpenAI.Embeddings.EmbeddingClient", $"{cm.Id}_embedding"); + break; + default: + throw new NotImplementedException(); + } } protected override ProvisionableResource EmitCore(CloudMachineInfrastructure cm) diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/AppConfigHelpers.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/AppConfigHelpers.cs index 4c1e97e052baf..aca1a23358140 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/AppConfigHelpers.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/AppConfigHelpers.cs @@ -53,6 +53,11 @@ private static void CreateAppConfig(string appsettingsPath, string? cmid) private static bool TryReadCmId(string appsettings, out string? cmid) { + if (!File.Exists(appsettings)) + { + cmid = null; + return false; + } using FileStream json = File.OpenRead(appsettings); using JsonDocument jd = JsonDocument.Parse(json); JsonElement je = jd.RootElement; diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs index 14ff5484d2fc2..74155f1b441f8 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs @@ -3,6 +3,7 @@ using System; using System.Collections.Generic; +using System.ComponentModel; using Azure.Core; using Azure.Provisioning; using Azure.Provisioning.Authorization; @@ -18,7 +19,7 @@ public class CloudMachineInfrastructure internal const string SB_PRIVATE_TOPIC = "cm_servicebus_topic_private"; internal const string SB_PRIVATE_SUB = "cm_servicebus_subscription_private"; - private readonly List _connections = []; + private readonly Dictionary _connections = []; private readonly Infrastructure _infrastructure = new("cm"); private readonly List _resources = []; internal List Endpoints { get; } = []; @@ -32,16 +33,6 @@ public class CloudMachineInfrastructure /// public ProvisioningParameter PrincipalIdParameter => new("principalId", typeof(string)); - ///// - ///// The common principalType parameter. - ///// - //public ProvisioningParameter PrincipalTypeParameter => new BicepParameter("principalType", typeof(string)); - - ///// - ///// The common principalName parameter. - ///// - //public ProvisioningParameter PrincipalNameParameter => new BicepParameter("principalName", typeof(string)); - public CloudMachineInfrastructure(string? cmId = default) { if (cmId == default) @@ -88,6 +79,9 @@ public void AddEndpoints() Endpoints.Add(endpointsType); } + [EditorBrowsable(EditorBrowsableState.Never)] + public Dictionary Connections => _connections; + public ProvisioningPlan Build(ProvisioningBuildOptions? context = default) { context ??= new ProvisioningBuildOptions(); From 7023becb83c9a2621c9e3e72573917abb4fcec93 Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Tue, 26 Nov 2024 09:07:30 -0800 Subject: [PATCH 05/24] cleaned up extensible connections --- .../AzureSdkExtensions/AppServiceFeature.cs | 4 +- .../src/AzureSdkExtensions/KeyVaultFeature.cs | 2 +- .../src/AzureSdkExtensions/OpenAIFeature.cs | 7 +- .../AzureSdkExtensions/OpenAIModelFeature.cs | 19 +++-- .../src/CDKLevel3/CloudMachineFeature.cs | 15 +++- .../CDKLevel3/CloudMachineInfrastructure.cs | 80 ++++++------------- .../CDKLevel3/EventGridSystemTopicFeature.cs | 2 +- .../src/CDKLevel3/RoleResolver.cs | 52 ++++++++++++ .../CDKLevel3/ServiceBusNamespaceFeature.cs | 4 +- .../ServiceBusSubscriptionFeature.cs | 2 +- .../src/CDKLevel3/ServiceBusTopicFeature.cs | 2 +- .../src/CDKLevel3/StorageFeature.cs | 8 +- .../SystemTopicEventSubscriptionFeature.cs | 4 +- 13 files changed, 116 insertions(+), 85 deletions(-) create mode 100644 sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/RoleResolver.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs index def1791387c5b..8d8ba98c634a3 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs @@ -27,7 +27,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf Sku = Sku, Kind = "app" }; - infrastructure.AddResource(hostingPlan); + infrastructure.AddConstruct(hostingPlan); WebSite appService = new("cm_website") { @@ -59,7 +59,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf ] } }; - infrastructure.AddResource(appService); + infrastructure.AddConstruct(appService); return appService; } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs index 0d8f55bdc6161..b924f6fe0f582 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs @@ -42,7 +42,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf ] }, }; - infrastructure.AddResource(keyVaultResource); + infrastructure.AddConstruct(keyVaultResource); RequiredSystemRoles.Add(keyVaultResource, [(KeyVaultBuiltInRole.GetBuiltInRoleName(KeyVaultBuiltInRole.KeyVaultAdministrator), KeyVaultBuiltInRole.KeyVaultAdministrator.ToString())]); return keyVaultResource; diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs index 68251847f06b6..62d4cfe19a120 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs @@ -15,10 +15,15 @@ internal class OpenAIFeature : CloudMachineFeature public OpenAIFeature() { } + protected internal override void AddTo(CloudMachineInfrastructure cm) + { + cm.Features.Add(this); + cm.Connections.Add("Azure.AI.OpenAI.AzureOpenAIClient", new Uri($"https://{cm.Id}.openai.azure.com")); + } protected override ProvisionableResource EmitCore(CloudMachineInfrastructure cloudMachine) { CognitiveServicesAccount cognitiveServices = CreateOpenAIAccount(cloudMachine); - cloudMachine.AddResource(cognitiveServices); + cloudMachine.AddConstruct(cognitiveServices); RequiredSystemRoles.Add(cognitiveServices, [(CognitiveServicesBuiltInRole.GetBuiltInRoleName(CognitiveServicesBuiltInRole.CognitiveServicesOpenAIContributor) ,CognitiveServicesBuiltInRole.CognitiveServicesOpenAIContributor.ToString())]); diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs index 44f1c73332f96..33b3fb9a9703d 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs @@ -3,10 +3,9 @@ using System; using System.Diagnostics; -using Azure.Core; +using System.Linq; using Azure.Provisioning.CloudMachine; using Azure.Provisioning.CognitiveServices; -using Azure.Provisioning.Expressions; using Azure.Provisioning.Primitives; namespace Azure.CloudMachine.OpenAI; @@ -27,17 +26,17 @@ public OpenAIModelFeature(string model, string modelVersion, AIModelKind kind = private static OpenAIFeature GetOrCreateOpenAI(CloudMachineInfrastructure cm) { - foreach (OpenAIFeature feature in cm.Features.FindAll()) - { - return feature; - } - var openAI = new OpenAIFeature(); + // TODO: is it OK that we return the first one? + OpenAIFeature? openAI = cm.Features.FindAll().FirstOrDefault(); + if (openAI != default) return openAI; + + openAI = new OpenAIFeature(); cm.AddFeature(openAI); - cm.Connections.Add("Azure.AI.OpenAI.AzureOpenAIClient", new Uri($"https://{cm.Id}.openai.azure.com")); + return openAI; } - public override void AddTo(CloudMachineInfrastructure cm) + protected internal override void AddTo(CloudMachineInfrastructure cm) { OpenAIFeature openAI = GetOrCreateOpenAI(cm); @@ -95,7 +94,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure cm) } }; - cm.AddResource(deployment); + cm.AddConstruct(deployment); return deployment; } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs index aa45ed532596b..57c435394af23 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs @@ -11,11 +11,9 @@ namespace Azure.Provisioning.CloudMachine; public abstract class CloudMachineFeature { - [EditorBrowsable(EditorBrowsableState.Never)] - public virtual void AddTo(CloudMachineInfrastructure cm) => cm.Features.Add(this); + protected internal virtual void AddTo(CloudMachineInfrastructure cm) => cm.Features.Add(this); - [EditorBrowsable(EditorBrowsableState.Never)] - public void Emit(CloudMachineInfrastructure cm) + internal void Emit(CloudMachineInfrastructure cm) { if (Emitted != null) return; @@ -36,4 +34,13 @@ protected static T EnsureEmits(CloudMachineFeature feature) return typed; throw new ArgumentException($"Expected resource of type {typeof(T).Name}, but got {feature.GetType().Name}"); } + + [EditorBrowsable(EditorBrowsableState.Never)] + public override string ToString() => base.ToString()!; + + [EditorBrowsable(EditorBrowsableState.Never)] + public override int GetHashCode() => base.GetHashCode(); + + [EditorBrowsable(EditorBrowsableState.Never)] + public override bool Equals(object? obj) => base.Equals(obj); } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs index 74155f1b441f8..b53f506994f57 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs @@ -4,7 +4,6 @@ using System; using System.Collections.Generic; using System.ComponentModel; -using Azure.Core; using Azure.Provisioning; using Azure.Provisioning.Authorization; using Azure.Provisioning.CloudMachine; @@ -19,18 +18,23 @@ public class CloudMachineInfrastructure internal const string SB_PRIVATE_TOPIC = "cm_servicebus_topic_private"; internal const string SB_PRIVATE_SUB = "cm_servicebus_subscription_private"; - private readonly Dictionary _connections = []; private readonly Infrastructure _infrastructure = new("cm"); private readonly List _resources = []; - internal List Endpoints { get; } = []; + internal List Endpoints { get; } = []; public FeatureCollection Features { get; } = new(); + + [EditorBrowsable(EditorBrowsableState.Never)] + public Dictionary Connections { get; } = []; + + [EditorBrowsable(EditorBrowsableState.Never)] public UserAssignedIdentity Identity { get; private set; } public string Id { get; private set; } /// /// The common principalId parameter. /// + [EditorBrowsable(EditorBrowsableState.Never)] public ProvisioningParameter PrincipalIdParameter => new("principalId", typeof(string)); public CloudMachineInfrastructure(string? cmId = default) @@ -62,14 +66,7 @@ public CloudMachineInfrastructure(string? cmId = default) Features.Add(new SystemTopicEventSubscriptionFeature("cm_eventgrid_subscription_blob", systemTopic, sbTopicPrivate, sbNamespace)); } - public void AddResource(NamedProvisionableConstruct resource) - { - _resources.Add(resource); - } - public void AddFeature(CloudMachineFeature feature) - { - feature.AddTo(this); - } + public void AddFeature(CloudMachineFeature feature) => feature.AddTo(this); public void AddEndpoints() { @@ -80,17 +77,16 @@ public void AddEndpoints() } [EditorBrowsable(EditorBrowsableState.Never)] - public Dictionary Connections => _connections; + public void AddConstruct(NamedProvisionableConstruct resource) + { + _resources.Add(resource); + } + [EditorBrowsable(EditorBrowsableState.Never)] public ProvisioningPlan Build(ProvisioningBuildOptions? context = default) { - context ??= new ProvisioningBuildOptions(); - Features.Emit(this); - // This must occur after the features have been emitted. - context.InfrastructureResolvers.Add(new RoleResolver(Features.RoleAnnotations, [Identity], [PrincipalIdParameter])); - // Always add a default location parameter. // azd assumes there will be a location parameter for every module. // The Infrastructure location resolver will resolve unset Location properties to this parameter. @@ -113,46 +109,18 @@ public ProvisioningPlan Build(ProvisioningBuildOptions? context = default) _infrastructure.Add(resource); } + context ??= new ProvisioningBuildOptions(); + // This must occur after the features have been emitted. + context.InfrastructureResolvers.Add(new RoleResolver(Features.RoleAnnotations, [Identity], [PrincipalIdParameter])); return _infrastructure.Build(context); } - internal class RoleResolver(Dictionary annotations, IEnumerable managedIdentities, IEnumerable> userPrincipals) : InfrastructureResolver - { - public override IEnumerable ResolveResources(IEnumerable resources, ProvisioningBuildOptions options) - { - foreach (Provisionable provisionable in base.ResolveResources(resources, options)) - { - yield return provisionable; - if (annotations.TryGetValue(provisionable, out (string RoleName, string RoleId)[]? roles) && provisionable is ProvisionableResource resource && roles is not null) - { - foreach ((string RoleName, string RoleId) in roles) - { - foreach (BicepValue userPrincipal in userPrincipals) - { - yield return new RoleAssignment($"{resource.BicepIdentifier}_{userPrincipal.Value.ToString().Replace('-', '_')}_{RoleName}") - { - Name = BicepFunction.CreateGuid(resource.BicepIdentifier, userPrincipal, BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", RoleId)), - Scope = new IdentifierExpression(resource.BicepIdentifier), - PrincipalType = RoleManagementPrincipalType.User, - RoleDefinitionId = BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", RoleId), - PrincipalId = userPrincipal - }; - } - - foreach (UserAssignedIdentity identity in managedIdentities) - { - yield return new RoleAssignment($"{resource.BicepIdentifier}_{identity.BicepIdentifier}_{RoleName}") - { - Name = BicepFunction.CreateGuid(resource.BicepIdentifier, identity.Id, BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", RoleId)), - Scope = new IdentifierExpression(resource.BicepIdentifier), - PrincipalType = RoleManagementPrincipalType.ServicePrincipal, - RoleDefinitionId = BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", RoleId), - PrincipalId = identity.PrincipalId - }; - } - } - } - } - } - } + [EditorBrowsable(EditorBrowsableState.Never)] + public override string ToString() => Id; + + [EditorBrowsable(EditorBrowsableState.Never)] + public override int GetHashCode() => base.GetHashCode(); + + [EditorBrowsable(EditorBrowsableState.Never)] + public override bool Equals(object? obj) => base.Equals(obj); } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs index 9daf87b7ae7e1..d1689c72ed3d1 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs @@ -26,7 +26,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf Name = topicName }; - infrastructure.AddResource(topic); + infrastructure.AddConstruct(topic); Emitted = topic; return topic; diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/RoleResolver.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/RoleResolver.cs new file mode 100644 index 0000000000000..f0958b7ca60f8 --- /dev/null +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/RoleResolver.cs @@ -0,0 +1,52 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +using System; +using System.Collections.Generic; +using Azure.Provisioning; +using Azure.Provisioning.Authorization; +using Azure.Provisioning.Expressions; +using Azure.Provisioning.Primitives; +using Azure.Provisioning.Roles; + +namespace Azure.CloudMachine; + +internal class RoleResolver(Dictionary annotations, IEnumerable managedIdentities, IEnumerable> userPrincipals) : InfrastructureResolver +{ + public override IEnumerable ResolveResources(IEnumerable resources, ProvisioningBuildOptions options) + { + foreach (Provisionable provisionable in base.ResolveResources(resources, options)) + { + yield return provisionable; + if (annotations.TryGetValue(provisionable, out (string RoleName, string RoleId)[]? roles) && provisionable is ProvisionableResource resource && roles is not null) + { + foreach ((string RoleName, string RoleId) in roles) + { + foreach (BicepValue userPrincipal in userPrincipals) + { + yield return new RoleAssignment($"{resource.BicepIdentifier}_{userPrincipal.Value.ToString().Replace('-', '_')}_{RoleName}") + { + Name = BicepFunction.CreateGuid(resource.BicepIdentifier, userPrincipal, BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", RoleId)), + Scope = new IdentifierExpression(resource.BicepIdentifier), + PrincipalType = RoleManagementPrincipalType.User, + RoleDefinitionId = BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", RoleId), + PrincipalId = userPrincipal + }; + } + + foreach (UserAssignedIdentity identity in managedIdentities) + { + yield return new RoleAssignment($"{resource.BicepIdentifier}_{identity.BicepIdentifier}_{RoleName}") + { + Name = BicepFunction.CreateGuid(resource.BicepIdentifier, identity.Id, BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", RoleId)), + Scope = new IdentifierExpression(resource.BicepIdentifier), + PrincipalType = RoleManagementPrincipalType.ServicePrincipal, + RoleDefinitionId = BicepFunction.GetSubscriptionResourceId("Microsoft.Authorization/roleDefinitions", RoleId), + PrincipalId = identity.PrincipalId + }; + } + } + } + } + } +} diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs index 395094679aab9..a52ad529a51e2 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs @@ -20,8 +20,8 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf }, Name = name, }; - infrastructure.AddResource(_serviceBusNamespace); - infrastructure.AddResource( + infrastructure.AddConstruct(_serviceBusNamespace); + infrastructure.AddConstruct( new ServiceBusNamespaceAuthorizationRule("cm_servicebus_auth_rule", "2021-11-01") { Parent = _serviceBusNamespace, diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs index ad6b360fc57e8..e2692714ca89e 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs @@ -27,7 +27,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf Status = ServiceBusMessagingEntityStatus.Active }; - infrastructure.AddResource(subscription); + infrastructure.AddConstruct(subscription); Emitted = subscription; return subscription; } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs index e3abb6b9c3d3a..c4fa34d5ba27e 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs @@ -24,7 +24,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf Status = ServiceBusMessagingEntityStatus.Active }; - infrastructure.AddResource(topic); + infrastructure.AddConstruct(topic); Emitted = topic; return topic; } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs index 45411ace4e854..fa108bc8ffc73 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs @@ -44,17 +44,17 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf UserAssignedIdentities = { { BicepFunction.Interpolate($"{infrastructure.Identity.Id}").Compile().ToString(), new UserAssignedIdentityDetails() } } } }; - infrastructure.AddResource(_storage); + infrastructure.AddConstruct(_storage); var _blobs = new BlobService("cm_storage_blobs") { Parent = _storage, }; - infrastructure.AddResource(_blobs); + infrastructure.AddConstruct(_blobs); foreach (var containerName in _containerNames) { - infrastructure.AddResource( + infrastructure.AddConstruct( new BlobContainer("cm_storage_blobs_container_" + containerName, "2023-01-01") { Parent = _blobs, @@ -71,7 +71,7 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf ]); // Placeholders for now. - infrastructure.AddResource(new ProvisioningOutput($"storage_name", typeof(string)) { Value = _storage.Name }); + infrastructure.AddConstruct(new ProvisioningOutput($"storage_name", typeof(string)) { Value = _storage.Name }); Emitted = _storage; return _storage; diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs index 5da011f35ee34..13970df4ec2fa 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs @@ -62,8 +62,8 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf }; subscription.DependsOn.Add(roleAssignment); - infrastructure.AddResource(subscription); - infrastructure.AddResource(roleAssignment); + infrastructure.AddConstruct(subscription); + infrastructure.AddConstruct(roleAssignment); Emitted = subscription; return subscription; From c89abeeecf71c5f1393c6f89461c4ee77e212db8 Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Tue, 26 Nov 2024 09:30:30 -0800 Subject: [PATCH 06/24] made connection collection strongly typed --- .../src/ClientConnectionOptions.cs | 32 +++++------ .../Azure.CloudMachine/src/ClientWorkspace.cs | 15 ++++++ .../src/CloudMachineClient.cs | 2 +- .../src/CloudMachineWorkspace.cs | 53 ++++++++----------- .../src/CoreServices/MessagingServices.cs | 6 +-- .../src/CoreServices/StorageServices.cs | 4 +- .../src/extensions/AzureOpenAIExtensions.cs | 8 +-- .../src/extensions/KeyVaultExtensions.cs | 4 +- .../src/AzureSdkExtensions/OpenAIFeature.cs | 5 +- .../AzureSdkExtensions/OpenAIModelFeature.cs | 5 +- .../CDKLevel3/CloudMachineInfrastructure.cs | 3 +- 11 files changed, 73 insertions(+), 64 deletions(-) diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/ClientConnectionOptions.cs b/sdk/cloudmachine/Azure.CloudMachine/src/ClientConnectionOptions.cs index f2afd6cba5076..03b1dfbae339a 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/ClientConnectionOptions.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/ClientConnectionOptions.cs @@ -2,6 +2,7 @@ // Licensed under the MIT License. using System; +using System.ComponentModel; namespace Azure.Core; @@ -10,6 +11,12 @@ namespace Azure.Core; /// public readonly struct ClientConnectionOptions { + /// + /// Do not use this constructor. It is only for the JSON serializer. + /// + [EditorBrowsable(EditorBrowsableState.Never)] + public ClientConnectionOptions() { } + /// /// Initializes a new instance of the struct with the specified endpoint and API key. /// @@ -19,19 +26,17 @@ public ClientConnectionOptions(Uri endpoint, string apiKey) { Endpoint = endpoint; ApiKeyCredential = apiKey; - ConnectionKind = ClientConnectionKind.ApiKey; + Authentication = ClientAuthenticationMethod.ApiKey; } /// /// Initializes a new instance of the struct with the specified endpoint and token credential. /// /// The endpoint URI. - /// The token credential. - public ClientConnectionOptions(Uri endpoint, TokenCredential credential) + public ClientConnectionOptions(Uri endpoint) { Endpoint = endpoint; - TokenCredential = credential; - ConnectionKind = ClientConnectionKind.EntraId; + Authentication = ClientAuthenticationMethod.EntraId; } /// @@ -40,14 +45,14 @@ public ClientConnectionOptions(Uri endpoint, TokenCredential credential) /// The subclient ID. public ClientConnectionOptions(string subclientId) { - Id = subclientId; - ConnectionKind = ClientConnectionKind.OutOfBand; + SubclientId = subclientId; + Authentication = ClientAuthenticationMethod.Subclient; } /// /// Gets the kind of connection used by the client. /// - public ClientConnectionKind ConnectionKind { get; } + public ClientAuthenticationMethod Authentication { get; } /// /// Gets the endpoint URI. @@ -57,23 +62,18 @@ public ClientConnectionOptions(string subclientId) /// /// Gets the subclient ID. /// - public string Id { get; } + public string SubclientId { get; } /// /// Gets the API key credential. /// public string ApiKeyCredential { get; } - - /// - /// Gets the token credential. - /// - public TokenCredential TokenCredential { get; } } /// /// Specifies the kind of connection used by the client. /// -public enum ClientConnectionKind +public enum ClientAuthenticationMethod { /// /// Represents a connection using Entra ID. @@ -88,5 +88,5 @@ public enum ClientConnectionKind /// /// Represents a connection using an out-of-band method. /// - OutOfBand + Subclient } diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/ClientWorkspace.cs b/sdk/cloudmachine/Azure.CloudMachine/src/ClientWorkspace.cs index e441cbd53a58b..45d83b4bf04a0 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/ClientWorkspace.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/ClientWorkspace.cs @@ -12,6 +12,15 @@ namespace Azure.Core; /// public abstract class ClientWorkspace { + /// + /// Initializes a new instance of the class with the specified token credential. + /// + /// + protected ClientWorkspace(TokenCredential credential) + { + Credential = credential; + } + /// /// Retrieves the connection options for a specified client type and instance ID. /// @@ -20,6 +29,12 @@ public abstract class ClientWorkspace /// The connection options for the specified client type and instance ID. public abstract ClientConnectionOptions GetConnectionOptions(Type clientType, string instanceId = default); + /// + /// Gets the token credential. + /// + [EditorBrowsable(EditorBrowsableState.Never)] + public TokenCredential Credential { get; } + /// /// Gets the cache of subclients. /// diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineClient.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineClient.cs index cde11e3c01b9e..0b5c7e08a3b2b 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineClient.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineClient.cs @@ -27,7 +27,7 @@ protected CloudMachineClient() /// The token credential. /// The configuration settings. /// - public CloudMachineClient(TokenCredential credential = default, IConfiguration configuration = default, Dictionary connections = default) + public CloudMachineClient(TokenCredential credential = default, IConfiguration configuration = default, Dictionary connections = default) #pragma warning restore AZC0007 // DO provide a minimal constructor that takes only the parameters required to connect to the service. : base(credential, configuration, connections) { diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs index e63b4fb9237e7..85e3bcbf2b766 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs @@ -19,8 +19,7 @@ namespace Azure.CloudMachine; /// public class CloudMachineWorkspace : ClientWorkspace { - private TokenCredential Credential { get; } - private Dictionary Connections { get; } + private Dictionary Connections { get; } /// /// The cloud machine ID. @@ -36,22 +35,9 @@ public class CloudMachineWorkspace : ClientWorkspace /// /// [SuppressMessage("Usage", "AZC0007:DO provide a minimal constructor that takes only the parameters required to connect to the service.", Justification = "")] - public CloudMachineWorkspace(TokenCredential credential = default, IConfiguration configuration = default, Dictionary connections = default) + public CloudMachineWorkspace(TokenCredential credential = default, IConfiguration configuration = default, Dictionary connections = default) + : base(BuildCredentail(credential)) { - if (credential != default) - { - Credential = credential; - } - else - { - // This environment variable is set by the CloudMachine App Service feature during provisioning. - Credential = Environment.GetEnvironmentVariable("CLOUDMACHINE_MANAGED_IDENTITY_CLIENT_ID") switch - { - string clientId when !string.IsNullOrEmpty(clientId) => new ManagedIdentityCredential(clientId), - _ => new ChainedTokenCredential(new AzureCliCredential(), new AzureDeveloperCliCredential()) - }; - } - if (connections != default) { Connections = connections; @@ -68,6 +54,20 @@ public CloudMachineWorkspace(TokenCredential credential = default, IConfiguratio }; } + private static TokenCredential BuildCredentail(TokenCredential credential) + { + if (credential == default) + { + // This environment variable is set by the CloudMachine App Service feature during provisioning. + credential = Environment.GetEnvironmentVariable("CLOUDMACHINE_MANAGED_IDENTITY_CLIENT_ID") switch + { + string clientId when !string.IsNullOrEmpty(clientId) => new ManagedIdentityCredential(clientId), + _ => new ChainedTokenCredential(new AzureCliCredential(), new AzureDeveloperCliCredential()) + }; + } + + return credential; + } /// /// Retrieves the connection options for a specified client type and instance ID. /// @@ -84,29 +84,18 @@ public override ClientConnectionOptions GetConnectionOptions(Type clientType, st return clientId switch { - "Azure.Security.KeyVault.Secrets.SecretClient" => new ClientConnectionOptions(new($"https://{Id}.vault.azure.net/"), Credential), - "Azure.Messaging.ServiceBus.ServiceBusClient" => new ClientConnectionOptions(new($"https://{Id}.servicebus.windows.net"), Credential), + "Azure.Storage.Blobs.BlobContainerClient" => new ClientConnectionOptions(new Uri($"https://{Id}.blob.core.windows.net/{instanceId ?? "default"}")), + "Azure.Security.KeyVault.Secrets.SecretClient" => new ClientConnectionOptions(new Uri($"https://{Id}.vault.azure.net/")), + "Azure.Messaging.ServiceBus.ServiceBusClient" => new ClientConnectionOptions(new Uri($"https://{Id}.servicebus.windows.net")), "Azure.Messaging.ServiceBus.ServiceBusSender" => new ClientConnectionOptions(instanceId ?? "cm_servicebus_default_topic"), "Azure.Messaging.ServiceBus.ServiceBusProcessor" => new ClientConnectionOptions("cm_servicebus_default_topic/cm_servicebus_subscription_default"), "Azure.Messaging.ServiceBus.ServiceBusProcessor$private" => new ClientConnectionOptions("cm_servicebus_topic_private/cm_servicebus_subscription_private"), - "Azure.Storage.Blobs.BlobContainerClient" => new ClientConnectionOptions(new($"https://{Id}.blob.core.windows.net/{instanceId ?? "default"}"), Credential), _ => GetExtensionConnection(clientId) }; ClientConnectionOptions GetExtensionConnection(string clientId) { - if (Connections.TryGetValue(clientId, out object connection)) - { - if (connection is Uri uri) - { - return new ClientConnectionOptions(uri, Credential); - } - if (connection is string str) - { - return new ClientConnectionOptions(str); - } - throw new NotImplementedException(); - } + if (Connections.TryGetValue(clientId, out ClientConnectionOptions connection)) return connection; throw new Exception($"unknown client {clientId}"); }; } diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/MessagingServices.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/MessagingServices.cs index 64bebd2efe61f..d7425c3b8464c 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/MessagingServices.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/MessagingServices.cs @@ -88,13 +88,13 @@ private ServiceBusSender CreateSender() ServiceBusClient client = GetServiceBusClient(); ClientConnectionOptions connection = _cm.GetConnectionOptions(typeof(ServiceBusSender), default); - ServiceBusSender sender = client.CreateSender(connection.Id); + ServiceBusSender sender = client.CreateSender(connection.SubclientId); return sender; } private ServiceBusClient CreateClient() { ClientConnectionOptions connection = _cm.GetConnectionOptions(typeof(ServiceBusClient), default); - ServiceBusClient client = new(connection.Endpoint!.AbsoluteUri, connection.TokenCredential); + ServiceBusClient client = new(connection.Endpoint!.AbsoluteUri, _cm.Credential); return client; } private ServiceBusProcessor CreateProcessor(string id) @@ -102,7 +102,7 @@ private ServiceBusProcessor CreateProcessor(string id) ServiceBusClient client = GetServiceBusClient(); ClientConnectionOptions connection = _cm.GetConnectionOptions(typeof(ServiceBusProcessor), id); - string[] topicAndSubscription = connection.Id.Split('/'); + string[] topicAndSubscription = connection.SubclientId.Split('/'); ServiceBusProcessor processor = client.CreateProcessor(topicAndSubscription[0], topicAndSubscription[1], new() { MaxConcurrentCalls = 5 }); processor.ProcessErrorAsync += (args) => throw new Exception("error processing event", args.Exception); return processor; diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs index 75a7992bef1f5..dce2bbedcefa6 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs @@ -30,7 +30,7 @@ private BlobContainerClient GetDefaultContainer() BlobContainerClient container = _cm.Subclients.Get(() => { ClientConnectionOptions connection = cm.GetConnectionOptions(typeof(BlobContainerClient), default); - BlobContainerClient container = new(connection.Endpoint, connection.TokenCredential); + BlobContainerClient container = new(connection.Endpoint, cm.Credential); return container; }); return container; @@ -43,7 +43,7 @@ private BlobContainerClient GetContainer(string containerName) BlobContainerClient container = cm.Subclients.Get(() => { ClientConnectionOptions connection = cm.GetConnectionOptions(typeof(BlobContainerClient), containerName); - BlobContainerClient container = new(connection.Endpoint, connection.TokenCredential); + BlobContainerClient container = new(connection.Endpoint, cm.Credential); return container; }); return container; diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/AzureOpenAIExtensions.cs b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/AzureOpenAIExtensions.cs index 7068300733b39..91b5f742f7d2a 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/AzureOpenAIExtensions.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/AzureOpenAIExtensions.cs @@ -83,9 +83,9 @@ public static string AsText(this ChatMessageContent content) private static AzureOpenAIClient CreateAzureOpenAIClient(this ClientWorkspace workspace) { ClientConnectionOptions connection = workspace.GetConnectionOptions(typeof(AzureOpenAIClient)); - if (connection.ConnectionKind == ClientConnectionKind.EntraId) + if (connection.Authentication == ClientAuthenticationMethod.EntraId) { - return new(connection.Endpoint, connection.TokenCredential); + return new(connection.Endpoint, workspace.Credential); } else { @@ -96,14 +96,14 @@ private static AzureOpenAIClient CreateAzureOpenAIClient(this ClientWorkspace wo private static ChatClient CreateChatClient(this ClientWorkspace workspace, AzureOpenAIClient client) { ClientConnectionOptions connection = workspace.GetConnectionOptions(typeof(ChatClient)); - ChatClient chat = client.GetChatClient(connection.Id); + ChatClient chat = client.GetChatClient(connection.SubclientId); return chat; } private static EmbeddingClient CreateEmbeddingsClient(this ClientWorkspace workspace, AzureOpenAIClient client) { ClientConnectionOptions connection = workspace.GetConnectionOptions(typeof(EmbeddingClient)); - EmbeddingClient embeddings = client.GetEmbeddingClient(connection.Id); + EmbeddingClient embeddings = client.GetEmbeddingClient(connection.SubclientId); return embeddings; } diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/KeyVaultExtensions.cs b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/KeyVaultExtensions.cs index ad6405da41f8c..a3abef2d2d57c 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/KeyVaultExtensions.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/KeyVaultExtensions.cs @@ -21,9 +21,9 @@ public static class KeyVaultExtensions public static SecretClient GetKeyVaultSecretsClient(this ClientWorkspace workspace) { ClientConnectionOptions connection = workspace.GetConnectionOptions(typeof(SecretClient)); - if (connection.ConnectionKind == ClientConnectionKind.EntraId) + if (connection.Authentication == ClientAuthenticationMethod.EntraId) { - return new(connection.Endpoint, connection.TokenCredential); + return new(connection.Endpoint, workspace.Credential); } throw new Exception("API key not supported"); } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs index 62d4cfe19a120..59815c8476736 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs @@ -2,6 +2,7 @@ // Licensed under the MIT License. using System; using System.Collections.Generic; +using Azure.Core; using Azure.Provisioning.CloudMachine; using Azure.Provisioning.CognitiveServices; using Azure.Provisioning.Primitives; @@ -18,7 +19,9 @@ public OpenAIFeature() protected internal override void AddTo(CloudMachineInfrastructure cm) { cm.Features.Add(this); - cm.Connections.Add("Azure.AI.OpenAI.AzureOpenAIClient", new Uri($"https://{cm.Id}.openai.azure.com")); + + ClientConnectionOptions connection = new(new Uri($"https://{cm.Id}.openai.azure.com")); + cm.Connections.Add("Azure.AI.OpenAI.AzureOpenAIClient", connection); } protected override ProvisionableResource EmitCore(CloudMachineInfrastructure cloudMachine) { diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs index 33b3fb9a9703d..7303f539fa53b 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs @@ -4,6 +4,7 @@ using System; using System.Diagnostics; using System.Linq; +using Azure.Core; using Azure.Provisioning.CloudMachine; using Azure.Provisioning.CognitiveServices; using Azure.Provisioning.Primitives; @@ -46,10 +47,10 @@ protected internal override void AddTo(CloudMachineInfrastructure cm) switch (Kind) { case AIModelKind.Chat: - cm.Connections.Add("OpenAI.Chat.ChatClient", $"{cm.Id}_chat"); + cm.Connections.Add("OpenAI.Chat.ChatClient", new($"{cm.Id}_chat")); break; case AIModelKind.Embedding: - cm.Connections.Add("OpenAI.Embeddings.EmbeddingClient", $"{cm.Id}_embedding"); + cm.Connections.Add("OpenAI.Embeddings.EmbeddingClient", new($"{cm.Id}_embedding")); break; default: throw new NotImplementedException(); diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs index b53f506994f57..dce138bb4929d 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs @@ -4,6 +4,7 @@ using System; using System.Collections.Generic; using System.ComponentModel; +using Azure.Core; using Azure.Provisioning; using Azure.Provisioning.Authorization; using Azure.Provisioning.CloudMachine; @@ -25,7 +26,7 @@ public class CloudMachineInfrastructure public FeatureCollection Features { get; } = new(); [EditorBrowsable(EditorBrowsableState.Never)] - public Dictionary Connections { get; } = []; + public Dictionary Connections { get; } = []; [EditorBrowsable(EditorBrowsableState.Never)] public UserAssignedIdentity Identity { get; private set; } From 41b17f43bbc70bc443cbf78b53258e8074d2b78b Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Tue, 26 Nov 2024 13:58:10 -0800 Subject: [PATCH 07/24] completly removed hardcoded connections --- ...nnectionOptions.cs => ClientConnection.cs} | 50 +++++++++---------- .../Azure.CloudMachine/src/ClientWorkspace.cs | 6 +-- .../src/CloudMachineClient.cs | 2 +- .../src/CloudMachineWorkspace.cs | 38 +++----------- .../src/ConnectionCollection.cs | 28 +++++++++++ .../src/CoreServices/MessagingServices.cs | 20 ++++---- .../src/CoreServices/StorageServices.cs | 10 ++-- .../src/extensions/AzureOpenAIExtensions.cs | 15 +++--- .../src/extensions/KeyVaultExtensions.cs | 4 +- .../src/AzureSdkExtensions/KeyVaultFeature.cs | 10 +++- .../src/AzureSdkExtensions/OpenAIFeature.cs | 4 +- .../AzureSdkExtensions/OpenAIModelFeature.cs | 4 +- .../CDKLevel3/CloudMachineInfrastructure.cs | 29 ++++++----- .../CDKLevel3/ServiceBusNamespaceFeature.cs | 7 +++ .../ServiceBusSubscriptionFeature.cs | 11 ++++ .../src/CDKLevel3/ServiceBusTopicFeature.cs | 28 +++++++++-- .../src/CDKLevel3/StorageFeature.cs | 8 +++ 17 files changed, 165 insertions(+), 109 deletions(-) rename sdk/cloudmachine/Azure.CloudMachine/src/{ClientConnectionOptions.cs => ClientConnection.cs} (60%) create mode 100644 sdk/cloudmachine/Azure.CloudMachine/src/ConnectionCollection.cs diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/ClientConnectionOptions.cs b/sdk/cloudmachine/Azure.CloudMachine/src/ClientConnection.cs similarity index 60% rename from sdk/cloudmachine/Azure.CloudMachine/src/ClientConnectionOptions.cs rename to sdk/cloudmachine/Azure.CloudMachine/src/ClientConnection.cs index 03b1dfbae339a..97f49d4ae10df 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/ClientConnectionOptions.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/ClientConnection.cs @@ -9,44 +9,38 @@ namespace Azure.Core; /// /// Represents the connection options for a client. /// -public readonly struct ClientConnectionOptions +public readonly struct ClientConnection { /// /// Do not use this constructor. It is only for the JSON serializer. /// [EditorBrowsable(EditorBrowsableState.Never)] - public ClientConnectionOptions() { } + public ClientConnection() { } /// - /// Initializes a new instance of the struct with the specified endpoint and API key. + /// Initializes a new instance of the struct with the specified endpoint and API key. /// - /// The endpoint URI. + /// + /// /// The API key credential. - public ClientConnectionOptions(Uri endpoint, string apiKey) + public ClientConnection(string id, string locator, string apiKey) { - Endpoint = endpoint; + Id = id; + Locator = locator; ApiKeyCredential = apiKey; Authentication = ClientAuthenticationMethod.ApiKey; } /// - /// Initializes a new instance of the struct with the specified endpoint and token credential. + /// Initializes a new instance of the struct with the specified subclient ID. /// - /// The endpoint URI. - public ClientConnectionOptions(Uri endpoint) + /// + /// The subclient ID. + /// + public ClientConnection(string id, string locator, ClientAuthenticationMethod auth = ClientAuthenticationMethod.EntraId) { - Endpoint = endpoint; - Authentication = ClientAuthenticationMethod.EntraId; - } - - /// - /// Initializes a new instance of the struct with the specified subclient ID. - /// - /// The subclient ID. - public ClientConnectionOptions(string subclientId) - { - SubclientId = subclientId; - Authentication = ClientAuthenticationMethod.Subclient; + Locator = locator; + Authentication = auth; } /// @@ -55,19 +49,25 @@ public ClientConnectionOptions(string subclientId) public ClientAuthenticationMethod Authentication { get; } /// - /// Gets the endpoint URI. + /// Gets the key. /// - public Uri Endpoint { get; } + public string Id { get; } /// - /// Gets the subclient ID. + /// This is either URI or name, or something like that. /// - public string SubclientId { get; } + public string Locator { get; } /// /// Gets the API key credential. /// public string ApiKeyCredential { get; } + + /// + /// Converts the connection to a URI. + /// + /// + public Uri ToUri() => new(Locator); } /// diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/ClientWorkspace.cs b/sdk/cloudmachine/Azure.CloudMachine/src/ClientWorkspace.cs index 45d83b4bf04a0..879e4099206c6 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/ClientWorkspace.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/ClientWorkspace.cs @@ -1,7 +1,6 @@ // Copyright (c) Microsoft Corporation. All rights reserved. // Licensed under the MIT License. -using System; using System.ComponentModel; namespace Azure.Core; @@ -24,10 +23,9 @@ protected ClientWorkspace(TokenCredential credential) /// /// Retrieves the connection options for a specified client type and instance ID. /// - /// The type of the client. - /// The instance ID of the client. + /// /// The connection options for the specified client type and instance ID. - public abstract ClientConnectionOptions GetConnectionOptions(Type clientType, string instanceId = default); + public abstract ClientConnection GetConnectionOptions(string connectionId); /// /// Gets the token credential. diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineClient.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineClient.cs index 0b5c7e08a3b2b..2d31947889926 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineClient.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineClient.cs @@ -27,7 +27,7 @@ protected CloudMachineClient() /// The token credential. /// The configuration settings. /// - public CloudMachineClient(TokenCredential credential = default, IConfiguration configuration = default, Dictionary connections = default) + public CloudMachineClient(TokenCredential credential = default, IConfiguration configuration = default, IEnumerable connections = default) #pragma warning restore AZC0007 // DO provide a minimal constructor that takes only the parameters required to connect to the service. : base(credential, configuration, connections) { diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs index 85e3bcbf2b766..34cabcff4f722 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs @@ -3,6 +3,7 @@ using System; using System.Collections.Generic; +using System.Collections.ObjectModel; using System.ComponentModel; using System.Diagnostics.CodeAnalysis; using System.IO; @@ -19,7 +20,7 @@ namespace Azure.CloudMachine; /// public class CloudMachineWorkspace : ClientWorkspace { - private Dictionary Connections { get; } + private ConnectionCollection Connections { get; } = []; /// /// The cloud machine ID. @@ -35,16 +36,12 @@ public class CloudMachineWorkspace : ClientWorkspace /// /// [SuppressMessage("Usage", "AZC0007:DO provide a minimal constructor that takes only the parameters required to connect to the service.", Justification = "")] - public CloudMachineWorkspace(TokenCredential credential = default, IConfiguration configuration = default, Dictionary connections = default) + public CloudMachineWorkspace(TokenCredential credential = default, IConfiguration configuration = default, IEnumerable connections = default) : base(BuildCredentail(credential)) { if (connections != default) { - Connections = connections; - } - else - { - Connections = new(); + Connections.AddRange(connections); } Id = configuration switch @@ -68,36 +65,17 @@ private static TokenCredential BuildCredentail(TokenCredential credential) return credential; } + /// /// Retrieves the connection options for a specified client type and instance ID. /// - /// - /// + /// /// /// [EditorBrowsable(EditorBrowsableState.Never)] - public override ClientConnectionOptions GetConnectionOptions(Type clientType, string instanceId) + public override ClientConnection GetConnectionOptions(string connectionId) { - string clientId = clientType.FullName; - if (instanceId != null && instanceId.StartsWith("$")) - clientId = $"{clientType.FullName}{instanceId}"; - - return clientId switch - { - "Azure.Storage.Blobs.BlobContainerClient" => new ClientConnectionOptions(new Uri($"https://{Id}.blob.core.windows.net/{instanceId ?? "default"}")), - "Azure.Security.KeyVault.Secrets.SecretClient" => new ClientConnectionOptions(new Uri($"https://{Id}.vault.azure.net/")), - "Azure.Messaging.ServiceBus.ServiceBusClient" => new ClientConnectionOptions(new Uri($"https://{Id}.servicebus.windows.net")), - "Azure.Messaging.ServiceBus.ServiceBusSender" => new ClientConnectionOptions(instanceId ?? "cm_servicebus_default_topic"), - "Azure.Messaging.ServiceBus.ServiceBusProcessor" => new ClientConnectionOptions("cm_servicebus_default_topic/cm_servicebus_subscription_default"), - "Azure.Messaging.ServiceBus.ServiceBusProcessor$private" => new ClientConnectionOptions("cm_servicebus_topic_private/cm_servicebus_subscription_private"), - _ => GetExtensionConnection(clientId) - }; - - ClientConnectionOptions GetExtensionConnection(string clientId) - { - if (Connections.TryGetValue(clientId, out ClientConnectionOptions connection)) return connection; - throw new Exception($"unknown client {clientId}"); - }; + return Connections[connectionId]; } /// diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/ConnectionCollection.cs b/sdk/cloudmachine/Azure.CloudMachine/src/ConnectionCollection.cs new file mode 100644 index 0000000000000..a4f8bb4a8d3c7 --- /dev/null +++ b/sdk/cloudmachine/Azure.CloudMachine/src/ConnectionCollection.cs @@ -0,0 +1,28 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +using System; +using System.Collections.Generic; +using System.Collections.ObjectModel; +using Azure.Core; + +namespace Azure.CloudMachine; + +/// +/// Represents the connection options for a client. +/// +public class ConnectionCollection : KeyedCollection +{ + /// + /// Initializes a new instance of the class. + /// + /// + /// + protected override string GetKeyForItem(ClientConnection item) => item.Id; + + internal void AddRange(IEnumerable connections) + { + foreach (ClientConnection connection in connections) + Add(connection); + } +} diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/MessagingServices.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/MessagingServices.cs index d7425c3b8464c..c8b2060de45bf 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/MessagingServices.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/MessagingServices.cs @@ -13,6 +13,8 @@ namespace Azure.CloudMachine; /// public readonly struct MessagingServices { + internal const string DEFAULT_SB_TOPIC = "cm_servicebus_default_topic"; + private readonly CloudMachineClient _cm; internal MessagingServices(CloudMachineClient cm) => _cm = cm; @@ -76,10 +78,10 @@ private ServiceBusSender GetServiceBusSender() return sender; } - internal ServiceBusProcessor GetServiceBusProcessor(string id) + internal ServiceBusProcessor GetServiceBusProcessor(string subscriptionName) { MessagingServices messagingServices = this; - ServiceBusProcessor processor = _cm.Subclients.Get(() => messagingServices.CreateProcessor(id), id); + ServiceBusProcessor processor = _cm.Subclients.Get(() => messagingServices.CreateProcessor(subscriptionName), subscriptionName); return processor; } @@ -87,22 +89,22 @@ private ServiceBusSender CreateSender() { ServiceBusClient client = GetServiceBusClient(); - ClientConnectionOptions connection = _cm.GetConnectionOptions(typeof(ServiceBusSender), default); - ServiceBusSender sender = client.CreateSender(connection.SubclientId); + ClientConnection connection = _cm.GetConnectionOptions(DEFAULT_SB_TOPIC); + ServiceBusSender sender = client.CreateSender(connection.Locator); return sender; } private ServiceBusClient CreateClient() { - ClientConnectionOptions connection = _cm.GetConnectionOptions(typeof(ServiceBusClient), default); - ServiceBusClient client = new(connection.Endpoint!.AbsoluteUri, _cm.Credential); + ClientConnection connection = _cm.GetConnectionOptions(typeof(ServiceBusClient).FullName); + ServiceBusClient client = new(connection.ToUri().AbsoluteUri, _cm.Credential); return client; } - private ServiceBusProcessor CreateProcessor(string id) + private ServiceBusProcessor CreateProcessor(string subscriptionName) { ServiceBusClient client = GetServiceBusClient(); - ClientConnectionOptions connection = _cm.GetConnectionOptions(typeof(ServiceBusProcessor), id); - string[] topicAndSubscription = connection.SubclientId.Split('/'); + ClientConnection connection = _cm.GetConnectionOptions(subscriptionName); + string[] topicAndSubscription = connection.Locator.Split('/'); ServiceBusProcessor processor = client.CreateProcessor(topicAndSubscription[0], topicAndSubscription[1], new() { MaxConcurrentCalls = 5 }); processor.ProcessErrorAsync += (args) => throw new Exception("error processing event", args.Exception); return processor; diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs index dce2bbedcefa6..31e9778e249ba 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs @@ -29,8 +29,8 @@ private BlobContainerClient GetDefaultContainer() CloudMachineClient cm = _cm; BlobContainerClient container = _cm.Subclients.Get(() => { - ClientConnectionOptions connection = cm.GetConnectionOptions(typeof(BlobContainerClient), default); - BlobContainerClient container = new(connection.Endpoint, cm.Credential); + ClientConnection connection = cm.GetConnectionOptions("Azure.Storage.Blobs.BlobContainerClient"); + BlobContainerClient container = new(connection.ToUri(), cm.Credential); return container; }); return container; @@ -42,8 +42,8 @@ private BlobContainerClient GetContainer(string containerName) CloudMachineClient cm = _cm; BlobContainerClient container = cm.Subclients.Get(() => { - ClientConnectionOptions connection = cm.GetConnectionOptions(typeof(BlobContainerClient), containerName); - BlobContainerClient container = new(connection.Endpoint, cm.Credential); + ClientConnection connection = cm.GetConnectionOptions($"{typeof(BlobContainerClient).FullName}@{containerName}"); + BlobContainerClient container = new(connection.ToUri(), cm.Credential); return container; }); return container; @@ -242,7 +242,7 @@ public void WhenUploaded(Action function) { CloudMachineClient cm = _cm; // TODO (Pri 0): once the cache gets GCed, we will stop receiving events - ServiceBusProcessor processor = cm.Messaging.GetServiceBusProcessor("$private"); + ServiceBusProcessor processor = cm.Messaging.GetServiceBusProcessor("cm_servicebus_subscription_private"); // TODO: How to unsubscribe? processor.ProcessMessageAsync += async (args) => { diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/AzureOpenAIExtensions.cs b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/AzureOpenAIExtensions.cs index 91b5f742f7d2a..34b01587d57d0 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/AzureOpenAIExtensions.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/AzureOpenAIExtensions.cs @@ -3,7 +3,6 @@ using System.ClientModel; using System.Collections.Generic; -using System.Runtime.CompilerServices; using System.Text; using Azure.AI.OpenAI; using Azure.Core; @@ -82,28 +81,28 @@ public static string AsText(this ChatMessageContent content) private static AzureOpenAIClient CreateAzureOpenAIClient(this ClientWorkspace workspace) { - ClientConnectionOptions connection = workspace.GetConnectionOptions(typeof(AzureOpenAIClient)); + ClientConnection connection = workspace.GetConnectionOptions(typeof(AzureOpenAIClient).FullName); if (connection.Authentication == ClientAuthenticationMethod.EntraId) { - return new(connection.Endpoint, workspace.Credential); + return new(connection.ToUri(), workspace.Credential); } else { - return new(connection.Endpoint, new ApiKeyCredential(connection.ApiKeyCredential!)); + return new(connection.ToUri(), new ApiKeyCredential(connection.ApiKeyCredential!)); } } private static ChatClient CreateChatClient(this ClientWorkspace workspace, AzureOpenAIClient client) { - ClientConnectionOptions connection = workspace.GetConnectionOptions(typeof(ChatClient)); - ChatClient chat = client.GetChatClient(connection.SubclientId); + ClientConnection connection = workspace.GetConnectionOptions(typeof(ChatClient).FullName); + ChatClient chat = client.GetChatClient(connection.Locator); return chat; } private static EmbeddingClient CreateEmbeddingsClient(this ClientWorkspace workspace, AzureOpenAIClient client) { - ClientConnectionOptions connection = workspace.GetConnectionOptions(typeof(EmbeddingClient)); - EmbeddingClient embeddings = client.GetEmbeddingClient(connection.SubclientId); + ClientConnection connection = workspace.GetConnectionOptions(typeof(EmbeddingClient).FullName); + EmbeddingClient embeddings = client.GetEmbeddingClient(connection.Locator); return embeddings; } diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/KeyVaultExtensions.cs b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/KeyVaultExtensions.cs index a3abef2d2d57c..9706b0a19f65d 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/KeyVaultExtensions.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/KeyVaultExtensions.cs @@ -20,10 +20,10 @@ public static class KeyVaultExtensions /// public static SecretClient GetKeyVaultSecretsClient(this ClientWorkspace workspace) { - ClientConnectionOptions connection = workspace.GetConnectionOptions(typeof(SecretClient)); + ClientConnection connection = workspace.GetConnectionOptions(typeof(SecretClient).FullName); if (connection.Authentication == ClientAuthenticationMethod.EntraId) { - return new(connection.Endpoint, workspace.Credential); + return new(connection.ToUri(), workspace.Credential); } throw new Exception("API key not supported"); } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs index b924f6fe0f582..279bce77ac78f 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs @@ -1,8 +1,8 @@ // Copyright (c) Microsoft Corporation. All rights reserved. // Licensed under the MIT License. -using System.Collections.Generic; -using Azure.Provisioning.Authorization; +using System; +using Azure.Core; using Azure.Provisioning.CloudMachine; using Azure.Provisioning.Expressions; using Azure.Provisioning.KeyVault; @@ -19,6 +19,12 @@ public KeyVaultFeature(KeyVaultSku? sku = default) sku ??= new KeyVaultSku { Name = KeyVaultSkuName.Standard, Family = KeyVaultSkuFamily.A, }; Sku = sku; } + + protected internal override void AddTo(CloudMachineInfrastructure cm) + { + base.AddTo(cm); + cm.Connections.Add(new ClientConnection("Azure.Security.KeyVault.Secrets.SecretClient", $"https://{cm.Id}.vault.azure.net/")); + } protected override ProvisionableResource EmitCore(CloudMachineInfrastructure infrastructure) { // Add a KeyVault to the CloudMachine infrastructure. diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs index 59815c8476736..3b160a5ac30ad 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs @@ -20,8 +20,8 @@ protected internal override void AddTo(CloudMachineInfrastructure cm) { cm.Features.Add(this); - ClientConnectionOptions connection = new(new Uri($"https://{cm.Id}.openai.azure.com")); - cm.Connections.Add("Azure.AI.OpenAI.AzureOpenAIClient", connection); + ClientConnection connection = new("Azure.AI.OpenAI.AzureOpenAIClient", $"https://{cm.Id}.openai.azure.com"); + cm.Connections.Add(connection); } protected override ProvisionableResource EmitCore(CloudMachineInfrastructure cloudMachine) { diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs index 7303f539fa53b..0742f21b75510 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs @@ -47,10 +47,10 @@ protected internal override void AddTo(CloudMachineInfrastructure cm) switch (Kind) { case AIModelKind.Chat: - cm.Connections.Add("OpenAI.Chat.ChatClient", new($"{cm.Id}_chat")); + cm.Connections.Add(new ClientConnection("OpenAI.Chat.ChatClient", $"{cm.Id}_chat")); break; case AIModelKind.Embedding: - cm.Connections.Add("OpenAI.Embeddings.EmbeddingClient", new($"{cm.Id}_embedding")); + cm.Connections.Add(new ClientConnection("OpenAI.Embeddings.EmbeddingClient", $"{cm.Id}_embedding")); break; default: throw new NotImplementedException(); diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs index dce138bb4929d..ee0265d331fa9 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs @@ -26,7 +26,7 @@ public class CloudMachineInfrastructure public FeatureCollection Features { get; } = new(); [EditorBrowsable(EditorBrowsableState.Never)] - public Dictionary Connections { get; } = []; + public ConnectionCollection Connections { get; } = []; [EditorBrowsable(EditorBrowsableState.Never)] public UserAssignedIdentity Identity { get; private set; } @@ -52,22 +52,21 @@ public CloudMachineInfrastructure(string? cmId = default) _infrastructure.Add(new ProvisioningOutput("cm_managed_identity_id", typeof(string)) { Value = Identity.Id }); // Add core features - var storage = new StorageFeature(Id); - Features.Add(storage); - var sbNamespace = new ServiceBusNamespaceFeature(Id); - Features.Add(sbNamespace); - var sbTopicPrivate = new ServiceBusTopicFeature("cm_servicebus_topic_private", sbNamespace); - Features.Add(sbTopicPrivate); - var sbTopicDefault = new ServiceBusTopicFeature("cm_servicebus_default_topic", sbNamespace); - Features.Add(sbTopicDefault); - Features.Add(new ServiceBusSubscriptionFeature("cm_servicebus_subscription_private", sbTopicPrivate)); - Features.Add(new ServiceBusSubscriptionFeature("cm_servicebus_subscription_default", sbTopicDefault)); - var systemTopic = new EventGridSystemTopicFeature(Id, storage); - Features.Add(systemTopic); - Features.Add(new SystemTopicEventSubscriptionFeature("cm_eventgrid_subscription_blob", systemTopic, sbTopicPrivate, sbNamespace)); + var storage = AddFeature(new StorageFeature(Id)); + var sbNamespace = AddFeature(new ServiceBusNamespaceFeature(Id)); + var sbTopicPrivate = AddFeature(new ServiceBusTopicFeature("cm_servicebus_topic_private", sbNamespace)); + var sbTopicDefault = AddFeature(new ServiceBusTopicFeature("cm_servicebus_default_topic", sbNamespace)); + AddFeature(new ServiceBusSubscriptionFeature("cm_servicebus_subscription_private", sbTopicPrivate)); // TODO: should private connections not be in the Connections collection? + AddFeature(new ServiceBusSubscriptionFeature("cm_servicebus_subscription_default", sbTopicDefault)); + var systemTopic = AddFeature(new EventGridSystemTopicFeature(Id, storage)); + AddFeature(new SystemTopicEventSubscriptionFeature("cm_eventgrid_subscription_blob", systemTopic, sbTopicPrivate, sbNamespace)); } - public void AddFeature(CloudMachineFeature feature) => feature.AddTo(this); + public T AddFeature(T feature) where T: CloudMachineFeature + { + feature.AddTo(this); + return feature; + } public void AddEndpoints() { diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs index a52ad529a51e2..3bfd76bc98dbf 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs @@ -1,6 +1,7 @@ // Copyright (c) Microsoft Corporation. All rights reserved. // Licensed under the MIT License. +using Azure.Core; using Azure.Provisioning.CloudMachine; using Azure.Provisioning.Primitives; using Azure.Provisioning.ServiceBus; @@ -38,4 +39,10 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf Emitted = _serviceBusNamespace; return _serviceBusNamespace; } + + protected internal override void AddTo(CloudMachineInfrastructure cm) + { + base.AddTo(cm); + cm.Connections.Add(new ClientConnection("Azure.Messaging.ServiceBus.ServiceBusClient", $"https://{cm.Id}.servicebus.windows.net/")); + } } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs index e2692714ca89e..35fc503dae275 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs @@ -2,6 +2,7 @@ // Licensed under the MIT License. using System; +using Azure.Core; using Azure.Provisioning.CloudMachine; using Azure.Provisioning.Primitives; using Azure.Provisioning.ServiceBus; @@ -31,4 +32,14 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf Emitted = subscription; return subscription; } + + protected internal override void AddTo(CloudMachineInfrastructure cm) + { + base.AddTo(cm); + ClientConnection connection = new( + $"{name}", + $"{parent.Name}/{name}" + ); + cm.Connections.Add(connection); + } } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs index c4fa34d5ba27e..e7e9f2dac9d45 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs @@ -2,20 +2,27 @@ // Licensed under the MIT License. using System; +using Azure.Core; using Azure.Provisioning.CloudMachine; using Azure.Provisioning.Primitives; using Azure.Provisioning.ServiceBus; namespace Azure.CloudMachine; -public class ServiceBusTopicFeature(string name, ServiceBusNamespaceFeature parent) : CloudMachineFeature +public class ServiceBusTopicFeature : CloudMachineFeature { + public ServiceBusTopicFeature(string name, ServiceBusNamespaceFeature parent) + { + Name = name; + _parent = parent; + } + protected override ProvisionableResource EmitCore(CloudMachineInfrastructure infrastructure) { - var topic = new ServiceBusTopic(name, "2021-11-01") + var topic = new ServiceBusTopic(Name, "2021-11-01") { - Name = name, - Parent = EnsureEmits(parent), + Name = Name, + Parent = EnsureEmits(_parent), MaxMessageSizeInKilobytes = 256, DefaultMessageTimeToLive = TimeSpan.FromDays(14), RequiresDuplicateDetection = false, @@ -28,4 +35,17 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf Emitted = topic; return topic; } + + protected internal override void AddTo(CloudMachineInfrastructure cm) + { + base.AddTo(cm); + cm.Connections.Add(new ClientConnection(Name, Name)); + } + + /// + /// The name of the topic. + /// + public string Name { get; } + + private ServiceBusNamespaceFeature _parent; } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs index fa108bc8ffc73..4b010b4dd5a46 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs @@ -1,8 +1,10 @@ // Copyright (c) Microsoft Corporation. All rights reserved. // Licensed under the MIT License. +using System; using System.Collections.Generic; using System.Linq; +using Azure.Core; using Azure.Provisioning; using Azure.Provisioning.CloudMachine; using Azure.Provisioning.Expressions; @@ -76,4 +78,10 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf Emitted = _storage; return _storage; } + + protected internal override void AddTo(CloudMachineInfrastructure cm) + { + base.AddTo(cm); + cm.Connections.Add(new ClientConnection("Azure.Storage.Blobs.BlobContainerClient@default", $"https://{cm.Id}.blob.core.windows.net/default")); + } } From 5ee1a99fe66888918ee4534b71ad858d74197753 Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Tue, 26 Nov 2024 14:59:47 -0800 Subject: [PATCH 08/24] made connection collection serializable --- .../src/ClientConnection.cs | 8 ++++ .../src/ConnectionCollection.cs | 38 +++++++++++++++++++ .../tests/CloudMachineTests.cs | 8 +++- .../tests/CommandsTests.cs | 16 ++++++++ ...udMachineTests.cs => ProvisioningTests.cs} | 9 +---- 5 files changed, 69 insertions(+), 10 deletions(-) create mode 100644 sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/CommandsTests.cs rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/{CloudMachineTests.cs => ProvisioningTests.cs} (83%) diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/ClientConnection.cs b/sdk/cloudmachine/Azure.CloudMachine/src/ClientConnection.cs index 97f49d4ae10df..a360af007cfca 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/ClientConnection.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/ClientConnection.cs @@ -3,6 +3,7 @@ using System; using System.ComponentModel; +using System.Text.Json.Serialization; namespace Azure.Core; @@ -39,6 +40,7 @@ public ClientConnection(string id, string locator, string apiKey) /// public ClientConnection(string id, string locator, ClientAuthenticationMethod auth = ClientAuthenticationMethod.EntraId) { + Id = id; Locator = locator; Authentication = auth; } @@ -68,6 +70,12 @@ public ClientConnection(string id, string locator, ClientAuthenticationMethod au /// /// public Uri ToUri() => new(Locator); + + /// + /// Returns a string representation of the connection. + /// + /// + public override string ToString() => $"{Id} => {Locator}"; } /// diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/ConnectionCollection.cs b/sdk/cloudmachine/Azure.CloudMachine/src/ConnectionCollection.cs index a4f8bb4a8d3c7..1a5087da74e34 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/ConnectionCollection.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/ConnectionCollection.cs @@ -4,6 +4,8 @@ using System; using System.Collections.Generic; using System.Collections.ObjectModel; +using System.Text.Json; +using System.Text.Json.Serialization; using Azure.Core; namespace Azure.CloudMachine; @@ -11,6 +13,7 @@ namespace Azure.CloudMachine; /// /// Represents the connection options for a client. /// +[JsonConverter(typeof(ConnectionCollectionConverter))] public class ConnectionCollection : KeyedCollection { /// @@ -26,3 +29,38 @@ internal void AddRange(IEnumerable connections) Add(connection); } } + +internal class ConnectionCollectionConverter : JsonConverter +{ + public override ConnectionCollection Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) + { + using JsonDocument document = JsonDocument.ParseValue(ref reader); + JsonElement json = document.RootElement; + + ConnectionCollection connections = []; + foreach (JsonElement connectionJson in json.EnumerateArray()) + { + string id = connectionJson.GetProperty("id").GetString(); + string locator = connectionJson.GetProperty("locator").GetString(); + string auth = connectionJson.GetProperty("auth").GetString(); + ClientAuthenticationMethod authMethod = (ClientAuthenticationMethod)Enum.Parse(typeof(ClientAuthenticationMethod), auth); + ClientConnection connection = new ClientConnection(id, locator, authMethod); + connections.Add(connection); + } + return connections; + } + + public override void Write(Utf8JsonWriter writer, ConnectionCollection value, JsonSerializerOptions options) + { + writer.WriteStartArray(); + foreach (ClientConnection connection in value) + { + writer.WriteStartObject(); + writer.WriteString("id", connection.Id); + writer.WriteString("locator", connection.Locator); + writer.WriteString("auth", connection.Authentication.ToString()); + writer.WriteEndObject(); + } + writer.WriteEndArray(); + } +} diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs b/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs index 302da7a4757d7..64154cd940010 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs @@ -15,6 +15,7 @@ using Microsoft.Extensions.Primitives; using System.Linq; using System.IO; +using System.Text.Json; namespace Azure.CloudMachine.Tests; @@ -27,9 +28,12 @@ public void TwoAppsOpenAI() infra.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); //if (args.Contains("-azd")) Azd.Init(infra); - var connections = infra.Connections; + ConnectionCollection connections = infra.Connections; + BinaryData serializedConnections = BinaryData.FromObjectAsJson(connections); + string text = serializedConnections.ToString(); + ConnectionCollection deserializedConnections = JsonSerializer.Deserialize(serializedConnections)!; - CloudMachineClient client = new(connections: connections); + CloudMachineClient client = new(connections: deserializedConnections); ChatClient chat = client.GetOpenAIChatClient(); } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/CommandsTests.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/CommandsTests.cs new file mode 100644 index 0000000000000..0afa31a6a375e --- /dev/null +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/CommandsTests.cs @@ -0,0 +1,16 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +using NUnit.Framework; + +namespace Azure.CloudMachine.Tests; + +public class CommandsTests +{ + [Ignore("no recordings yet")] + [Test] + public void ListModels() + { + CloudMachineCommands.Execute(["-ai", "chat"], exitProcessIfHandled: false); + } +} diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/CloudMachineTests.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/ProvisioningTests.cs similarity index 83% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/CloudMachineTests.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/ProvisioningTests.cs index d0e96c617051d..c1867e84a6829 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/CloudMachineTests.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/ProvisioningTests.cs @@ -13,7 +13,7 @@ namespace Azure.CloudMachine.Tests; -public class CloudMachineTests +public class ProvisioningTests { [Test] public void GenerateBicep() @@ -28,11 +28,4 @@ public void GenerateBicep() string expectedBicep = File.ReadAllText(Path.Combine(TestContext.CurrentContext.TestDirectory, "Data", "GenerateBicep.bicep")).Replace("\r\n", Environment.NewLine); Assert.AreEqual(expectedBicep, actualBicep); } - - [Ignore("no recordings yet")] - [Test] - public void ListModels() - { - CloudMachineCommands.Execute(["-ai", "chat"], exitProcessIfHandled: false); - } } From 8fb301cd62d175eaa0b5c5f652d77485463db79c Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Tue, 26 Nov 2024 15:37:04 -0800 Subject: [PATCH 09/24] refactored tests --- .../src/CoreServices/StorageServices.cs | 31 ++-- .../tests/CloudMachineTests.cs | 156 +----------------- .../tests/CloudMachineTests_rag.cs | 131 +++++++++++++++ .../tests/ConnectionTests.cs | 76 +++++++++ .../CDKLevel3/CloudMachineInfrastructure.cs | 8 +- 5 files changed, 228 insertions(+), 174 deletions(-) create mode 100644 sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests_rag.cs create mode 100644 sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs index 31e9778e249ba..5ab6d3c9cf160 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CoreServices/StorageServices.cs @@ -12,6 +12,7 @@ using Azure.Storage.Blobs.Specialized; using Azure.Messaging.ServiceBus; using ContentType = Azure.Core.ContentType; +using System.ComponentModel; namespace Azure.CloudMachine; @@ -24,25 +25,21 @@ public readonly struct StorageServices internal StorageServices(CloudMachineClient cm) => _cm = cm; - private BlobContainerClient GetDefaultContainer() - { - CloudMachineClient cm = _cm; - BlobContainerClient container = _cm.Subclients.Get(() => - { - ClientConnection connection = cm.GetConnectionOptions("Azure.Storage.Blobs.BlobContainerClient"); - BlobContainerClient container = new(connection.ToUri(), cm.Credential); - return container; - }); - return container; - } - - private BlobContainerClient GetContainer(string containerName) + // TODO: do we want Azure.Storage.Blobs in the public API? This would prevent us from using a custom implementation. + /// + /// Gets the container client. + /// + /// + /// + [EditorBrowsable(EditorBrowsableState.Never)] + public BlobContainerClient GetContainer(string containerName = default) { - string blobContainerClientId = typeof(BlobContainerClient).FullName; + if (containerName == default) containerName = "default"; + string blobContainerClientId = $"{typeof(BlobContainerClient).FullName}@{containerName}"; CloudMachineClient cm = _cm; BlobContainerClient container = cm.Subclients.Get(() => { - ClientConnection connection = cm.GetConnectionOptions($"{typeof(BlobContainerClient).FullName}@{containerName}"); + ClientConnection connection = cm.GetConnectionOptions(blobContainerClientId); BlobContainerClient container = new(connection.ToUri(), cm.Credential); return container; }); @@ -111,7 +108,7 @@ public async Task UploadAsync(Stream fileStream, string name = default, private BlockBlobClient GetBlobClient(ref string name) { - BlobContainerClient container = GetDefaultContainer(); + BlobContainerClient container = GetContainer(default); if (name == default) name = $"b{Guid.NewGuid()}"; BlockBlobClient client = container.GetBlockBlobClient(name); return client; @@ -206,7 +203,7 @@ public async Task DeleteAsync(string path) private BlobClient GetBlobClientFromPath(string path, string containerName) { - BlobContainerClient _blobContainer = GetDefaultContainer(); + BlobContainerClient _blobContainer = GetContainer(default); string blobPath = ConvertPathToBlobPath(path, _blobContainer); if (containerName is null) { diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs b/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs index 64154cd940010..d629098e63ca4 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs @@ -10,58 +10,11 @@ using Azure.CloudMachine.KeyVault; using NUnit.Framework; using OpenAI.Chat; -using Microsoft.Extensions.Configuration; -using System.Collections.Generic; -using Microsoft.Extensions.Primitives; -using System.Linq; -using System.IO; -using System.Text.Json; namespace Azure.CloudMachine.Tests; -public class CloudMachineTests +public partial class CloudMachineTests { - [Test] - public void TwoAppsOpenAI() - { - CloudMachineInfrastructure infra = new(); - infra.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); - //if (args.Contains("-azd")) Azd.Init(infra); - - ConnectionCollection connections = infra.Connections; - BinaryData serializedConnections = BinaryData.FromObjectAsJson(connections); - string text = serializedConnections.ToString(); - ConnectionCollection deserializedConnections = JsonSerializer.Deserialize(serializedConnections)!; - - CloudMachineClient client = new(connections: deserializedConnections); - ChatClient chat = client.GetOpenAIChatClient(); - } - - //[Test] - //public void SingleAppOpenAI(string[] args) - //{ - // CloudMachineClient client = new(); - // client.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); - // if (args.Contains("-azd")) Azd.Init(client); - - // ChatClient chat = client.GetOpenAIChatClient(); - //} - - [Test] - public void Configuration() - { - CloudMachineCommands.Execute(["-bicep"], (infrastructure) => - { - infrastructure.AddFeature(new KeyVaultFeature()); - infrastructure.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); - infrastructure.AddFeature(new OpenAIModelFeature("text-embedding-ada-002", "2", AIModelKind.Embedding)); - }, exitProcessIfHandled: false); - - CloudMachineWorkspace cm = new(); - Console.WriteLine(cm.Id); - var embeddings = cm.GetOpenAIEmbeddingsClient(); - } - [Ignore("no recordings yet")] [Test] [TestCase([new string[] { "-bicep" }])] @@ -166,111 +119,4 @@ public void Demo(string[] args) // go! cm.Messaging.SendJson("Tell me something about Redmond, WA."); } - - [Ignore("no recordings yet")] - [Test] - public void EmbeddingsDemo() - { - string[] testMessages = [ - "When did the continuation policy change for DefaultAzureCredential?", - "Do I love Seattle?", - "What is the current time?", - "What's the weather in Seattle?", - "Do you think I would like the weather there?", - ]; - - CloudMachineClient cm = new(default, new MockConfiguration("cmec4615e3fdfa44e")); - var chat = cm.GetOpenAIChatClient(); - var embeddings = cm.GetOpenAIEmbeddingsClient(); - - // helpers - ChatTools tools = new(typeof(MyFunctions)); - EmbeddingsVectorbase vectors = new(embeddings, null, 1000); - vectors.Add("I love Seattle."); - vectors.Add(File.ReadAllText(@"C:\Users\chriss\Desktop\Identity-README.md")); - - ChatCompletionOptions options = new(); - foreach (var definition in tools.Definitions) - { - options.Tools.Add(definition); - } - options.ToolChoice = ChatToolChoice.CreateAutoChoice(); - - List prompt = new(); - - foreach (var testMessage in testMessages) - { - Console.WriteLine($"u: {testMessage}"); - IEnumerable relatedItems = vectors.Find(testMessage); - foreach (VectorbaseEntry relatedItem in relatedItems) - { - prompt.Add(ChatMessage.CreateSystemMessage(relatedItem.Data.ToString())); - } - - prompt.Add(ChatMessage.CreateUserMessage(testMessage)); - CallOpenAI(); - // filter the prompt to only include the user message - var responses = prompt.Where(message => message is AssistantChatMessage).Select(m => m.Content[0].Text).ToList(); - } - void CallOpenAI() - { - bool requiresAction; - do - { - requiresAction = false; - var completion = chat.CompleteChat(prompt, options).Value; - switch (completion.FinishReason) - { - case ChatFinishReason.ToolCalls: - // TODO: figure out why the model is returning bogus tool call results. - // prompt.Add(new AssistantChatMessage(completion)); - // IEnumerable callResults = tools.CallAll(completion.ToolCalls); - // prompt.AddRange(callResults); - requiresAction = true; - break; - case ChatFinishReason.Length: - Console.WriteLine("Incomplete model output due to MaxTokens parameter or token limit exceeded."); - break; - case ChatFinishReason.ContentFilter: - Console.WriteLine("Omitted content due to a content filter flag."); - break; - case ChatFinishReason.Stop: - prompt.Add(new AssistantChatMessage(completion)); - break; - default: - throw new NotImplementedException("Unknown finish reason."); - } - } while (requiresAction); - } - } - - public static class MyFunctions - { - [System.ComponentModel.Description("Returns the current weather at the specified location")] - public static string GetCurrentWeather(string location, string? unit = default) => $"1 million degrees {unit}"; - - [System.ComponentModel.Description("Returns the user's current location")] - public static string GetCurrentLocation() => "Planet Earth"; - - [System.ComponentModel.Description("Returns the current time.")] - public static string GetCurrentTime() => DateTimeOffset.Now.ToString("t"); - } - - private class MockConfiguration : IConfiguration - { - private readonly string _cmId; - - public MockConfiguration(string cmId) - { - _cmId = cmId; - } - - public string? this[string key] { get => _cmId; set => throw new NotImplementedException(); } - - public IEnumerable GetChildren() => throw new NotImplementedException(); - - public IChangeToken GetReloadToken() => throw new NotImplementedException(); - - public IConfigurationSection GetSection(string key) => throw new NotImplementedException(); - } } diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests_rag.cs b/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests_rag.cs new file mode 100644 index 0000000000000..9e10a425db168 --- /dev/null +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests_rag.cs @@ -0,0 +1,131 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +#nullable enable + +using System; +using System.Threading; +using Azure.Security.KeyVault.Secrets; +using Azure.CloudMachine.OpenAI; +using Azure.CloudMachine.KeyVault; +using NUnit.Framework; +using OpenAI.Chat; +using Microsoft.Extensions.Configuration; +using System.Collections.Generic; +using Microsoft.Extensions.Primitives; +using System.Linq; +using System.IO; +using System.Text.Json; +using Azure.Storage.Blobs; + +namespace Azure.CloudMachine.Tests; + +public partial class CloudMachineTests +{ + [Ignore("no recordings yet")] + [Test] + public void EmbeddingsDemo() + { + string[] testMessages = [ + "When did the continuation policy change for DefaultAzureCredential?", + "Do I love Seattle?", + "What is the current time?", + "What's the weather in Seattle?", + "Do you think I would like the weather there?", + ]; + + CloudMachineClient cm = new(default, new MockConfiguration("cmec4615e3fdfa44e")); + var chat = cm.GetOpenAIChatClient(); + var embeddings = cm.GetOpenAIEmbeddingsClient(); + + // helpers + ChatTools tools = new(typeof(MyFunctions)); + EmbeddingsVectorbase vectors = new(embeddings, null, 1000); + vectors.Add("I love Seattle."); + vectors.Add(File.ReadAllText(@"C:\Users\chriss\Desktop\Identity-README.md")); + + ChatCompletionOptions options = new(); + foreach (var definition in tools.Definitions) + { + options.Tools.Add(definition); + } + options.ToolChoice = ChatToolChoice.CreateAutoChoice(); + + List prompt = new(); + + foreach (var testMessage in testMessages) + { + Console.WriteLine($"u: {testMessage}"); + IEnumerable relatedItems = vectors.Find(testMessage); + foreach (VectorbaseEntry relatedItem in relatedItems) + { + prompt.Add(ChatMessage.CreateSystemMessage(relatedItem.Data.ToString())); + } + + prompt.Add(ChatMessage.CreateUserMessage(testMessage)); + CallOpenAI(); + // filter the prompt to only include the user message + var responses = prompt.Where(message => message is AssistantChatMessage).Select(m => m.Content[0].Text).ToList(); + } + void CallOpenAI() + { + bool requiresAction; + do + { + requiresAction = false; + var completion = chat.CompleteChat(prompt, options).Value; + switch (completion.FinishReason) + { + case ChatFinishReason.ToolCalls: + // TODO: figure out why the model is returning bogus tool call results. + // prompt.Add(new AssistantChatMessage(completion)); + // IEnumerable callResults = tools.CallAll(completion.ToolCalls); + // prompt.AddRange(callResults); + requiresAction = true; + break; + case ChatFinishReason.Length: + Console.WriteLine("Incomplete model output due to MaxTokens parameter or token limit exceeded."); + break; + case ChatFinishReason.ContentFilter: + Console.WriteLine("Omitted content due to a content filter flag."); + break; + case ChatFinishReason.Stop: + prompt.Add(new AssistantChatMessage(completion)); + break; + default: + throw new NotImplementedException("Unknown finish reason."); + } + } while (requiresAction); + } + } + + public static class MyFunctions + { + [System.ComponentModel.Description("Returns the current weather at the specified location")] + public static string GetCurrentWeather(string location, string? unit = default) => $"1 million degrees {unit}"; + + [System.ComponentModel.Description("Returns the user's current location")] + public static string GetCurrentLocation() => "Planet Earth"; + + [System.ComponentModel.Description("Returns the current time.")] + public static string GetCurrentTime() => DateTimeOffset.Now.ToString("t"); + } + + private class MockConfiguration : IConfiguration + { + private readonly string _cmId; + + public MockConfiguration(string cmId) + { + _cmId = cmId; + } + + public string? this[string key] { get => _cmId; set => throw new NotImplementedException(); } + + public IEnumerable GetChildren() => throw new NotImplementedException(); + + public IChangeToken GetReloadToken() => throw new NotImplementedException(); + + public IConfigurationSection GetSection(string key) => throw new NotImplementedException(); + } +} diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs b/sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs new file mode 100644 index 0000000000000..36724bcfc0b04 --- /dev/null +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs @@ -0,0 +1,76 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +#nullable enable + +using System; +using Azure.CloudMachine.OpenAI; +using NUnit.Framework; +using OpenAI.Chat; +using System.Text.Json; +using Azure.Storage.Blobs; +using Azure.CloudMachine.KeyVault; + +namespace Azure.CloudMachine.Tests; + +public class ConnectionTests +{ + [Test] + public void TwoApps() + { + // app 1 (with a dependency on the CDK) + CloudMachineInfrastructure infra = new(); + infra.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); + //if (args.Contains("-azd")) Azd.Init(infra); + BinaryData serializedConnections = BinaryData.FromObjectAsJson(infra.Connections); + + // app 2 (no dependency on the CDK) + ConnectionCollection deserializedConnections = JsonSerializer.Deserialize(serializedConnections)!; + CloudMachineClient client = new(connections: deserializedConnections); + + ChatClient chat = client.GetOpenAIChatClient(); + StorageServices storage = client.Storage; + BlobContainerClient container = storage.GetContainer(default); + MessagingServices messaging = client.Messaging; + } + + [Test] + public void TwoClients() + { + CloudMachineInfrastructure infra = new(); + infra.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); + + CloudMachineClient client = infra.GetClient(); + + ChatClient chat = client.GetOpenAIChatClient(); + StorageServices storage = client.Storage; + BlobContainerClient container = storage.GetContainer(default); + MessagingServices messaging = client.Messaging; + } + + //[Test] + //public void SingleAppOpenAI(string[] args) + //{ + // CloudMachineClient client = new(); + // client.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); + // if (args.Contains("-azd")) + // Azd.Init(client); + + // ChatClient chat = client.GetOpenAIChatClient(); + //} + + [Test] + public void Configuration() + { + CloudMachineCommands.Execute(["-bicep"], (infrastructure) => + { + infrastructure.AddFeature(new KeyVaultFeature()); + infrastructure.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); + infrastructure.AddFeature(new OpenAIModelFeature("text-embedding-ada-002", "2", AIModelKind.Embedding)); + }, exitProcessIfHandled: false); + + CloudMachineWorkspace cm = new CloudMachineClient(); + Console.WriteLine(cm.Id); + var embeddings = cm.GetOpenAIEmbeddingsClient(); + } +} diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs index ee0265d331fa9..98d9f391f44a1 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs @@ -4,9 +4,7 @@ using System; using System.Collections.Generic; using System.ComponentModel; -using Azure.Core; using Azure.Provisioning; -using Azure.Provisioning.Authorization; using Azure.Provisioning.CloudMachine; using Azure.Provisioning.Expressions; using Azure.Provisioning.Primitives; @@ -32,6 +30,12 @@ public class CloudMachineInfrastructure public UserAssignedIdentity Identity { get; private set; } public string Id { get; private set; } + public CloudMachineClient GetClient() + { + CloudMachineClient client = new(connections: Connections); + return client; + } + /// /// The common principalId parameter. /// From 1a3b16d5e0708b5603cde193c3a3af214276610d Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Tue, 26 Nov 2024 15:58:33 -0800 Subject: [PATCH 10/24] connection scenario tests --- .../src/CloudMachineWorkspace.cs | 6 ++- .../tests/ConnectionTests.cs | 48 ++++++++++++------- .../src/CDKLevel3/CloudMachineCommands.cs | 28 ++++++++++- 3 files changed, 62 insertions(+), 20 deletions(-) diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs index 34cabcff4f722..e2d42567b839e 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs @@ -20,7 +20,11 @@ namespace Azure.CloudMachine; /// public class CloudMachineWorkspace : ClientWorkspace { - private ConnectionCollection Connections { get; } = []; + /// + /// subclient connections. + /// + [EditorBrowsable(EditorBrowsableState.Never)] + public ConnectionCollection Connections { get; } = []; /// /// The cloud machine ID. diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs b/sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs index 36724bcfc0b04..698d432cf751b 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs @@ -15,6 +15,14 @@ namespace Azure.CloudMachine.Tests; public class ConnectionTests { + private static void ValidateClient(CloudMachineClient client) + { + ChatClient chat = client.GetOpenAIChatClient(); + StorageServices storage = client.Storage; + BlobContainerClient container = storage.GetContainer(default); + MessagingServices messaging = client.Messaging; + } + [Test] public void TwoApps() { @@ -28,10 +36,7 @@ public void TwoApps() ConnectionCollection deserializedConnections = JsonSerializer.Deserialize(serializedConnections)!; CloudMachineClient client = new(connections: deserializedConnections); - ChatClient chat = client.GetOpenAIChatClient(); - StorageServices storage = client.Storage; - BlobContainerClient container = storage.GetContainer(default); - MessagingServices messaging = client.Messaging; + ValidateClient(client); } [Test] @@ -42,22 +47,29 @@ public void TwoClients() CloudMachineClient client = infra.GetClient(); - ChatClient chat = client.GetOpenAIChatClient(); - StorageServices storage = client.Storage; - BlobContainerClient container = storage.GetContainer(default); - MessagingServices messaging = client.Messaging; + ValidateClient(client); } - //[Test] - //public void SingleAppOpenAI(string[] args) - //{ - // CloudMachineClient client = new(); - // client.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); - // if (args.Contains("-azd")) - // Azd.Init(client); + [Test] + public void SingleClientAdd() + { + CloudMachineClient client = new(); + client.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); + //if (args.Contains("-azd")) Azd.Init(client); + + ChatClient chat = client.GetOpenAIChatClient(); + } - // ChatClient chat = client.GetOpenAIChatClient(); - //} + [Test] + public void SingleClientConfigure() + { + CloudMachineClient client = new(); + client.Configure((infrastructure) => + { + infrastructure.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); + }); + ValidateClient(client); + } [Test] public void Configuration() @@ -69,7 +81,7 @@ public void Configuration() infrastructure.AddFeature(new OpenAIModelFeature("text-embedding-ada-002", "2", AIModelKind.Embedding)); }, exitProcessIfHandled: false); - CloudMachineWorkspace cm = new CloudMachineClient(); + CloudMachineClient cm = new(); Console.WriteLine(cm.Id); var embeddings = cm.GetOpenAIEmbeddingsClient(); } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs index f388f9d6fca32..448916f519110 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs @@ -9,11 +9,37 @@ using Azure.AI.OpenAI; using Azure.Core; using Azure.Identity; +using Azure.Provisioning.CloudMachine; namespace Azure.CloudMachine; -public class CloudMachineCommands +public static class CloudMachineCommands { + public static T AddFeature(this CloudMachineClient client, T feature) where T : CloudMachineFeature + { + CloudMachineInfrastructure infra = client.Subclients.Get(() => + { + CloudMachineInfrastructure infra = new CloudMachineInfrastructure(); + return infra; + }); + infra.AddFeature(feature); + return feature; + } + + public static void Configure(this CloudMachineClient client, Action? configure = default) + { + CloudMachineInfrastructure cmi = new(client.Id); + if (configure != default) + { + configure(cmi); + } + foreach (ClientConnection clientConnection in cmi.Connections) + { + client.Connections.Add(clientConnection); + } + Azd.Init(cmi); + } + public static bool Execute(string[] args, Action? configure = default, bool exitProcessIfHandled = true) { if (args.Length < 1) From 5c758140a581e62d58711117e687211bb5e00f78 Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Wed, 27 Nov 2024 09:50:33 -0800 Subject: [PATCH 11/24] basic scenarios work and tests pass --- .../tests/ConnectionTests.cs | 62 +++++++++---------- .../AzureSdkExtensions/AppServiceFeature.cs | 2 +- .../src/AzureSdkExtensions/KeyVaultFeature.cs | 8 +-- .../src/AzureSdkExtensions/OpenAIFeature.cs | 19 +++--- .../AzureSdkExtensions/OpenAIModelFeature.cs | 29 ++++----- .../src/CDKLevel3/Azd.cs | 6 ++ .../CDKLevel3/CloudMachineClientExtensions.cs | 54 ++++++++++++++++ .../src/CDKLevel3/CloudMachineCommands.cs | 37 ++--------- .../src/CDKLevel3/CloudMachineFeature.cs | 12 ++-- .../CDKLevel3/CloudMachineInfrastructure.cs | 9 +-- .../CDKLevel3/EventGridSystemTopicFeature.cs | 2 +- .../CDKLevel3/ServiceBusNamespaceFeature.cs | 7 +-- .../ServiceBusSubscriptionFeature.cs | 7 +-- .../src/CDKLevel3/ServiceBusTopicFeature.cs | 7 +-- .../src/CDKLevel3/StorageFeature.cs | 7 +-- .../SystemTopicEventSubscriptionFeature.cs | 2 +- 16 files changed, 149 insertions(+), 121 deletions(-) create mode 100644 sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineClientExtensions.cs diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs b/sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs index 698d432cf751b..df30d955cabf2 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs @@ -4,25 +4,33 @@ #nullable enable using System; +using System.Linq; +using System.Text.Json; +using Azure.CloudMachine.KeyVault; using Azure.CloudMachine.OpenAI; +using Azure.Storage.Blobs; using NUnit.Framework; using OpenAI.Chat; -using System.Text.Json; -using Azure.Storage.Blobs; -using Azure.CloudMachine.KeyVault; namespace Azure.CloudMachine.Tests; public class ConnectionTests { - private static void ValidateClient(CloudMachineClient client) + [Test] + [TestCase([new string[0]])] + public void TwoClients(string[] args) { - ChatClient chat = client.GetOpenAIChatClient(); - StorageServices storage = client.Storage; - BlobContainerClient container = storage.GetContainer(default); - MessagingServices messaging = client.Messaging; + CloudMachineInfrastructure infra = new(); + infra.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); + if (args.Contains("-azd")) Azd.Init(infra); + + CloudMachineClient client = infra.GetClient(); + + ValidateClient(client); } + // this tests the scenario where provisioning is done in one app, but runtime is done by another app + // the connections needs to be serialized and deserialized [Test] public void TwoApps() { @@ -39,23 +47,15 @@ public void TwoApps() ValidateClient(client); } + // TODO: maybe this is too hacky. do we really need this? [Test] - public void TwoClients() - { - CloudMachineInfrastructure infra = new(); - infra.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); - - CloudMachineClient client = infra.GetClient(); - - ValidateClient(client); - } - - [Test] - public void SingleClientAdd() + [TestCase([new string[0]])] + public void SingleClientAdd(string[] args) { CloudMachineClient client = new(); client.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); - //if (args.Contains("-azd")) Azd.Init(client); + + if (args.Contains("-azd")) Azd.Init(client); ChatClient chat = client.GetOpenAIChatClient(); } @@ -66,23 +66,19 @@ public void SingleClientConfigure() CloudMachineClient client = new(); client.Configure((infrastructure) => { + infrastructure.AddFeature(new KeyVaultFeature()); infrastructure.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); + infrastructure.AddFeature(new OpenAIModelFeature("text-embedding-ada-002", "2", AIModelKind.Embedding)); }); ValidateClient(client); + var embeddings = client.GetOpenAIEmbeddingsClient(); } - [Test] - public void Configuration() + private static void ValidateClient(CloudMachineClient client) { - CloudMachineCommands.Execute(["-bicep"], (infrastructure) => - { - infrastructure.AddFeature(new KeyVaultFeature()); - infrastructure.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); - infrastructure.AddFeature(new OpenAIModelFeature("text-embedding-ada-002", "2", AIModelKind.Embedding)); - }, exitProcessIfHandled: false); - - CloudMachineClient cm = new(); - Console.WriteLine(cm.Id); - var embeddings = cm.GetOpenAIEmbeddingsClient(); + ChatClient chat = client.GetOpenAIChatClient(); + StorageServices storage = client.Storage; + BlobContainerClient container = storage.GetContainer(default); + MessagingServices messaging = client.Messaging; } } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs index 8d8ba98c634a3..77be44f5904a3 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs @@ -18,7 +18,7 @@ public AppServiceFeature() Sku = new AppServiceSkuDescription { Tier = "Free", Name = "F1" }; } - protected override ProvisionableResource EmitCore(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure infrastructure) { //Add a App Service to the CloudMachine infrastructure. AppServicePlan hostingPlan = new("cm_hosting_plan") diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs index 279bce77ac78f..6cf7fb655196f 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs @@ -20,12 +20,12 @@ public KeyVaultFeature(KeyVaultSku? sku = default) Sku = sku; } - protected internal override void AddTo(CloudMachineInfrastructure cm) + protected internal override void EmitConnections(ConnectionCollection connections, string cmId) { - base.AddTo(cm); - cm.Connections.Add(new ClientConnection("Azure.Security.KeyVault.Secrets.SecretClient", $"https://{cm.Id}.vault.azure.net/")); + connections.Add(new ClientConnection("Azure.Security.KeyVault.Secrets.SecretClient", $"https://{cmId}.vault.azure.net/")); } - protected override ProvisionableResource EmitCore(CloudMachineInfrastructure infrastructure) + + protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure infrastructure) { // Add a KeyVault to the CloudMachine infrastructure. KeyVaultService keyVaultResource = new("cm_kv") diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs index 3b160a5ac30ad..97a1200360c38 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs @@ -16,14 +16,7 @@ internal class OpenAIFeature : CloudMachineFeature public OpenAIFeature() { } - protected internal override void AddTo(CloudMachineInfrastructure cm) - { - cm.Features.Add(this); - - ClientConnection connection = new("Azure.AI.OpenAI.AzureOpenAIClient", $"https://{cm.Id}.openai.azure.com"); - cm.Connections.Add(connection); - } - protected override ProvisionableResource EmitCore(CloudMachineInfrastructure cloudMachine) + protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure cloudMachine) { CognitiveServicesAccount cognitiveServices = CreateOpenAIAccount(cloudMachine); cloudMachine.AddConstruct(cognitiveServices); @@ -46,6 +39,16 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure clo return cognitiveServices; } + protected internal override void EmitConnections(ConnectionCollection connections, string cmId) + { + ClientConnection connection = new("Azure.AI.OpenAI.AzureOpenAIClient", $"https://{cmId}.openai.azure.com"); + + if (!connections.Contains(connection)) + { + connections.Add(connection); + } + } + internal void AddModel(OpenAIModelFeature model) { if (model.Account != null) diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs index 0742f21b75510..e3b269fe61b05 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs @@ -25,39 +25,36 @@ public OpenAIModelFeature(string model, string modelVersion, AIModelKind kind = internal OpenAIFeature Account { get; set; } = default!; - private static OpenAIFeature GetOrCreateOpenAI(CloudMachineInfrastructure cm) + protected internal override void EmitFeatures(FeatureCollection features, string cmId) { // TODO: is it OK that we return the first one? - OpenAIFeature? openAI = cm.Features.FindAll().FirstOrDefault(); - if (openAI != default) return openAI; - - openAI = new OpenAIFeature(); - cm.AddFeature(openAI); - - return openAI; + OpenAIFeature? openAI = features.FindAll().FirstOrDefault(); + if (openAI == default) + { + openAI = new OpenAIFeature(); // TODO: we need to add connection + features.Add(openAI); + } + openAI.AddModel(this); } - protected internal override void AddTo(CloudMachineInfrastructure cm) + protected internal override void EmitConnections(ConnectionCollection connections, string cmId) { - OpenAIFeature openAI = GetOrCreateOpenAI(cm); - - openAI.AddModel(this); - + Account.EmitConnections(connections, cmId); // add connections switch (Kind) { case AIModelKind.Chat: - cm.Connections.Add(new ClientConnection("OpenAI.Chat.ChatClient", $"{cm.Id}_chat")); + connections.Add(new ClientConnection("OpenAI.Chat.ChatClient", $"{cmId}_chat")); break; case AIModelKind.Embedding: - cm.Connections.Add(new ClientConnection("OpenAI.Embeddings.EmbeddingClient", $"{cm.Id}_embedding")); + connections.Add(new ClientConnection("OpenAI.Embeddings.EmbeddingClient", $"{cmId}_embedding")); break; default: throw new NotImplementedException(); } } - protected override ProvisionableResource EmitCore(CloudMachineInfrastructure cm) + protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure cm) { string name = Kind switch { diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/Azd.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/Azd.cs index 094b3d101bde2..f1a94f21ca235 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/Azd.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/Azd.cs @@ -19,6 +19,12 @@ public static class Azd private const string MainBicepName = "main"; private const string ResourceGroupVersion = "2024-03-01"; + public static void Init(CloudMachineClient client, string? infraDirectory = default) + { + CloudMachineInfrastructure infra = client.GetInfrastructure(); + Init(infra, infraDirectory); + } + public static void Init(CloudMachineInfrastructure infra, string? infraDirectory = default) { if (infraDirectory == default) infraDirectory = Path.Combine(".", "infra"); diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineClientExtensions.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineClientExtensions.cs new file mode 100644 index 0000000000000..7559dd4437623 --- /dev/null +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineClientExtensions.cs @@ -0,0 +1,54 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +using System; +using System.ComponentModel; +using Azure.Core; +using Azure.Provisioning.CloudMachine; + +namespace Azure.CloudMachine; + +public static class CloudMachineClientExtensions +{ + public static void Configure(this CloudMachineClient client, Action? configure = default) + { + CloudMachineInfrastructure cmi = new(client.Id); + if (configure != default) + { + configure(cmi); + } + foreach (ClientConnection clientConnection in cmi.Connections) + { + client.Connections.Add(clientConnection); + } + Azd.Init(cmi); + } + + public static T AddFeature(this CloudMachineClient client, T feature) where T : CloudMachineFeature + { + CloudMachineInfrastructure infra = client.GetInfrastructure(); + infra.AddFeature(feature); + CopyConnections(infra.Connections, client.Connections); + return feature; + } + + [EditorBrowsable(EditorBrowsableState.Never)] + public static CloudMachineInfrastructure GetInfrastructure(this CloudMachineClient client) + { + return client.Subclients.Get(() => + { + CloudMachineInfrastructure infra = new CloudMachineInfrastructure(client.Id); + return infra; + }); + } + + private static void CopyConnections(ConnectionCollection from, ConnectionCollection to) + { + foreach (var connection in from) + { + if (!to.Contains(connection)) { + to.Add(connection); + } + } + } +} diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs index 448916f519110..1e80f6405bcd2 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs @@ -4,42 +4,17 @@ using System; using System.ClientModel.Primitives; using System.ClientModel.TypeSpec; +using System.Collections.Generic; using System.IO; using System.Text.Json; using Azure.AI.OpenAI; using Azure.Core; using Azure.Identity; -using Azure.Provisioning.CloudMachine; namespace Azure.CloudMachine; public static class CloudMachineCommands { - public static T AddFeature(this CloudMachineClient client, T feature) where T : CloudMachineFeature - { - CloudMachineInfrastructure infra = client.Subclients.Get(() => - { - CloudMachineInfrastructure infra = new CloudMachineInfrastructure(); - return infra; - }); - infra.AddFeature(feature); - return feature; - } - - public static void Configure(this CloudMachineClient client, Action? configure = default) - { - CloudMachineInfrastructure cmi = new(client.Id); - if (configure != default) - { - configure(cmi); - } - foreach (ClientConnection clientConnection in cmi.Connections) - { - client.Connections.Add(clientConnection); - } - Azd.Init(cmi); - } - public static bool Execute(string[] args, Action? configure = default, bool exitProcessIfHandled = true) { if (args.Length < 1) @@ -60,7 +35,7 @@ public static bool Execute(string[] args, Action? co if (args[0] == "-tsp") { - GenerateTsp(cmi); + GenerateTsp(cmi.Endpoints); return Handled(exitProcessIfHandled); } @@ -143,11 +118,11 @@ private static void ListAzureOpenaAIModels(string cmid, string? option) } } - private static void GenerateTsp(CloudMachineInfrastructure cmi) + private static void GenerateTsp(IEnumerable operationGroups) { - foreach (Type endpoints in cmi.Endpoints) + foreach (Type operationGroup in operationGroups) { - string name = endpoints.Name; + string name = operationGroup.Name; if (name.StartsWith("I")) name = name.Substring(1); string directory = Path.Combine(".", "tsp"); @@ -156,7 +131,7 @@ private static void GenerateTsp(CloudMachineInfrastructure cmi) if (File.Exists(tspFile)) File.Delete(tspFile); using FileStream stream = File.OpenWrite(tspFile); - TypeSpecWriter.WriteServer(stream, endpoints); + TypeSpecWriter.WriteServer(stream, operationGroup); } } } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs index 57c435394af23..97f552a08138d 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs @@ -11,18 +11,18 @@ namespace Azure.Provisioning.CloudMachine; public abstract class CloudMachineFeature { - protected internal virtual void AddTo(CloudMachineInfrastructure cm) => cm.Features.Add(this); + protected abstract ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure cm); + protected internal virtual void EmitConnections(ConnectionCollection connections, string cmId) { } + protected internal virtual void EmitFeatures(FeatureCollection features, string cmId) + => features.Add(this); internal void Emit(CloudMachineInfrastructure cm) { - if (Emitted != null) - return; - ProvisionableResource provisionable = EmitCore(cm); + if (Emitted != null) return; + ProvisionableResource provisionable = EmitInfrastructure(cm); Emitted = provisionable; } - protected abstract ProvisionableResource EmitCore(CloudMachineInfrastructure cm); - [EditorBrowsable(EditorBrowsableState.Never)] public ProvisionableResource Emitted { get; protected set; } = default!; diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs index 98d9f391f44a1..dc915b35596fe 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs @@ -18,7 +18,7 @@ public class CloudMachineInfrastructure internal const string SB_PRIVATE_SUB = "cm_servicebus_subscription_private"; private readonly Infrastructure _infrastructure = new("cm"); - private readonly List _resources = []; + private readonly List _constrcuts = []; internal List Endpoints { get; } = []; public FeatureCollection Features { get; } = new(); @@ -68,7 +68,8 @@ public CloudMachineInfrastructure(string? cmId = default) public T AddFeature(T feature) where T: CloudMachineFeature { - feature.AddTo(this); + feature.EmitFeatures(Features, Id); + feature.EmitConnections(Connections, Id); return feature; } @@ -83,7 +84,7 @@ public void AddEndpoints() [EditorBrowsable(EditorBrowsableState.Never)] public void AddConstruct(NamedProvisionableConstruct resource) { - _resources.Add(resource); + _constrcuts.Add(resource); } [EditorBrowsable(EditorBrowsableState.Never)] @@ -108,7 +109,7 @@ public ProvisioningPlan Build(ProvisioningBuildOptions? context = default) _infrastructure.Add(Identity); // Add any add-on resources to the infrastructure. - foreach (Provisionable resource in _resources) + foreach (Provisionable resource in _constrcuts) { _infrastructure.Add(resource); } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs index d1689c72ed3d1..7f505f0537c60 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs @@ -12,7 +12,7 @@ namespace Azure.CloudMachine; public class EventGridSystemTopicFeature(string topicName, CloudMachineFeature source) : CloudMachineFeature { - protected override ProvisionableResource EmitCore(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure infrastructure) { var topic = new SystemTopic("cm_eventgrid_topic", "2022-06-15") { diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs index 3bfd76bc98dbf..c04f504ab10b5 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs @@ -10,7 +10,7 @@ namespace Azure.CloudMachine; public class ServiceBusNamespaceFeature(string name, ServiceBusSkuName sku = ServiceBusSkuName.Standard, ServiceBusSkuTier tier = ServiceBusSkuTier.Standard) : CloudMachineFeature { - protected override ProvisionableResource EmitCore(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure infrastructure) { var _serviceBusNamespace = new ServiceBusNamespace("cm_servicebus") { @@ -40,9 +40,8 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf return _serviceBusNamespace; } - protected internal override void AddTo(CloudMachineInfrastructure cm) + protected internal override void EmitConnections(ConnectionCollection connections, string cmId) { - base.AddTo(cm); - cm.Connections.Add(new ClientConnection("Azure.Messaging.ServiceBus.ServiceBusClient", $"https://{cm.Id}.servicebus.windows.net/")); + connections.Add(new ClientConnection("Azure.Messaging.ServiceBus.ServiceBusClient", $"https://{cmId}.servicebus.windows.net/")); } } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs index 35fc503dae275..5b5478b141ad4 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs @@ -11,7 +11,7 @@ namespace Azure.CloudMachine; public class ServiceBusSubscriptionFeature(string name, ServiceBusTopicFeature parent) : CloudMachineFeature { - protected override ProvisionableResource EmitCore(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure infrastructure) { var subscription = new ServiceBusSubscription(name, "2021-11-01") { @@ -33,13 +33,12 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf return subscription; } - protected internal override void AddTo(CloudMachineInfrastructure cm) + protected internal override void EmitConnections(ConnectionCollection connections, string cmId) { - base.AddTo(cm); ClientConnection connection = new( $"{name}", $"{parent.Name}/{name}" ); - cm.Connections.Add(connection); + connections.Add(connection); } } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs index e7e9f2dac9d45..254a4c4abe349 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs @@ -17,7 +17,7 @@ public ServiceBusTopicFeature(string name, ServiceBusNamespaceFeature parent) _parent = parent; } - protected override ProvisionableResource EmitCore(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure infrastructure) { var topic = new ServiceBusTopic(Name, "2021-11-01") { @@ -36,10 +36,9 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf return topic; } - protected internal override void AddTo(CloudMachineInfrastructure cm) + protected internal override void EmitConnections(ConnectionCollection connections, string cmId) { - base.AddTo(cm); - cm.Connections.Add(new ClientConnection(Name, Name)); + connections.Add(new ClientConnection(Name, Name)); } /// diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs index 4b010b4dd5a46..37ad4050cc948 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs @@ -30,7 +30,7 @@ public StorageFeature(string accountName, StorageSkuName sku = StorageSkuName.St _containerNames = ["default"]; } - protected override ProvisionableResource EmitCore(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure infrastructure) { var _storage = new StorageAccount("cm_storage", StorageAccount.ResourceVersions.V2023_01_01) @@ -79,9 +79,8 @@ protected override ProvisionableResource EmitCore(CloudMachineInfrastructure inf return _storage; } - protected internal override void AddTo(CloudMachineInfrastructure cm) + protected internal override void EmitConnections(ConnectionCollection connections, string cmId) { - base.AddTo(cm); - cm.Connections.Add(new ClientConnection("Azure.Storage.Blobs.BlobContainerClient@default", $"https://{cm.Id}.blob.core.windows.net/default")); + connections.Add(new ClientConnection("Azure.Storage.Blobs.BlobContainerClient@default", $"https://{cmId}.blob.core.windows.net/default")); } } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs index 13970df4ec2fa..a32154435d1e1 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs @@ -12,7 +12,7 @@ namespace Azure.CloudMachine; public class SystemTopicEventSubscriptionFeature(string name, EventGridSystemTopicFeature parent, ServiceBusTopicFeature destination, ServiceBusNamespaceFeature parentNamespace) : CloudMachineFeature { - protected override ProvisionableResource EmitCore(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure infrastructure) { ServiceBusNamespace serviceBusNamespace = EnsureEmits(parentNamespace); From 17cd244ea8e19e79a1dbb0bd832b0b77e917490e Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Wed, 27 Nov 2024 10:41:39 -0800 Subject: [PATCH 12/24] cleaned up tests --- .../tests/CloudMachineTests.cs | 210 +++++++++--------- .../tests/CloudMachineTests_rag.cs | 2 +- .../tests/GettingStarted.cs | 29 +++ 3 files changed, 136 insertions(+), 105 deletions(-) create mode 100644 sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs b/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs index d629098e63ca4..9546906bc19ae 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests.cs @@ -13,110 +13,112 @@ namespace Azure.CloudMachine.Tests; +// TODO: we need recordings to enable these tests + public partial class CloudMachineTests { - [Ignore("no recordings yet")] - [Test] - [TestCase([new string[] { "-bicep" }])] - [TestCase([new string[] { "" }])] - public void Storage(string[] args) - { - if (CloudMachineCommands.Execute(args, exitProcessIfHandled: false)) return; - - ManualResetEventSlim eventSlim = new(false); - CloudMachineClient cm = new(); - - cm.Storage.WhenUploaded((StorageFile file) => - { - var data = file.Download(); - Console.WriteLine(data.ToString()); - Assert.AreEqual("{\"Foo\":5,\"Bar\":true}", data.ToString()); - eventSlim.Set(); - }); - var uploaded = cm.Storage.UploadJson(new - { - Foo = 5, - Bar = true - }); - BinaryData downloaded = cm.Storage.Download(uploaded); - Console.WriteLine(downloaded.ToString()); - eventSlim.Wait(); - } - - [Ignore("no recordings yet")] - [Test] - [TestCase([new string[] { "-bicep" }])] - [TestCase([new string[] { "" }])] - public void OpenAI(string[] args) - { - if (CloudMachineCommands.Execute(args, (infrastructure) => - { - infrastructure.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); - }, exitProcessIfHandled: false)) return; - - CloudMachineWorkspace cm = new(); - ChatClient chat = cm.GetOpenAIChatClient(); - ChatCompletion completion = chat.CompleteChat("Is Azure programming easy?"); - Console.WriteLine(completion.AsText()); - } - - [Ignore("no recordings yet")] - [Test] - [TestCase([new string[] { "-bicep" }])] - [TestCase([new string[] { "" }])] - public void KeyVault(string[] args) - { - if (CloudMachineCommands.Execute(args, (cm) => - { - cm.AddFeature(new KeyVaultFeature()); - }, exitProcessIfHandled: false)) return; - - CloudMachineWorkspace cm = new(); - SecretClient secrets = cm.GetKeyVaultSecretsClient(); - secrets.SetSecret("testsecret", "don't tell anybody"); - } - - [Ignore("no recordings yet")] - [Test] - [TestCase([new string[] { "-bicep" }])] - [TestCase([new string[] { "" }])] - public void Messaging(string[] args) - { - CloudMachineCommands.Execute(args); - - CloudMachineClient cm = new(); - cm.Messaging.WhenMessageReceived(message => - { - Console.WriteLine(message); - Assert.True(message != null); - }); - cm.Messaging.SendJson(new - { - Foo = 5, - Bar = true - }); - } - - [Ignore("no recordings yet")] - [Test] - [TestCase([new string[] { "-bicep" }])] - [TestCase([new string[] { "" }])] - public void Demo(string[] args) - { - if (CloudMachineCommands.Execute(args, exitProcessIfHandled: false)) return; - - CloudMachineClient cm = new(); - - // setup - cm.Messaging.WhenMessageReceived((string message) => cm.Storage.Upload(BinaryData.FromString(message))); - cm.Storage.WhenUploaded((StorageFile file) => - { - var content = file.Download(); - ChatCompletion completion = cm.GetOpenAIChatClient().CompleteChat(content.ToString()); - Console.WriteLine(completion.Content[0].Text); - }); - - // go! - cm.Messaging.SendJson("Tell me something about Redmond, WA."); - } +// [Ignore("no recordings yet")] +// [Test] +// [TestCase([new string[] { "-bicep" }])] +// [TestCase([new string[] { "" }])] +// public void Storage(string[] args) +// { +// if (CloudMachineCommands.Execute(args, exitProcessIfHandled: false)) return; + +// ManualResetEventSlim eventSlim = new(false); +// CloudMachineClient cm = new(); + +// cm.Storage.WhenUploaded((StorageFile file) => +// { +// var data = file.Download(); +// Console.WriteLine(data.ToString()); +// Assert.AreEqual("{\"Foo\":5,\"Bar\":true}", data.ToString()); +// eventSlim.Set(); +// }); +// var uploaded = cm.Storage.UploadJson(new +// { +// Foo = 5, +// Bar = true +// }); +// BinaryData downloaded = cm.Storage.Download(uploaded); +// Console.WriteLine(downloaded.ToString()); +// eventSlim.Wait(); +// } + +// [Ignore("no recordings yet")] +// [Test] +// [TestCase([new string[] { "-bicep" }])] +// [TestCase([new string[] { "" }])] +// public void OpenAI(string[] args) +// { +// if (CloudMachineCommands.Execute(args, (infrastructure) => +// { +// infrastructure.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); +// }, exitProcessIfHandled: false)) return; + +// CloudMachineWorkspace cm = new(); +// ChatClient chat = cm.GetOpenAIChatClient(); +// ChatCompletion completion = chat.CompleteChat("Is Azure programming easy?"); +// Console.WriteLine(completion.AsText()); +// } + +// [Ignore("no recordings yet")] +// [Test] +// [TestCase([new string[] { "-bicep" }])] +// [TestCase([new string[] { "" }])] +// public void KeyVault(string[] args) +// { +// if (CloudMachineCommands.Execute(args, (cm) => +// { +// cm.AddFeature(new KeyVaultFeature()); +// }, exitProcessIfHandled: false)) return; + +// CloudMachineWorkspace cm = new(); +// SecretClient secrets = cm.GetKeyVaultSecretsClient(); +// secrets.SetSecret("testsecret", "don't tell anybody"); +// } + +// [Ignore("no recordings yet")] +// [Test] +// [TestCase([new string[] { "-bicep" }])] +// [TestCase([new string[] { "" }])] +// public void Messaging(string[] args) +// { +// CloudMachineCommands.Execute(args); + +// CloudMachineClient cm = new(); +// cm.Messaging.WhenMessageReceived(message => +// { +// Console.WriteLine(message); +// Assert.True(message != null); +// }); +// cm.Messaging.SendJson(new +// { +// Foo = 5, +// Bar = true +// }); +// } + +// [Ignore("no recordings yet")] +// [Test] +// [TestCase([new string[] { "-bicep" }])] +// [TestCase([new string[] { "" }])] +// public void Demo(string[] args) +// { +// if (CloudMachineCommands.Execute(args, exitProcessIfHandled: false)) return; + +// CloudMachineClient cm = new(); + +// // setup +// cm.Messaging.WhenMessageReceived((string message) => cm.Storage.Upload(BinaryData.FromString(message))); +// cm.Storage.WhenUploaded((StorageFile file) => +// { +// var content = file.Download(); +// ChatCompletion completion = cm.GetOpenAIChatClient().CompleteChat(content.ToString()); +// Console.WriteLine(completion.Content[0].Text); +// }); + +// // go! +// cm.Messaging.SendJson("Tell me something about Redmond, WA."); +// } } diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests_rag.cs b/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests_rag.cs index 9e10a425db168..c4061f87c55ff 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests_rag.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/CloudMachineTests_rag.cs @@ -24,7 +24,7 @@ public partial class CloudMachineTests { [Ignore("no recordings yet")] [Test] - public void EmbeddingsDemo() + public void RagDemo() { string[] testMessages = [ "When did the continuation policy change for DefaultAzureCredential?", diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs b/sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs new file mode 100644 index 0000000000000..aafe684038954 --- /dev/null +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs @@ -0,0 +1,29 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +#nullable enable + +using System; +using Azure.CloudMachine.OpenAI; +using NUnit.Framework; +using System.Linq; + +namespace Azure.CloudMachine.Tests; + +public partial class CloudMachineTests +{ + [Test] + [TestCase([new string[] { "-azd" }])] + public void GettingStarted(string[] args) + { + CloudMachineInfrastructure infra = new(); + infra.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); + + if (args.Contains("-bicep")) { + Azd.Init(infra); + return; + } + + CloudMachineClient client = infra.GetClient(); + } +} From 2b624d9be6507b29fb47e7c9ae5e726e7e20d7b2 Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Wed, 27 Nov 2024 10:54:04 -0800 Subject: [PATCH 13/24] hardened appsettings.json reading code --- .../Azure.CloudMachine/src/CloudMachineWorkspace.cs | 5 +++-- sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs | 1 + 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs index e2d42567b839e..8121d6339681c 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs @@ -113,10 +113,11 @@ internal static string ReadOrCreateCmid() writer.WriteEndObject(); writer.WriteEndObject(); writer.Flush(); + file.Close(); return cmid; } - using FileStream json = File.OpenRead(appsettings); + using FileStream json = new FileStream(appsettings, FileMode.Open, FileAccess.Read, FileShare.Read); using JsonDocument jd = JsonDocument.Parse(json); JsonElement je = jd.RootElement; // attempt to read CM configuration from existing configuration file @@ -144,7 +145,7 @@ internal static string ReadOrCreateCmid() cmProperties.Add("ID", cmid); obj.Add("CloudMachine", cmProperties); - using FileStream file = File.OpenWrite(appsettings); + using FileStream file = new FileStream(appsettings, FileMode.Open, FileAccess.Write, FileShare.None); JsonWriterOptions writerOptions = new() { Indented = true, diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs b/sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs index aafe684038954..b25a2ad0d91d8 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs @@ -24,6 +24,7 @@ public void GettingStarted(string[] args) return; } + // TODO: we need to allow newing up the client. CloudMachineClient client = infra.GetClient(); } } From cef84aa1352a5387877a3a29f85c493f43935497 Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Wed, 27 Nov 2024 10:58:14 -0800 Subject: [PATCH 14/24] added more to getting started --- .../src/extensions/AzureOpenAIExtensions.cs | 7 +++++++ .../Azure.CloudMachine/tests/GettingStarted.cs | 5 +++++ 2 files changed, 12 insertions(+) diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/AzureOpenAIExtensions.cs b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/AzureOpenAIExtensions.cs index 34b01587d57d0..741d05b5eeedc 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/extensions/AzureOpenAIExtensions.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/extensions/AzureOpenAIExtensions.cs @@ -48,6 +48,13 @@ public static EmbeddingClient GetOpenAIEmbeddingsClient(this ClientWorkspace wor return embeddingsClient; } + /// + /// returns full text of all parts. + /// + /// + public static string AsText(this ClientResult completionResult) + => AsText(completionResult.Value); + /// /// returns full text of all parts. /// diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs b/sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs index b25a2ad0d91d8..526828d4e9326 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs @@ -7,6 +7,7 @@ using Azure.CloudMachine.OpenAI; using NUnit.Framework; using System.Linq; +using OpenAI.Chat; namespace Azure.CloudMachine.Tests; @@ -26,5 +27,9 @@ public void GettingStarted(string[] args) // TODO: we need to allow newing up the client. CloudMachineClient client = infra.GetClient(); + + ChatClient chat = client.GetOpenAIChatClient(); + string completion = chat.CompleteChat("List all noble gases.").AsText(); + Console.WriteLine(completion); } } From c9ec840c1f4fd504629e1344423f3bead3eef8a4 Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Wed, 27 Nov 2024 11:20:58 -0800 Subject: [PATCH 15/24] started to move built-in connections --- .../src/CDKLevel3/CloudMachineConnections.cs | 21 +++++++++++++++++++ .../src/CDKLevel3/StorageFeature.cs | 5 ++--- 2 files changed, 23 insertions(+), 3 deletions(-) create mode 100644 sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineConnections.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineConnections.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineConnections.cs new file mode 100644 index 0000000000000..cd280067de2b6 --- /dev/null +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineConnections.cs @@ -0,0 +1,21 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +using Azure.Core; + +namespace Azure.CloudMachine.Core +{ + internal static class CloudMachineConnections + { + public static readonly string DefaultBlobContainerName = "default"; + + public static ClientConnection CreateDefaultBlobContainerConnection(string cmId) + { + ClientConnection connection = new( + $"Azure.Storage.Blobs.BlobContainerClient@{DefaultBlobContainerName}", + $"https://{cmId}.blob.core.windows.net/{DefaultBlobContainerName}" + ); + return connection; + } + } +} diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs index 37ad4050cc948..1d6302ad08d20 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs @@ -4,6 +4,7 @@ using System; using System.Collections.Generic; using System.Linq; +using Azure.CloudMachine.Core; using Azure.Core; using Azure.Provisioning; using Azure.Provisioning.CloudMachine; @@ -80,7 +81,5 @@ protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastr } protected internal override void EmitConnections(ConnectionCollection connections, string cmId) - { - connections.Add(new ClientConnection("Azure.Storage.Blobs.BlobContainerClient@default", $"https://{cmId}.blob.core.windows.net/default")); - } + => connections.Add(CloudMachineConnections.CreateDefaultBlobContainerConnection(cmId)); } From 82aacbf9cce349a916767412ea94fc2296a4f212 Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Wed, 27 Nov 2024 11:27:02 -0800 Subject: [PATCH 16/24] organized folders --- .../src/{CDKLevel3 => }/CloudMachineClientExtensions.cs | 0 .../CloudMachineFeature.cs | 0 .../CloudMachineInfrastructure.cs | 0 .../FeatureCollection.cs | 0 .../src/{CDKLevel3 => CloudMachineInfrastructure}/RoleResolver.cs | 0 .../src/{CDKLevel3 => FeaturesBuiltIn}/CloudMachineConnections.cs | 0 .../{CDKLevel3 => FeaturesBuiltIn}/EventGridSystemTopicFeature.cs | 0 .../{CDKLevel3 => FeaturesBuiltIn}/ServiceBusNamespaceFeature.cs | 0 .../ServiceBusSubscriptionFeature.cs | 0 .../src/{CDKLevel3 => FeaturesBuiltIn}/ServiceBusTopicFeature.cs | 0 .../src/{CDKLevel3 => FeaturesBuiltIn}/StorageFeature.cs | 0 .../SystemTopicEventSubscriptionFeature.cs | 0 .../AppServiceFeature.cs | 0 .../{AzureSdkExtensions => FeaturesExtensions}/KeyVaultFeature.cs | 0 .../{AzureSdkExtensions => FeaturesExtensions}/OpenAIFeature.cs | 0 .../OpenAIModelFeature.cs | 0 .../src/{CDKLevel3 => Tooling}/AppConfigHelpers.cs | 0 .../src/{CDKLevel3 => Tooling}/Azd.cs | 0 .../src/{CDKLevel3 => Tooling}/CloudMachineCommands.cs | 0 19 files changed, 0 insertions(+), 0 deletions(-) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{CDKLevel3 => }/CloudMachineClientExtensions.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{CDKLevel3 => CloudMachineInfrastructure}/CloudMachineFeature.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{CDKLevel3 => CloudMachineInfrastructure}/CloudMachineInfrastructure.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{CDKLevel3 => CloudMachineInfrastructure}/FeatureCollection.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{CDKLevel3 => CloudMachineInfrastructure}/RoleResolver.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{CDKLevel3 => FeaturesBuiltIn}/CloudMachineConnections.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{CDKLevel3 => FeaturesBuiltIn}/EventGridSystemTopicFeature.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{CDKLevel3 => FeaturesBuiltIn}/ServiceBusNamespaceFeature.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{CDKLevel3 => FeaturesBuiltIn}/ServiceBusSubscriptionFeature.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{CDKLevel3 => FeaturesBuiltIn}/ServiceBusTopicFeature.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{CDKLevel3 => FeaturesBuiltIn}/StorageFeature.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{CDKLevel3 => FeaturesBuiltIn}/SystemTopicEventSubscriptionFeature.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{AzureSdkExtensions => FeaturesExtensions}/AppServiceFeature.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{AzureSdkExtensions => FeaturesExtensions}/KeyVaultFeature.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{AzureSdkExtensions => FeaturesExtensions}/OpenAIFeature.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{AzureSdkExtensions => FeaturesExtensions}/OpenAIModelFeature.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{CDKLevel3 => Tooling}/AppConfigHelpers.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{CDKLevel3 => Tooling}/Azd.cs (100%) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{CDKLevel3 => Tooling}/CloudMachineCommands.cs (100%) diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineClientExtensions.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineClientExtensions.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineClientExtensions.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineClientExtensions.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineFeature.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineFeature.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineInfrastructure.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/FeatureCollection.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/FeatureCollection.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/FeatureCollection.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/FeatureCollection.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/RoleResolver.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/RoleResolver.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/RoleResolver.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/RoleResolver.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineConnections.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/CloudMachineConnections.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineConnections.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/CloudMachineConnections.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/EventGridSystemTopicFeature.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/EventGridSystemTopicFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/EventGridSystemTopicFeature.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusNamespaceFeature.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusNamespaceFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusNamespaceFeature.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusSubscriptionFeature.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusSubscriptionFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusSubscriptionFeature.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusTopicFeature.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/ServiceBusTopicFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusTopicFeature.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/StorageFeature.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/StorageFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/StorageFeature.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/SystemTopicEventSubscriptionFeature.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/SystemTopicEventSubscriptionFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/SystemTopicEventSubscriptionFeature.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/AppServiceFeature.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/AppServiceFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/AppServiceFeature.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/KeyVaultFeature.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/KeyVaultFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/KeyVaultFeature.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIFeature.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIFeature.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIModelFeature.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/AzureSdkExtensions/OpenAIModelFeature.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIModelFeature.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/AppConfigHelpers.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/AppConfigHelpers.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/AppConfigHelpers.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/AppConfigHelpers.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/Azd.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/Azd.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/Azd.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/Azd.cs diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/CloudMachineCommands.cs similarity index 100% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CDKLevel3/CloudMachineCommands.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/CloudMachineCommands.cs From 59b39099ae1fd872b38ad1899849099006196199 Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Wed, 27 Nov 2024 16:04:22 -0800 Subject: [PATCH 17/24] all tests pass --- .../tests/GettingStarted.cs | 2 +- .../CloudMachineConnections.cs | 9 + .../CloudMachineFeature.cs | 26 +- .../CloudMachineInfrastructure.cs | 9 +- .../FeatureCollection.cs | 14 +- .../EventGridSystemTopicFeature.cs | 11 +- .../src/FeaturesBuiltIn/InternalConstants.cs | 10 + .../ServiceBusNamespaceFeature.cs | 8 +- .../ServiceBusSubscriptionFeature.cs | 3 +- .../FeaturesBuiltIn/ServiceBusTopicFeature.cs | 3 +- .../FeaturesBuiltIn/StorageAccountFeature.cs | 103 ++++++ .../src/FeaturesBuiltIn/StorageFeature.cs | 85 ----- .../SystemTopicEventSubscriptionFeature.cs | 6 +- .../FeaturesExtensions/AppServiceFeature.cs | 2 +- .../src/FeaturesExtensions/KeyVaultFeature.cs | 2 +- .../src/FeaturesExtensions/OpenAIFeature.cs | 27 +- .../FeaturesExtensions/OpenAIModelFeature.cs | 41 ++- ...ure.Provisioning.CloudMachine.Tests.csproj | 3 + .../tests/Data/GenerateBicep.bicep | 2 - .../tests/Data/JustCloudMachine.bicep | 240 ++++++++++++++ .../tests/Data/OpenAI.bicep | 312 ++++++++++++++++++ .../tests/ProvisioningTests.cs | 33 +- 22 files changed, 792 insertions(+), 159 deletions(-) rename sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/{FeaturesBuiltIn => CloudMachineInfrastructure}/CloudMachineConnections.cs (67%) create mode 100644 sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/InternalConstants.cs create mode 100644 sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/StorageAccountFeature.cs delete mode 100644 sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/StorageFeature.cs create mode 100644 sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Data/JustCloudMachine.bicep create mode 100644 sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Data/OpenAI.bicep diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs b/sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs index 526828d4e9326..ee097e04ba377 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/GettingStarted.cs @@ -20,7 +20,7 @@ public void GettingStarted(string[] args) CloudMachineInfrastructure infra = new(); infra.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); - if (args.Contains("-bicep")) { + if (args.Contains("-azd")) { Azd.Init(infra); return; } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/CloudMachineConnections.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineConnections.cs similarity index 67% rename from sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/CloudMachineConnections.cs rename to sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineConnections.cs index cd280067de2b6..c86355df92bad 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/CloudMachineConnections.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineConnections.cs @@ -17,5 +17,14 @@ public static ClientConnection CreateDefaultBlobContainerConnection(string cmId) ); return connection; } + + public static ClientConnection CreateDefaultServiceBusConnection(string cmId) + { + ClientConnection connection = new( + "Azure.Messaging.ServiceBus.ServiceBusClient", + $"https://{cmId}.servicebus.windows.net/" + ); + return connection; + } } } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineFeature.cs index 97f552a08138d..31b2f30600a2a 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineFeature.cs @@ -11,20 +11,34 @@ namespace Azure.Provisioning.CloudMachine; public abstract class CloudMachineFeature { - protected abstract ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure cm); + private ProvisionableResource? _resource; + + protected abstract ProvisionableResource EmitConstructs(CloudMachineInfrastructure cm); protected internal virtual void EmitConnections(ConnectionCollection connections, string cmId) { } protected internal virtual void EmitFeatures(FeatureCollection features, string cmId) => features.Add(this); - internal void Emit(CloudMachineInfrastructure cm) + internal ProvisionableResource Emit(CloudMachineInfrastructure cm) { - if (Emitted != null) return; - ProvisionableResource provisionable = EmitInfrastructure(cm); - Emitted = provisionable; + if (_resource == null) + { + ProvisionableResource provisionable = EmitConstructs(cm); + _resource = provisionable; + } + return Emitted; } [EditorBrowsable(EditorBrowsableState.Never)] - public ProvisionableResource Emitted { get; protected set; } = default!; + public ProvisionableResource Emitted { + get + { + if (_resource == null) + { + throw new InvalidOperationException("Feature has not been emitted yet."); + } + return _resource; + } + } protected internal Dictionary RequiredSystemRoles { get; } = []; diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs index dc915b35596fe..abf9fa78fd6ca 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs @@ -14,9 +14,6 @@ namespace Azure.CloudMachine; public class CloudMachineInfrastructure { - internal const string SB_PRIVATE_TOPIC = "cm_servicebus_topic_private"; - internal const string SB_PRIVATE_SUB = "cm_servicebus_subscription_private"; - private readonly Infrastructure _infrastructure = new("cm"); private readonly List _constrcuts = []; @@ -56,13 +53,15 @@ public CloudMachineInfrastructure(string? cmId = default) _infrastructure.Add(new ProvisioningOutput("cm_managed_identity_id", typeof(string)) { Value = Identity.Id }); // Add core features - var storage = AddFeature(new StorageFeature(Id)); + var storage = AddFeature(new StorageAccountFeature(Id)); + var blobs = AddFeature(new BlobServiceFeature(storage)); + var defaultContainer = AddFeature(new BlobContainerFeature(blobs)); var sbNamespace = AddFeature(new ServiceBusNamespaceFeature(Id)); var sbTopicPrivate = AddFeature(new ServiceBusTopicFeature("cm_servicebus_topic_private", sbNamespace)); var sbTopicDefault = AddFeature(new ServiceBusTopicFeature("cm_servicebus_default_topic", sbNamespace)); AddFeature(new ServiceBusSubscriptionFeature("cm_servicebus_subscription_private", sbTopicPrivate)); // TODO: should private connections not be in the Connections collection? AddFeature(new ServiceBusSubscriptionFeature("cm_servicebus_subscription_default", sbTopicDefault)); - var systemTopic = AddFeature(new EventGridSystemTopicFeature(Id, storage)); + var systemTopic = AddFeature(new EventGridSystemTopicFeature(Id, storage, "Microsoft.Storage.StorageAccounts")); AddFeature(new SystemTopicEventSubscriptionFeature("cm_eventgrid_subscription_blob", systemTopic, sbTopicPrivate, sbNamespace)); } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/FeatureCollection.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/FeatureCollection.cs index 204a3045c720f..07e24da9bd102 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/FeatureCollection.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/FeatureCollection.cs @@ -2,6 +2,7 @@ // Licensed under the MIT License. using System; +using System.Collections; using System.Collections.Generic; using System.Linq; using Azure.Provisioning.CloudMachine; @@ -9,7 +10,7 @@ namespace Azure.CloudMachine; -public class FeatureCollection +public class FeatureCollection : IEnumerable { private CloudMachineFeature[] _items = new CloudMachineFeature[4]; private int _count; @@ -60,4 +61,15 @@ internal void Emit(CloudMachineInfrastructure infrastructure) feature.Emit(infrastructure); } } + + public IEnumerator GetEnumerator() + { + for (int i=0; i < _count; i++) + { + yield return _items[i]; + } + } + + IEnumerator IEnumerable.GetEnumerator() + => GetEnumerator(); } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/EventGridSystemTopicFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/EventGridSystemTopicFeature.cs index 7f505f0537c60..15eb89af792f9 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/EventGridSystemTopicFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/EventGridSystemTopicFeature.cs @@ -1,6 +1,7 @@ // Copyright (c) Microsoft Corporation. All rights reserved. // Licensed under the MIT License. +using Azure.CloudMachine.Core; using Azure.Provisioning.CloudMachine; using Azure.Provisioning.EventGrid; using Azure.Provisioning.Expressions; @@ -10,13 +11,13 @@ namespace Azure.CloudMachine; -public class EventGridSystemTopicFeature(string topicName, CloudMachineFeature source) : CloudMachineFeature +internal class EventGridSystemTopicFeature(string topicName, CloudMachineFeature source, string topicType) : CloudMachineFeature { - protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure infrastructure) { - var topic = new SystemTopic("cm_eventgrid_topic", "2022-06-15") + var topic = new SystemTopic("cm_eventgrid_topic", InternalConstants.EventGridTopicVersion) { - TopicType = "Microsoft.Storage.StorageAccounts", + TopicType = topicType, Source = EnsureEmits(source).Id, Identity = new() { @@ -27,8 +28,6 @@ protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastr }; infrastructure.AddConstruct(topic); - - Emitted = topic; return topic; } } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/InternalConstants.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/InternalConstants.cs new file mode 100644 index 0000000000000..6387aa81b0015 --- /dev/null +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/InternalConstants.cs @@ -0,0 +1,10 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +namespace Azure.CloudMachine.Core +{ + internal static class InternalConstants + { + internal const string EventGridTopicVersion = "2022-06-15"; + } +} diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusNamespaceFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusNamespaceFeature.cs index c04f504ab10b5..1da3accf1756b 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusNamespaceFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusNamespaceFeature.cs @@ -1,6 +1,7 @@ // Copyright (c) Microsoft Corporation. All rights reserved. // Licensed under the MIT License. +using Azure.CloudMachine.Core; using Azure.Core; using Azure.Provisioning.CloudMachine; using Azure.Provisioning.Primitives; @@ -10,7 +11,7 @@ namespace Azure.CloudMachine; public class ServiceBusNamespaceFeature(string name, ServiceBusSkuName sku = ServiceBusSkuName.Standard, ServiceBusSkuTier tier = ServiceBusSkuTier.Standard) : CloudMachineFeature { - protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure infrastructure) { var _serviceBusNamespace = new ServiceBusNamespace("cm_servicebus") { @@ -36,12 +37,9 @@ protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastr (ServiceBusBuiltInRole.GetBuiltInRoleName(ServiceBusBuiltInRole.AzureServiceBusDataOwner), ServiceBusBuiltInRole.AzureServiceBusDataOwner.ToString()), ]); - Emitted = _serviceBusNamespace; return _serviceBusNamespace; } protected internal override void EmitConnections(ConnectionCollection connections, string cmId) - { - connections.Add(new ClientConnection("Azure.Messaging.ServiceBus.ServiceBusClient", $"https://{cmId}.servicebus.windows.net/")); - } + => connections.Add(CloudMachineConnections.CreateDefaultServiceBusConnection(cmId)); } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusSubscriptionFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusSubscriptionFeature.cs index 5b5478b141ad4..71d069294a545 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusSubscriptionFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusSubscriptionFeature.cs @@ -11,7 +11,7 @@ namespace Azure.CloudMachine; public class ServiceBusSubscriptionFeature(string name, ServiceBusTopicFeature parent) : CloudMachineFeature { - protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure infrastructure) { var subscription = new ServiceBusSubscription(name, "2021-11-01") { @@ -29,7 +29,6 @@ protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastr }; infrastructure.AddConstruct(subscription); - Emitted = subscription; return subscription; } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusTopicFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusTopicFeature.cs index 254a4c4abe349..de468816cee6f 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusTopicFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusTopicFeature.cs @@ -17,7 +17,7 @@ public ServiceBusTopicFeature(string name, ServiceBusNamespaceFeature parent) _parent = parent; } - protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure infrastructure) { var topic = new ServiceBusTopic(Name, "2021-11-01") { @@ -32,7 +32,6 @@ protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastr }; infrastructure.AddConstruct(topic); - Emitted = topic; return topic; } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/StorageAccountFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/StorageAccountFeature.cs new file mode 100644 index 0000000000000..bb97b2393aa71 --- /dev/null +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/StorageAccountFeature.cs @@ -0,0 +1,103 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +using System.Collections.Generic; +using System.Linq; +using Azure.CloudMachine.Core; +using Azure.Core; +using Azure.Provisioning; +using Azure.Provisioning.CloudMachine; +using Azure.Provisioning.Expressions; +using Azure.Provisioning.Primitives; +using Azure.Provisioning.Resources; +using Azure.Provisioning.Storage; + +namespace Azure.CloudMachine; + +internal class StorageAccountFeature : CloudMachineFeature +{ + private readonly StorageSkuName _skuName; + private readonly string _name; + public StorageAccountFeature(string accountName, StorageSkuName sku = StorageSkuName.StandardLrs) + { + _skuName = sku; + _name = accountName; + } + + protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure infrastructure) + { + var storage = new StorageAccount("cm_storage", StorageAccount.ResourceVersions.V2023_01_01) + { + Name = _name, + Kind = StorageKind.StorageV2, + Sku = new StorageSku { Name = _skuName }, + IsHnsEnabled = true, + AllowBlobPublicAccess = false, + Identity = new() + { + ManagedServiceIdentityType = ManagedServiceIdentityType.UserAssigned, + UserAssignedIdentities = { { BicepFunction.Interpolate($"{infrastructure.Identity.Id}").Compile().ToString(), new UserAssignedIdentityDetails() } } + } + }; + infrastructure.AddConstruct(storage); + + RequiredSystemRoles.Add(storage, + [ + (StorageBuiltInRole.GetBuiltInRoleName(StorageBuiltInRole.StorageBlobDataContributor),StorageBuiltInRole.StorageBlobDataContributor.ToString()), + (StorageBuiltInRole.GetBuiltInRoleName(StorageBuiltInRole.StorageTableDataContributor), StorageBuiltInRole.StorageTableDataContributor.ToString()) + ] + ); + return storage; + } +} + +internal class BlobContainerFeature : CloudMachineFeature +{ + public string ContainerName { get; } + public BlobServiceFeature Parent { get; } + + public BlobContainerFeature(BlobServiceFeature parent, string? containerName = default) + { + if (containerName == default) containerName = CloudMachineConnections.DefaultBlobContainerName; + ContainerName = containerName; + Parent = parent; + } + protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure cm) + { + BlobContainer container = new($"cm_storage_blobs_container_{ContainerName}", "2023-01-01") + { + Parent = (BlobService)Parent.Emitted, + Name = ContainerName + }; + cm.AddConstruct(container); + return container; + } + + protected internal override void EmitConnections(ConnectionCollection connections, string cmId) + { + ClientConnection connection = new( + $"Azure.Storage.Blobs.BlobContainerClient@{ContainerName}", + $"https://{cmId}.blob.core.windows.net/{ContainerName}" + ); + connections.Add(connection); + } +} + +internal class BlobServiceFeature : CloudMachineFeature +{ + public StorageAccountFeature Account { get; } + + public BlobServiceFeature(StorageAccountFeature account) + { + Account = account; + } + protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure cm) + { + BlobService blobs = new("cm_storage_blobs") + { + Parent = (StorageAccount)Account.Emitted, + }; + cm.AddConstruct(blobs); + return blobs; + } +} diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/StorageFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/StorageFeature.cs deleted file mode 100644 index 1d6302ad08d20..0000000000000 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/StorageFeature.cs +++ /dev/null @@ -1,85 +0,0 @@ -// Copyright (c) Microsoft Corporation. All rights reserved. -// Licensed under the MIT License. - -using System; -using System.Collections.Generic; -using System.Linq; -using Azure.CloudMachine.Core; -using Azure.Core; -using Azure.Provisioning; -using Azure.Provisioning.CloudMachine; -using Azure.Provisioning.Expressions; -using Azure.Provisioning.Primitives; -using Azure.Provisioning.Resources; -using Azure.Provisioning.Storage; - -namespace Azure.CloudMachine; - -public class StorageFeature : CloudMachineFeature -{ - private readonly List _containerNames; - private readonly StorageSkuName _skuName; - private readonly string _name; - - public StorageFeature(string accountName, StorageSkuName sku = StorageSkuName.StandardLrs, IEnumerable? containerNames = null) - { - _skuName = sku; - _name = accountName; - if (containerNames != null) - _containerNames = containerNames.ToList(); - else - _containerNames = ["default"]; - } - - protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure infrastructure) - { - var _storage = - new StorageAccount("cm_storage", StorageAccount.ResourceVersions.V2023_01_01) - { - Name = _name, - Kind = StorageKind.StorageV2, - Sku = new StorageSku { Name = _skuName }, - IsHnsEnabled = true, - AllowBlobPublicAccess = false, - Identity = new() - { - ManagedServiceIdentityType = ManagedServiceIdentityType.UserAssigned, - UserAssignedIdentities = { { BicepFunction.Interpolate($"{infrastructure.Identity.Id}").Compile().ToString(), new UserAssignedIdentityDetails() } } - } - }; - infrastructure.AddConstruct(_storage); - - var _blobs = new BlobService("cm_storage_blobs") - { - Parent = _storage, - }; - infrastructure.AddConstruct(_blobs); - - foreach (var containerName in _containerNames) - { - infrastructure.AddConstruct( - new BlobContainer("cm_storage_blobs_container_" + containerName, "2023-01-01") - { - Parent = _blobs, - Name = containerName - } - ); - } - - RequiredSystemRoles.Add( - _storage, - [ - (StorageBuiltInRole.GetBuiltInRoleName(StorageBuiltInRole.StorageBlobDataContributor),StorageBuiltInRole.StorageBlobDataContributor.ToString()), - (StorageBuiltInRole.GetBuiltInRoleName(StorageBuiltInRole.StorageTableDataContributor), StorageBuiltInRole.StorageTableDataContributor.ToString()) - ]); - - // Placeholders for now. - infrastructure.AddConstruct(new ProvisioningOutput($"storage_name", typeof(string)) { Value = _storage.Name }); - - Emitted = _storage; - return _storage; - } - - protected internal override void EmitConnections(ConnectionCollection connections, string cmId) - => connections.Add(CloudMachineConnections.CreateDefaultBlobContainerConnection(cmId)); -} diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/SystemTopicEventSubscriptionFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/SystemTopicEventSubscriptionFeature.cs index a32154435d1e1..467159d98f0d8 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/SystemTopicEventSubscriptionFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/SystemTopicEventSubscriptionFeature.cs @@ -10,9 +10,9 @@ namespace Azure.CloudMachine; -public class SystemTopicEventSubscriptionFeature(string name, EventGridSystemTopicFeature parent, ServiceBusTopicFeature destination, ServiceBusNamespaceFeature parentNamespace) : CloudMachineFeature +internal class SystemTopicEventSubscriptionFeature(string name, EventGridSystemTopicFeature parent, ServiceBusTopicFeature destination, ServiceBusNamespaceFeature parentNamespace) : CloudMachineFeature { - protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure infrastructure) { ServiceBusNamespace serviceBusNamespace = EnsureEmits(parentNamespace); @@ -64,8 +64,6 @@ protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastr infrastructure.AddConstruct(subscription); infrastructure.AddConstruct(roleAssignment); - - Emitted = subscription; return subscription; } } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/AppServiceFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/AppServiceFeature.cs index 77be44f5904a3..a0c823686ae44 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/AppServiceFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/AppServiceFeature.cs @@ -18,7 +18,7 @@ public AppServiceFeature() Sku = new AppServiceSkuDescription { Tier = "Free", Name = "F1" }; } - protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure infrastructure) { //Add a App Service to the CloudMachine infrastructure. AppServicePlan hostingPlan = new("cm_hosting_plan") diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/KeyVaultFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/KeyVaultFeature.cs index 6cf7fb655196f..92e09a5b50a92 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/KeyVaultFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/KeyVaultFeature.cs @@ -25,7 +25,7 @@ protected internal override void EmitConnections(ConnectionCollection connection connections.Add(new ClientConnection("Azure.Security.KeyVault.Secrets.SecretClient", $"https://{cmId}.vault.azure.net/")); } - protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure infrastructure) { // Add a KeyVault to the CloudMachine infrastructure. KeyVaultService keyVaultResource = new("cm_kv") diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIFeature.cs index 97a1200360c38..4a5911e77b5f6 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIFeature.cs @@ -11,31 +11,16 @@ namespace Azure.CloudMachine.OpenAI; internal class OpenAIFeature : CloudMachineFeature { - private readonly List _models = []; - public OpenAIFeature() { } - protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure cloudMachine) + protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure cloudMachine) { CognitiveServicesAccount cognitiveServices = CreateOpenAIAccount(cloudMachine); cloudMachine.AddConstruct(cognitiveServices); RequiredSystemRoles.Add(cognitiveServices, [(CognitiveServicesBuiltInRole.GetBuiltInRoleName(CognitiveServicesBuiltInRole.CognitiveServicesOpenAIContributor) ,CognitiveServicesBuiltInRole.CognitiveServicesOpenAIContributor.ToString())]); - Emitted = cognitiveServices; - - OpenAIModelFeature? previous = null; - foreach (OpenAIModelFeature model in _models) - { - model.Emit(cloudMachine); - if (previous != null) - { - model.Emitted.DependsOn.Add(previous.Emitted); - } - previous = model; - } - return cognitiveServices; } @@ -49,16 +34,6 @@ protected internal override void EmitConnections(ConnectionCollection connection } } - internal void AddModel(OpenAIModelFeature model) - { - if (model.Account != null) - { - throw new InvalidOperationException("Model already added to an account"); - } - model.Account = this; - _models.Add(model); - } - internal static CognitiveServicesAccount CreateOpenAIAccount(CloudMachineInfrastructure cm) { return new("openai") diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIModelFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIModelFeature.cs index e3b269fe61b05..ce5346dd4141e 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIModelFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIModelFeature.cs @@ -34,7 +34,8 @@ protected internal override void EmitFeatures(FeatureCollection features, string openAI = new OpenAIFeature(); // TODO: we need to add connection features.Add(openAI); } - openAI.AddModel(this); + Account = openAI; + features.Add(this); } protected internal override void EmitConnections(ConnectionCollection connections, string cmId) @@ -54,8 +55,11 @@ protected internal override void EmitConnections(ConnectionCollection connection } } - protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastructure cm) + protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure cm) { + if (Account == null) throw new InvalidOperationException("Account must be set before emitting"); + if (Account.Emitted == null) throw new InvalidOperationException("Account must be emitted before emitting"); + string name = Kind switch { AIModelKind.Chat => $"{cm.Id}_chat", @@ -63,13 +67,7 @@ protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastr _ => throw new NotImplementedException() }; - Debug.Assert(Account != null); - ProvisionableResource emitted = Account!.Emitted; - if (emitted == null) - { - Account.Emit(cm); - } - CognitiveServicesAccount parent = (CognitiveServicesAccount)Account!.Emitted; + CognitiveServicesAccount parent = (CognitiveServicesAccount)Account.Emitted; CognitiveServicesAccountDeployment deployment = new($"openai_{name}", "2024-06-01-preview") { Parent = parent, @@ -89,12 +87,33 @@ protected override ProvisionableResource EmitInfrastructure(CloudMachineInfrastr { Capacity = 120, Name = "Standard" - } + }, }; - cm.AddConstruct(deployment); + // deployments need to have dependson set! + OpenAIModelFeature? previous = FindPrevious(cm, this); + if (previous != null) + { + if (previous.Emitted == null) throw new InvalidOperationException("Previous must be emitted"); + CognitiveServicesAccountDeployment previousDeployment = (CognitiveServicesAccountDeployment)previous.Emitted; + deployment.DependsOn.Add(previousDeployment); + } + cm.AddConstruct(deployment); return deployment; + + OpenAIModelFeature? FindPrevious(CloudMachineInfrastructure cm, OpenAIModelFeature current) + { + OpenAIModelFeature? previous = default; + foreach (var feature in cm.Features) + { + if (feature == current) + return previous; + if (feature is OpenAIModelFeature oaim) + previous = oaim; + } + throw new InvalidOperationException("current not found in infrastructure"); + } } } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Azure.Provisioning.CloudMachine.Tests.csproj b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Azure.Provisioning.CloudMachine.Tests.csproj index 44a160bb6cac6..9393c88763dbb 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Azure.Provisioning.CloudMachine.Tests.csproj +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Azure.Provisioning.CloudMachine.Tests.csproj @@ -16,5 +16,8 @@ + + Always + diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Data/GenerateBicep.bicep b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Data/GenerateBicep.bicep index 1bbb649a44b4f..eebc294e692e1 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Data/GenerateBicep.bicep +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Data/GenerateBicep.bicep @@ -396,5 +396,3 @@ resource cm_website 'Microsoft.Web/sites@2024-04-01' = { } output cm_managed_identity_id string = cm_identity.id - -output storage_name string = cm_storage.name \ No newline at end of file diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Data/JustCloudMachine.bicep b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Data/JustCloudMachine.bicep new file mode 100644 index 0000000000000..4657dd5b9f348 --- /dev/null +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Data/JustCloudMachine.bicep @@ -0,0 +1,240 @@ +@description('The location for the resource(s) to be deployed.') +param location string = resourceGroup().location + +@description('The objectId of the current user principal.') +param principalId string + +resource cm_identity 'Microsoft.ManagedIdentity/userAssignedIdentities@2023-01-31' = { + name: 'cm0c420d2f21084cd' + location: location +} + +resource cm_storage 'Microsoft.Storage/storageAccounts@2023-01-01' = { + name: 'cm0c420d2f21084cd' + kind: 'StorageV2' + location: location + sku: { + name: 'Standard_LRS' + } + properties: { + allowBlobPublicAccess: false + isHnsEnabled: true + } + identity: { + type: 'UserAssigned' + userAssignedIdentities: { + '${cm_identity.id}': { } + } + } +} + +resource cm_storage_00000000_0000_0000_0000_000000000000_StorageBlobDataContributor 'Microsoft.Authorization/roleAssignments@2022-04-01' = { + name: guid('cm_storage', principalId, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', 'ba92f5b4-2d11-453d-a403-e96b0029c9fe')) + properties: { + principalId: principalId + roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', 'ba92f5b4-2d11-453d-a403-e96b0029c9fe') + principalType: 'User' + } + scope: cm_storage +} + +resource cm_storage_cm_identity_StorageBlobDataContributor 'Microsoft.Authorization/roleAssignments@2022-04-01' = { + name: guid('cm_storage', cm_identity.id, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', 'ba92f5b4-2d11-453d-a403-e96b0029c9fe')) + properties: { + principalId: cm_identity.properties.principalId + roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', 'ba92f5b4-2d11-453d-a403-e96b0029c9fe') + principalType: 'ServicePrincipal' + } + scope: cm_storage +} + +resource cm_storage_00000000_0000_0000_0000_000000000000_StorageTableDataContributor 'Microsoft.Authorization/roleAssignments@2022-04-01' = { + name: guid('cm_storage', principalId, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '0a9a7e1f-b9d0-4cc4-a60d-0319b160aaa3')) + properties: { + principalId: principalId + roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '0a9a7e1f-b9d0-4cc4-a60d-0319b160aaa3') + principalType: 'User' + } + scope: cm_storage +} + +resource cm_storage_cm_identity_StorageTableDataContributor 'Microsoft.Authorization/roleAssignments@2022-04-01' = { + name: guid('cm_storage', cm_identity.id, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '0a9a7e1f-b9d0-4cc4-a60d-0319b160aaa3')) + properties: { + principalId: cm_identity.properties.principalId + roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '0a9a7e1f-b9d0-4cc4-a60d-0319b160aaa3') + principalType: 'ServicePrincipal' + } + scope: cm_storage +} + +resource cm_storage_blobs 'Microsoft.Storage/storageAccounts/blobServices@2024-01-01' = { + name: 'default' + parent: cm_storage +} + +resource cm_storage_blobs_container_default 'Microsoft.Storage/storageAccounts/blobServices/containers@2023-01-01' = { + name: 'default' + parent: cm_storage_blobs +} + +resource cm_servicebus 'Microsoft.ServiceBus/namespaces@2024-01-01' = { + name: 'cm0c420d2f21084cd' + location: location + sku: { + name: 'Standard' + tier: 'Standard' + } +} + +resource cm_servicebus_00000000_0000_0000_0000_000000000000_AzureServiceBusDataOwner 'Microsoft.Authorization/roleAssignments@2022-04-01' = { + name: guid('cm_servicebus', principalId, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '090c5cfd-751d-490a-894a-3ce6f1109419')) + properties: { + principalId: principalId + roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '090c5cfd-751d-490a-894a-3ce6f1109419') + principalType: 'User' + } + scope: cm_servicebus +} + +resource cm_servicebus_cm_identity_AzureServiceBusDataOwner 'Microsoft.Authorization/roleAssignments@2022-04-01' = { + name: guid('cm_servicebus', cm_identity.id, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '090c5cfd-751d-490a-894a-3ce6f1109419')) + properties: { + principalId: cm_identity.properties.principalId + roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '090c5cfd-751d-490a-894a-3ce6f1109419') + principalType: 'ServicePrincipal' + } + scope: cm_servicebus +} + +resource cm_servicebus_auth_rule 'Microsoft.ServiceBus/namespaces/AuthorizationRules@2021-11-01' = { + name: take('cm_servicebus_auth_rule-${uniqueString(resourceGroup().id)}', 50) + properties: { + rights: [ + 'Listen' + 'Send' + 'Manage' + ] + } + parent: cm_servicebus +} + +resource cm_servicebus_topic_private 'Microsoft.ServiceBus/namespaces/topics@2021-11-01' = { + name: 'cm_servicebus_topic_private' + properties: { + defaultMessageTimeToLive: 'P14D' + enableBatchedOperations: true + maxMessageSizeInKilobytes: 256 + requiresDuplicateDetection: false + status: 'Active' + supportOrdering: true + } + parent: cm_servicebus +} + +resource cm_servicebus_default_topic 'Microsoft.ServiceBus/namespaces/topics@2021-11-01' = { + name: 'cm_servicebus_default_topic' + properties: { + defaultMessageTimeToLive: 'P14D' + enableBatchedOperations: true + maxMessageSizeInKilobytes: 256 + requiresDuplicateDetection: false + status: 'Active' + supportOrdering: true + } + parent: cm_servicebus +} + +resource cm_servicebus_subscription_private 'Microsoft.ServiceBus/namespaces/topics/subscriptions@2021-11-01' = { + name: 'cm_servicebus_subscription_private' + properties: { + deadLetteringOnFilterEvaluationExceptions: true + deadLetteringOnMessageExpiration: true + defaultMessageTimeToLive: 'P14D' + enableBatchedOperations: true + isClientAffine: false + lockDuration: 'PT30S' + maxDeliveryCount: 10 + requiresSession: false + status: 'Active' + } + parent: cm_servicebus_topic_private +} + +resource cm_servicebus_subscription_default 'Microsoft.ServiceBus/namespaces/topics/subscriptions@2021-11-01' = { + name: 'cm_servicebus_subscription_default' + properties: { + deadLetteringOnFilterEvaluationExceptions: true + deadLetteringOnMessageExpiration: true + defaultMessageTimeToLive: 'P14D' + enableBatchedOperations: true + isClientAffine: false + lockDuration: 'PT30S' + maxDeliveryCount: 10 + requiresSession: false + status: 'Active' + } + parent: cm_servicebus_default_topic +} + +resource cm_eventgrid_topic 'Microsoft.EventGrid/systemTopics@2022-06-15' = { + name: 'cm0c420d2f21084cd' + location: location + identity: { + type: 'UserAssigned' + userAssignedIdentities: { + '${cm_identity.id}': { } + } + } + properties: { + source: cm_storage.id + topicType: 'Microsoft.Storage.StorageAccounts' + } +} + +resource cm_eventgrid_subscription_blob 'Microsoft.EventGrid/systemTopics/eventSubscriptions@2022-06-15' = { + name: 'cm_eventgrid_subscription_blob' + properties: { + deliveryWithResourceIdentity: { + identity: { + type: 'UserAssigned' + userAssignedIdentity: cm_identity.id + } + destination: { + endpointType: 'ServiceBusTopic' + properties: { + resourceId: cm_servicebus_topic_private.id + } + } + } + eventDeliverySchema: 'EventGridSchema' + filter: { + includedEventTypes: [ + 'Microsoft.Storage.BlobCreated' + 'Microsoft.Storage.BlobDeleted' + 'Microsoft.Storage.BlobRenamed' + ] + enableAdvancedFilteringOnArrays: true + } + retryPolicy: { + maxDeliveryAttempts: 30 + eventTimeToLiveInMinutes: 1440 + } + } + parent: cm_eventgrid_topic + dependsOn: [ + cm_servicebus_cm0c420d2f21084cd_role + ] +} + +resource cm_servicebus_cm0c420d2f21084cd_role 'Microsoft.Authorization/roleAssignments@2022-04-01' = { + name: guid(cm_servicebus.id, cm_identity.id, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '69a216fc-b8fb-44d8-bc22-1f3c2cd27a39')) + properties: { + principalId: cm_identity.properties.principalId + roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '69a216fc-b8fb-44d8-bc22-1f3c2cd27a39') + principalType: 'ServicePrincipal' + } + scope: cm_servicebus +} + +output cm_managed_identity_id string = cm_identity.id diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Data/OpenAI.bicep b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Data/OpenAI.bicep new file mode 100644 index 0000000000000..ef7b1529ef131 --- /dev/null +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/Data/OpenAI.bicep @@ -0,0 +1,312 @@ +@description('The location for the resource(s) to be deployed.') +param location string = resourceGroup().location + +@description('The objectId of the current user principal.') +param principalId string + +resource cm_identity 'Microsoft.ManagedIdentity/userAssignedIdentities@2023-01-31' = { + name: 'cm0c420d2f21084cd' + location: location +} + +resource cm_storage 'Microsoft.Storage/storageAccounts@2023-01-01' = { + name: 'cm0c420d2f21084cd' + kind: 'StorageV2' + location: location + sku: { + name: 'Standard_LRS' + } + properties: { + allowBlobPublicAccess: false + isHnsEnabled: true + } + identity: { + type: 'UserAssigned' + userAssignedIdentities: { + '${cm_identity.id}': { } + } + } +} + +resource cm_storage_00000000_0000_0000_0000_000000000000_StorageBlobDataContributor 'Microsoft.Authorization/roleAssignments@2022-04-01' = { + name: guid('cm_storage', principalId, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', 'ba92f5b4-2d11-453d-a403-e96b0029c9fe')) + properties: { + principalId: principalId + roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', 'ba92f5b4-2d11-453d-a403-e96b0029c9fe') + principalType: 'User' + } + scope: cm_storage +} + +resource cm_storage_cm_identity_StorageBlobDataContributor 'Microsoft.Authorization/roleAssignments@2022-04-01' = { + name: guid('cm_storage', cm_identity.id, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', 'ba92f5b4-2d11-453d-a403-e96b0029c9fe')) + properties: { + principalId: cm_identity.properties.principalId + roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', 'ba92f5b4-2d11-453d-a403-e96b0029c9fe') + principalType: 'ServicePrincipal' + } + scope: cm_storage +} + +resource cm_storage_00000000_0000_0000_0000_000000000000_StorageTableDataContributor 'Microsoft.Authorization/roleAssignments@2022-04-01' = { + name: guid('cm_storage', principalId, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '0a9a7e1f-b9d0-4cc4-a60d-0319b160aaa3')) + properties: { + principalId: principalId + roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '0a9a7e1f-b9d0-4cc4-a60d-0319b160aaa3') + principalType: 'User' + } + scope: cm_storage +} + +resource cm_storage_cm_identity_StorageTableDataContributor 'Microsoft.Authorization/roleAssignments@2022-04-01' = { + name: guid('cm_storage', cm_identity.id, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '0a9a7e1f-b9d0-4cc4-a60d-0319b160aaa3')) + properties: { + principalId: cm_identity.properties.principalId + roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '0a9a7e1f-b9d0-4cc4-a60d-0319b160aaa3') + principalType: 'ServicePrincipal' + } + scope: cm_storage +} + +resource cm_storage_blobs 'Microsoft.Storage/storageAccounts/blobServices@2024-01-01' = { + name: 'default' + parent: cm_storage +} + +resource cm_storage_blobs_container_default 'Microsoft.Storage/storageAccounts/blobServices/containers@2023-01-01' = { + name: 'default' + parent: cm_storage_blobs +} + +resource cm_servicebus 'Microsoft.ServiceBus/namespaces@2024-01-01' = { + name: 'cm0c420d2f21084cd' + location: location + sku: { + name: 'Standard' + tier: 'Standard' + } +} + +resource cm_servicebus_00000000_0000_0000_0000_000000000000_AzureServiceBusDataOwner 'Microsoft.Authorization/roleAssignments@2022-04-01' = { + name: guid('cm_servicebus', principalId, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '090c5cfd-751d-490a-894a-3ce6f1109419')) + properties: { + principalId: principalId + roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '090c5cfd-751d-490a-894a-3ce6f1109419') + principalType: 'User' + } + scope: cm_servicebus +} + +resource cm_servicebus_cm_identity_AzureServiceBusDataOwner 'Microsoft.Authorization/roleAssignments@2022-04-01' = { + name: guid('cm_servicebus', cm_identity.id, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '090c5cfd-751d-490a-894a-3ce6f1109419')) + properties: { + principalId: cm_identity.properties.principalId + roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '090c5cfd-751d-490a-894a-3ce6f1109419') + principalType: 'ServicePrincipal' + } + scope: cm_servicebus +} + +resource cm_servicebus_auth_rule 'Microsoft.ServiceBus/namespaces/AuthorizationRules@2021-11-01' = { + name: take('cm_servicebus_auth_rule-${uniqueString(resourceGroup().id)}', 50) + properties: { + rights: [ + 'Listen' + 'Send' + 'Manage' + ] + } + parent: cm_servicebus +} + +resource cm_servicebus_topic_private 'Microsoft.ServiceBus/namespaces/topics@2021-11-01' = { + name: 'cm_servicebus_topic_private' + properties: { + defaultMessageTimeToLive: 'P14D' + enableBatchedOperations: true + maxMessageSizeInKilobytes: 256 + requiresDuplicateDetection: false + status: 'Active' + supportOrdering: true + } + parent: cm_servicebus +} + +resource cm_servicebus_default_topic 'Microsoft.ServiceBus/namespaces/topics@2021-11-01' = { + name: 'cm_servicebus_default_topic' + properties: { + defaultMessageTimeToLive: 'P14D' + enableBatchedOperations: true + maxMessageSizeInKilobytes: 256 + requiresDuplicateDetection: false + status: 'Active' + supportOrdering: true + } + parent: cm_servicebus +} + +resource cm_servicebus_subscription_private 'Microsoft.ServiceBus/namespaces/topics/subscriptions@2021-11-01' = { + name: 'cm_servicebus_subscription_private' + properties: { + deadLetteringOnFilterEvaluationExceptions: true + deadLetteringOnMessageExpiration: true + defaultMessageTimeToLive: 'P14D' + enableBatchedOperations: true + isClientAffine: false + lockDuration: 'PT30S' + maxDeliveryCount: 10 + requiresSession: false + status: 'Active' + } + parent: cm_servicebus_topic_private +} + +resource cm_servicebus_subscription_default 'Microsoft.ServiceBus/namespaces/topics/subscriptions@2021-11-01' = { + name: 'cm_servicebus_subscription_default' + properties: { + deadLetteringOnFilterEvaluationExceptions: true + deadLetteringOnMessageExpiration: true + defaultMessageTimeToLive: 'P14D' + enableBatchedOperations: true + isClientAffine: false + lockDuration: 'PT30S' + maxDeliveryCount: 10 + requiresSession: false + status: 'Active' + } + parent: cm_servicebus_default_topic +} + +resource cm_eventgrid_topic 'Microsoft.EventGrid/systemTopics@2022-06-15' = { + name: 'cm0c420d2f21084cd' + location: location + identity: { + type: 'UserAssigned' + userAssignedIdentities: { + '${cm_identity.id}': { } + } + } + properties: { + source: cm_storage.id + topicType: 'Microsoft.Storage.StorageAccounts' + } +} + +resource cm_eventgrid_subscription_blob 'Microsoft.EventGrid/systemTopics/eventSubscriptions@2022-06-15' = { + name: 'cm_eventgrid_subscription_blob' + properties: { + deliveryWithResourceIdentity: { + identity: { + type: 'UserAssigned' + userAssignedIdentity: cm_identity.id + } + destination: { + endpointType: 'ServiceBusTopic' + properties: { + resourceId: cm_servicebus_topic_private.id + } + } + } + eventDeliverySchema: 'EventGridSchema' + filter: { + includedEventTypes: [ + 'Microsoft.Storage.BlobCreated' + 'Microsoft.Storage.BlobDeleted' + 'Microsoft.Storage.BlobRenamed' + ] + enableAdvancedFilteringOnArrays: true + } + retryPolicy: { + maxDeliveryAttempts: 30 + eventTimeToLiveInMinutes: 1440 + } + } + parent: cm_eventgrid_topic + dependsOn: [ + cm_servicebus_cm0c420d2f21084cd_role + ] +} + +resource cm_servicebus_cm0c420d2f21084cd_role 'Microsoft.Authorization/roleAssignments@2022-04-01' = { + name: guid(cm_servicebus.id, cm_identity.id, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '69a216fc-b8fb-44d8-bc22-1f3c2cd27a39')) + properties: { + principalId: cm_identity.properties.principalId + roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', '69a216fc-b8fb-44d8-bc22-1f3c2cd27a39') + principalType: 'ServicePrincipal' + } + scope: cm_servicebus +} + +resource openai 'Microsoft.CognitiveServices/accounts@2024-10-01' = { + name: 'cm0c420d2f21084cd' + location: location + kind: 'OpenAI' + properties: { + customSubDomainName: 'cm0c420d2f21084cd' + publicNetworkAccess: 'Enabled' + } + sku: { + name: 'S0' + } +} + +resource openai_00000000_0000_0000_0000_000000000000_CognitiveServicesOpenAIContributor 'Microsoft.Authorization/roleAssignments@2022-04-01' = { + name: guid('openai', principalId, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', 'a001fd3d-188f-4b5d-821b-7da978bf7442')) + properties: { + principalId: principalId + roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', 'a001fd3d-188f-4b5d-821b-7da978bf7442') + principalType: 'User' + } + scope: openai +} + +resource openai_cm_identity_CognitiveServicesOpenAIContributor 'Microsoft.Authorization/roleAssignments@2022-04-01' = { + name: guid('openai', cm_identity.id, subscriptionResourceId('Microsoft.Authorization/roleDefinitions', 'a001fd3d-188f-4b5d-821b-7da978bf7442')) + properties: { + principalId: cm_identity.properties.principalId + roleDefinitionId: subscriptionResourceId('Microsoft.Authorization/roleDefinitions', 'a001fd3d-188f-4b5d-821b-7da978bf7442') + principalType: 'ServicePrincipal' + } + scope: openai +} + +resource openai_cm0c420d2f21084cd_chat 'Microsoft.CognitiveServices/accounts/deployments@2024-06-01-preview' = { + name: 'cm0c420d2f21084cd_chat' + properties: { + model: { + format: 'OpenAI' + name: 'gpt-35-turbo' + version: '0125' + } + raiPolicyName: 'Microsoft.DefaultV2' + versionUpgradeOption: 'OnceNewDefaultVersionAvailable' + } + sku: { + name: 'Standard' + capacity: 120 + } + parent: openai +} + +resource openai_cm0c420d2f21084cd_embedding 'Microsoft.CognitiveServices/accounts/deployments@2024-06-01-preview' = { + name: 'cm0c420d2f21084cd_embedding' + properties: { + model: { + format: 'OpenAI' + name: 'text-embedding-ada-002' + version: '2' + } + raiPolicyName: 'Microsoft.DefaultV2' + versionUpgradeOption: 'OnceNewDefaultVersionAvailable' + } + sku: { + name: 'Standard' + capacity: 120 + } + parent: openai + dependsOn: [ + openai_cm0c420d2f21084cd_chat + ] +} + +output cm_managed_identity_id string = cm_identity.id diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/ProvisioningTests.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/ProvisioningTests.cs index c1867e84a6829..56ed939ceed94 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/ProvisioningTests.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/ProvisioningTests.cs @@ -15,6 +15,27 @@ namespace Azure.CloudMachine.Tests; public class ProvisioningTests { + [Test] + public void JustCloudMachine() + { + CloudMachineInfrastructure infra = new("cm0c420d2f21084cd"); + string actualBicep = infra.Build().Compile().FirstOrDefault().Value; + string expectedBicep = LoadTestFile("JustCloudMachine.bicep"); + Assert.AreEqual(expectedBicep, actualBicep); + } + + [Test] + public void OpenAI() + { + CloudMachineInfrastructure infra = new("cm0c420d2f21084cd"); + infra.AddFeature(new OpenAIModelFeature("gpt-35-turbo", "0125")); + infra.AddFeature(new OpenAIModelFeature("text-embedding-ada-002", "2", AIModelKind.Embedding)); + + string actualBicep = infra.Build().Compile().FirstOrDefault().Value; + string expectedBicep = LoadTestFile("OpenAI.bicep"); + Assert.AreEqual(expectedBicep, actualBicep); + } + [Test] public void GenerateBicep() { @@ -25,7 +46,17 @@ public void GenerateBicep() infra.AddFeature(new AppServiceFeature()); string actualBicep = infra.Build().Compile().FirstOrDefault().Value; - string expectedBicep = File.ReadAllText(Path.Combine(TestContext.CurrentContext.TestDirectory, "Data", "GenerateBicep.bicep")).Replace("\r\n", Environment.NewLine); + string expectedBicep = LoadTestFile("GenerateBicep.bicep"); Assert.AreEqual(expectedBicep, actualBicep); } + + private static string LoadTestFile(string filename) + { + string contents = File.ReadAllText(Path.Combine(TestContext.CurrentContext.TestDirectory, "Data", filename)); + contents = contents.Replace("\r\n", Environment.NewLine); + while (contents.EndsWith(Environment.NewLine)) { + contents = contents.Substring(0, contents.Length - Environment.NewLine.Length); + } + return contents; + } } From 25a8544fcce5bd51f459b75af0c89bb38424fddb Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Wed, 27 Nov 2024 16:10:26 -0800 Subject: [PATCH 18/24] updated APIs --- .../api/Azure.CloudMachine.net8.0.cs | 96 ++++++++++----- .../api/Azure.CloudMachine.netstandard2.0.cs | 96 ++++++++++----- .../Azure.Provisioning.CloudMachine.net8.0.cs | 112 +++++++++--------- ...rovisioning.CloudMachine.netstandard2.0.cs | 112 +++++++++--------- 4 files changed, 248 insertions(+), 168 deletions(-) diff --git a/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.net8.0.cs b/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.net8.0.cs index 2bfb5ef447534..09f4d6544fdc0 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.net8.0.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.net8.0.cs @@ -1,26 +1,65 @@ +namespace Azure +{ + public partial class RestCallFailedException : System.Exception + { + public RestCallFailedException(string message, System.ClientModel.Primitives.PipelineResponse response) { } + } + public partial class RestClient + { + public RestClient() { } + public RestClient(System.ClientModel.Primitives.PipelinePolicy auth) { } + public static Azure.RestClient Shared { get { throw null; } } + public System.ClientModel.Primitives.PipelineMessage Create(string method, System.Uri uri) { throw null; } + public System.ClientModel.Primitives.PipelineResponse Get(string uri, System.ClientModel.Primitives.RequestOptions options = null) { throw null; } + public System.ClientModel.Primitives.PipelineResponse Patch(string uri, System.ClientModel.BinaryContent content, System.ClientModel.Primitives.RequestOptions options = null) { throw null; } + public System.ClientModel.Primitives.PipelineResponse Post(string uri, System.ClientModel.BinaryContent content, System.ClientModel.Primitives.RequestOptions options = null) { throw null; } + public System.ClientModel.Primitives.PipelineResponse Put(string uri, System.ClientModel.BinaryContent content, System.ClientModel.Primitives.RequestOptions options = null) { throw null; } + public System.ClientModel.Primitives.PipelineResponse Send(System.ClientModel.Primitives.PipelineMessage message, System.ClientModel.Primitives.RequestOptions options = null) { throw null; } + } + public partial class RestClientOptions : System.ClientModel.Primitives.ClientPipelineOptions + { + public RestClientOptions() { } + } +} +namespace Azure.AI.OpenAI +{ + public partial class TokenCredentialAuthenticationPolicy : System.ClientModel.Primitives.PipelinePolicy + { + public TokenCredentialAuthenticationPolicy(Azure.Core.TokenCredential credential, System.Collections.Generic.IEnumerable scopes, System.TimeSpan? refreshOffset = default(System.TimeSpan?)) { } + public override void Process(System.ClientModel.Primitives.PipelineMessage message, System.Collections.Generic.IReadOnlyList pipeline, int currentIndex) { } + public override System.Threading.Tasks.ValueTask ProcessAsync(System.ClientModel.Primitives.PipelineMessage message, System.Collections.Generic.IReadOnlyList pipeline, int currentIndex) { throw null; } + } +} namespace Azure.CloudMachine { public partial class CloudMachineClient : Azure.CloudMachine.CloudMachineWorkspace { - protected CloudMachineClient() : base (default(Azure.Core.TokenCredential), default(Microsoft.Extensions.Configuration.IConfiguration)) { } - public CloudMachineClient(Azure.Core.TokenCredential credential = null, Microsoft.Extensions.Configuration.IConfiguration configuration = null) : base (default(Azure.Core.TokenCredential), default(Microsoft.Extensions.Configuration.IConfiguration)) { } + protected CloudMachineClient() : base (default(Azure.Core.TokenCredential), default(Microsoft.Extensions.Configuration.IConfiguration), default(System.Collections.Generic.IEnumerable)) { } + public CloudMachineClient(Azure.Core.TokenCredential credential = null, Microsoft.Extensions.Configuration.IConfiguration configuration = null, System.Collections.Generic.IEnumerable connections = null) : base (default(Azure.Core.TokenCredential), default(Microsoft.Extensions.Configuration.IConfiguration), default(System.Collections.Generic.IEnumerable)) { } public Azure.CloudMachine.MessagingServices Messaging { get { throw null; } } public Azure.CloudMachine.StorageServices Storage { get { throw null; } } } public partial class CloudMachineWorkspace : Azure.Core.ClientWorkspace { - public CloudMachineWorkspace(Azure.Core.TokenCredential credential = null, Microsoft.Extensions.Configuration.IConfiguration configuration = null) { } + public CloudMachineWorkspace(Azure.Core.TokenCredential credential = null, Microsoft.Extensions.Configuration.IConfiguration configuration = null, System.Collections.Generic.IEnumerable connections = null) : base (default(Azure.Core.TokenCredential)) { } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.CloudMachine.ConnectionCollection Connections { get { throw null; } } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public string Id { get { throw null; } } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public override bool Equals(object obj) { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public override Azure.Core.ClientConnectionOptions GetConnectionOptions(System.Type clientType, string instanceId) { throw null; } + public override Azure.Core.ClientConnection GetConnectionOptions(string connectionId) { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public override int GetHashCode() { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public override string ToString() { throw null; } } + public partial class ConnectionCollection : System.Collections.ObjectModel.KeyedCollection + { + public ConnectionCollection() { } + protected override string GetKeyForItem(Azure.Core.ClientConnection item) { throw null; } + } [System.Runtime.InteropServices.StructLayoutAttribute(System.Runtime.InteropServices.LayoutKind.Sequential)] public readonly partial struct MessagingServices { @@ -39,6 +78,7 @@ internal StorageFile() { } public void Delete() { } public System.Threading.Tasks.Task DeleteAsync() { throw null; } public System.BinaryData Download() { throw null; } + public System.Threading.Tasks.Task DownloadAsync() { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public override bool Equals(object obj) { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] @@ -56,12 +96,14 @@ public void Delete(string path) { } public System.Threading.Tasks.Task DeleteAsync(string path) { throw null; } public System.BinaryData Download(string path) { throw null; } public System.Threading.Tasks.Task DownloadAsync(string path) { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.Storage.Blobs.BlobContainerClient GetContainer(string containerName = null) { throw null; } public string Upload(System.BinaryData data, string name = null, bool overwrite = false) { throw null; } public string Upload(System.IO.Stream fileStream, string name = null, string contentType = null, bool overwrite = false) { throw null; } public System.Threading.Tasks.Task UploadAsync(System.BinaryData data, string name = null, bool overwrite = false) { throw null; } public System.Threading.Tasks.Task UploadAsync(System.IO.Stream fileStream, string name = null, string contentType = null, bool overwrite = false) { throw null; } - public string UploadJson(object json, string name = null, bool overwrite = false) { throw null; } - public System.Threading.Tasks.Task UploadJsonAsync(object json, string name = null, bool overwrite = false) { throw null; } + public string UploadJson(object serializable, string name = null, bool overwrite = false) { throw null; } + public System.Threading.Tasks.Task UploadJsonAsync(object serializable, string name = null, bool overwrite = false) { throw null; } public void WhenUploaded(System.Action function) { } public void WhenUploaded(System.Action function) { } } @@ -80,7 +122,8 @@ public static partial class AzureOpenAIExtensions public static void Add(this System.Collections.Generic.List messages, OpenAI.Chat.ChatCompletion completion) { } public static void Add(this System.Collections.Generic.List messages, System.Collections.Generic.IEnumerable entries) { } public static string AsText(this OpenAI.Chat.ChatCompletion completion) { throw null; } - public static string AsText(this OpenAI.Chat.ChatMessageContent completion) { throw null; } + public static string AsText(this OpenAI.Chat.ChatMessageContent content) { throw null; } + public static string AsText(this System.ClientModel.ClientResult completionResult) { throw null; } public static OpenAI.Chat.ChatClient GetOpenAIChatClient(this Azure.Core.ClientWorkspace workspace) { throw null; } public static OpenAI.Embeddings.EmbeddingClient GetOpenAIEmbeddingsClient(this Azure.Core.ClientWorkspace workspace) { throw null; } public static void Trim(this System.Collections.Generic.List messages) { } @@ -94,11 +137,6 @@ public void Add(System.Type functions) { } public string Call(OpenAI.Chat.ChatToolCall call) { throw null; } public string Call(string name, object[] arguments) { throw null; } public System.Collections.Generic.IEnumerable CallAll(System.Collections.Generic.IEnumerable toolCalls) { throw null; } - protected string ClrToJsonTypeUtf16(System.Type clrType) { throw null; } - protected System.ReadOnlySpan ClrToJsonTypeUtf8(System.Type clrType) { throw null; } - protected virtual string GetMethodInfoToDescription(System.Reflection.MethodInfo function) { throw null; } - protected virtual string GetMethodInfoToName(System.Reflection.MethodInfo function) { throw null; } - protected virtual string GetParameterInfoToDescription(System.Reflection.ParameterInfo parameter) { throw null; } public static implicit operator OpenAI.Chat.ChatCompletionOptions (Azure.CloudMachine.OpenAI.ChatTools tools) { throw null; } } public partial class EmbeddingsVectorbase @@ -135,36 +173,40 @@ protected VectorbaseStore() { } } namespace Azure.Core { + public enum ClientAuthenticationMethod + { + EntraId = 0, + ApiKey = 1, + Subclient = 2, + } public partial class ClientCache { public ClientCache() { } public T Get(System.Func value, string id = null) where T : class { throw null; } } - public enum ClientConnectionKind - { - EntraId = 0, - ApiKey = 1, - OutOfBand = 2, - } [System.Runtime.InteropServices.StructLayoutAttribute(System.Runtime.InteropServices.LayoutKind.Sequential)] - public readonly partial struct ClientConnectionOptions + public readonly partial struct ClientConnection { private readonly object _dummy; private readonly int _dummyPrimitive; - public ClientConnectionOptions(string subclientId) { throw null; } - public ClientConnectionOptions(System.Uri endpoint, Azure.Core.TokenCredential credential) { throw null; } - public ClientConnectionOptions(System.Uri endpoint, string apiKey) { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public ClientConnection() { throw null; } + public ClientConnection(string id, string locator, Azure.Core.ClientAuthenticationMethod auth = Azure.Core.ClientAuthenticationMethod.EntraId) { throw null; } + public ClientConnection(string id, string locator, string apiKey) { throw null; } public string ApiKeyCredential { get { throw null; } } - public Azure.Core.ClientConnectionKind ConnectionKind { get { throw null; } } - public System.Uri Endpoint { get { throw null; } } + public Azure.Core.ClientAuthenticationMethod Authentication { get { throw null; } } public string Id { get { throw null; } } - public Azure.Core.TokenCredential TokenCredential { get { throw null; } } + public string Locator { get { throw null; } } + public override string ToString() { throw null; } + public System.Uri ToUri() { throw null; } } public abstract partial class ClientWorkspace { - protected ClientWorkspace() { } + protected ClientWorkspace(Azure.Core.TokenCredential credential) { } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.Core.TokenCredential Credential { get { throw null; } } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public Azure.Core.ClientCache Subclients { get { throw null; } } - public abstract Azure.Core.ClientConnectionOptions GetConnectionOptions(System.Type clientType, string instanceId = null); + public abstract Azure.Core.ClientConnection GetConnectionOptions(string connectionId); } } diff --git a/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.netstandard2.0.cs b/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.netstandard2.0.cs index 2bfb5ef447534..09f4d6544fdc0 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.netstandard2.0.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.netstandard2.0.cs @@ -1,26 +1,65 @@ +namespace Azure +{ + public partial class RestCallFailedException : System.Exception + { + public RestCallFailedException(string message, System.ClientModel.Primitives.PipelineResponse response) { } + } + public partial class RestClient + { + public RestClient() { } + public RestClient(System.ClientModel.Primitives.PipelinePolicy auth) { } + public static Azure.RestClient Shared { get { throw null; } } + public System.ClientModel.Primitives.PipelineMessage Create(string method, System.Uri uri) { throw null; } + public System.ClientModel.Primitives.PipelineResponse Get(string uri, System.ClientModel.Primitives.RequestOptions options = null) { throw null; } + public System.ClientModel.Primitives.PipelineResponse Patch(string uri, System.ClientModel.BinaryContent content, System.ClientModel.Primitives.RequestOptions options = null) { throw null; } + public System.ClientModel.Primitives.PipelineResponse Post(string uri, System.ClientModel.BinaryContent content, System.ClientModel.Primitives.RequestOptions options = null) { throw null; } + public System.ClientModel.Primitives.PipelineResponse Put(string uri, System.ClientModel.BinaryContent content, System.ClientModel.Primitives.RequestOptions options = null) { throw null; } + public System.ClientModel.Primitives.PipelineResponse Send(System.ClientModel.Primitives.PipelineMessage message, System.ClientModel.Primitives.RequestOptions options = null) { throw null; } + } + public partial class RestClientOptions : System.ClientModel.Primitives.ClientPipelineOptions + { + public RestClientOptions() { } + } +} +namespace Azure.AI.OpenAI +{ + public partial class TokenCredentialAuthenticationPolicy : System.ClientModel.Primitives.PipelinePolicy + { + public TokenCredentialAuthenticationPolicy(Azure.Core.TokenCredential credential, System.Collections.Generic.IEnumerable scopes, System.TimeSpan? refreshOffset = default(System.TimeSpan?)) { } + public override void Process(System.ClientModel.Primitives.PipelineMessage message, System.Collections.Generic.IReadOnlyList pipeline, int currentIndex) { } + public override System.Threading.Tasks.ValueTask ProcessAsync(System.ClientModel.Primitives.PipelineMessage message, System.Collections.Generic.IReadOnlyList pipeline, int currentIndex) { throw null; } + } +} namespace Azure.CloudMachine { public partial class CloudMachineClient : Azure.CloudMachine.CloudMachineWorkspace { - protected CloudMachineClient() : base (default(Azure.Core.TokenCredential), default(Microsoft.Extensions.Configuration.IConfiguration)) { } - public CloudMachineClient(Azure.Core.TokenCredential credential = null, Microsoft.Extensions.Configuration.IConfiguration configuration = null) : base (default(Azure.Core.TokenCredential), default(Microsoft.Extensions.Configuration.IConfiguration)) { } + protected CloudMachineClient() : base (default(Azure.Core.TokenCredential), default(Microsoft.Extensions.Configuration.IConfiguration), default(System.Collections.Generic.IEnumerable)) { } + public CloudMachineClient(Azure.Core.TokenCredential credential = null, Microsoft.Extensions.Configuration.IConfiguration configuration = null, System.Collections.Generic.IEnumerable connections = null) : base (default(Azure.Core.TokenCredential), default(Microsoft.Extensions.Configuration.IConfiguration), default(System.Collections.Generic.IEnumerable)) { } public Azure.CloudMachine.MessagingServices Messaging { get { throw null; } } public Azure.CloudMachine.StorageServices Storage { get { throw null; } } } public partial class CloudMachineWorkspace : Azure.Core.ClientWorkspace { - public CloudMachineWorkspace(Azure.Core.TokenCredential credential = null, Microsoft.Extensions.Configuration.IConfiguration configuration = null) { } + public CloudMachineWorkspace(Azure.Core.TokenCredential credential = null, Microsoft.Extensions.Configuration.IConfiguration configuration = null, System.Collections.Generic.IEnumerable connections = null) : base (default(Azure.Core.TokenCredential)) { } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.CloudMachine.ConnectionCollection Connections { get { throw null; } } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public string Id { get { throw null; } } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public override bool Equals(object obj) { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public override Azure.Core.ClientConnectionOptions GetConnectionOptions(System.Type clientType, string instanceId) { throw null; } + public override Azure.Core.ClientConnection GetConnectionOptions(string connectionId) { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public override int GetHashCode() { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public override string ToString() { throw null; } } + public partial class ConnectionCollection : System.Collections.ObjectModel.KeyedCollection + { + public ConnectionCollection() { } + protected override string GetKeyForItem(Azure.Core.ClientConnection item) { throw null; } + } [System.Runtime.InteropServices.StructLayoutAttribute(System.Runtime.InteropServices.LayoutKind.Sequential)] public readonly partial struct MessagingServices { @@ -39,6 +78,7 @@ internal StorageFile() { } public void Delete() { } public System.Threading.Tasks.Task DeleteAsync() { throw null; } public System.BinaryData Download() { throw null; } + public System.Threading.Tasks.Task DownloadAsync() { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public override bool Equals(object obj) { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] @@ -56,12 +96,14 @@ public void Delete(string path) { } public System.Threading.Tasks.Task DeleteAsync(string path) { throw null; } public System.BinaryData Download(string path) { throw null; } public System.Threading.Tasks.Task DownloadAsync(string path) { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.Storage.Blobs.BlobContainerClient GetContainer(string containerName = null) { throw null; } public string Upload(System.BinaryData data, string name = null, bool overwrite = false) { throw null; } public string Upload(System.IO.Stream fileStream, string name = null, string contentType = null, bool overwrite = false) { throw null; } public System.Threading.Tasks.Task UploadAsync(System.BinaryData data, string name = null, bool overwrite = false) { throw null; } public System.Threading.Tasks.Task UploadAsync(System.IO.Stream fileStream, string name = null, string contentType = null, bool overwrite = false) { throw null; } - public string UploadJson(object json, string name = null, bool overwrite = false) { throw null; } - public System.Threading.Tasks.Task UploadJsonAsync(object json, string name = null, bool overwrite = false) { throw null; } + public string UploadJson(object serializable, string name = null, bool overwrite = false) { throw null; } + public System.Threading.Tasks.Task UploadJsonAsync(object serializable, string name = null, bool overwrite = false) { throw null; } public void WhenUploaded(System.Action function) { } public void WhenUploaded(System.Action function) { } } @@ -80,7 +122,8 @@ public static partial class AzureOpenAIExtensions public static void Add(this System.Collections.Generic.List messages, OpenAI.Chat.ChatCompletion completion) { } public static void Add(this System.Collections.Generic.List messages, System.Collections.Generic.IEnumerable entries) { } public static string AsText(this OpenAI.Chat.ChatCompletion completion) { throw null; } - public static string AsText(this OpenAI.Chat.ChatMessageContent completion) { throw null; } + public static string AsText(this OpenAI.Chat.ChatMessageContent content) { throw null; } + public static string AsText(this System.ClientModel.ClientResult completionResult) { throw null; } public static OpenAI.Chat.ChatClient GetOpenAIChatClient(this Azure.Core.ClientWorkspace workspace) { throw null; } public static OpenAI.Embeddings.EmbeddingClient GetOpenAIEmbeddingsClient(this Azure.Core.ClientWorkspace workspace) { throw null; } public static void Trim(this System.Collections.Generic.List messages) { } @@ -94,11 +137,6 @@ public void Add(System.Type functions) { } public string Call(OpenAI.Chat.ChatToolCall call) { throw null; } public string Call(string name, object[] arguments) { throw null; } public System.Collections.Generic.IEnumerable CallAll(System.Collections.Generic.IEnumerable toolCalls) { throw null; } - protected string ClrToJsonTypeUtf16(System.Type clrType) { throw null; } - protected System.ReadOnlySpan ClrToJsonTypeUtf8(System.Type clrType) { throw null; } - protected virtual string GetMethodInfoToDescription(System.Reflection.MethodInfo function) { throw null; } - protected virtual string GetMethodInfoToName(System.Reflection.MethodInfo function) { throw null; } - protected virtual string GetParameterInfoToDescription(System.Reflection.ParameterInfo parameter) { throw null; } public static implicit operator OpenAI.Chat.ChatCompletionOptions (Azure.CloudMachine.OpenAI.ChatTools tools) { throw null; } } public partial class EmbeddingsVectorbase @@ -135,36 +173,40 @@ protected VectorbaseStore() { } } namespace Azure.Core { + public enum ClientAuthenticationMethod + { + EntraId = 0, + ApiKey = 1, + Subclient = 2, + } public partial class ClientCache { public ClientCache() { } public T Get(System.Func value, string id = null) where T : class { throw null; } } - public enum ClientConnectionKind - { - EntraId = 0, - ApiKey = 1, - OutOfBand = 2, - } [System.Runtime.InteropServices.StructLayoutAttribute(System.Runtime.InteropServices.LayoutKind.Sequential)] - public readonly partial struct ClientConnectionOptions + public readonly partial struct ClientConnection { private readonly object _dummy; private readonly int _dummyPrimitive; - public ClientConnectionOptions(string subclientId) { throw null; } - public ClientConnectionOptions(System.Uri endpoint, Azure.Core.TokenCredential credential) { throw null; } - public ClientConnectionOptions(System.Uri endpoint, string apiKey) { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public ClientConnection() { throw null; } + public ClientConnection(string id, string locator, Azure.Core.ClientAuthenticationMethod auth = Azure.Core.ClientAuthenticationMethod.EntraId) { throw null; } + public ClientConnection(string id, string locator, string apiKey) { throw null; } public string ApiKeyCredential { get { throw null; } } - public Azure.Core.ClientConnectionKind ConnectionKind { get { throw null; } } - public System.Uri Endpoint { get { throw null; } } + public Azure.Core.ClientAuthenticationMethod Authentication { get { throw null; } } public string Id { get { throw null; } } - public Azure.Core.TokenCredential TokenCredential { get { throw null; } } + public string Locator { get { throw null; } } + public override string ToString() { throw null; } + public System.Uri ToUri() { throw null; } } public abstract partial class ClientWorkspace { - protected ClientWorkspace() { } + protected ClientWorkspace(Azure.Core.TokenCredential credential) { } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.Core.TokenCredential Credential { get { throw null; } } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public Azure.Core.ClientCache Subclients { get { throw null; } } - public abstract Azure.Core.ClientConnectionOptions GetConnectionOptions(System.Type clientType, string instanceId = null); + public abstract Azure.Core.ClientConnection GetConnectionOptions(string connectionId); } } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs index dedf171961694..6e7ba8f19b01b 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs @@ -1,88 +1,80 @@ -namespace Azure +namespace Azure.CloudMachine { - public partial class RestCallFailedException : System.Exception + public static partial class Azd { - public RestCallFailedException(string message, System.ClientModel.Primitives.PipelineResponse response) { } + public static void Init(Azure.CloudMachine.CloudMachineClient client, string? infraDirectory = null) { } + public static void Init(Azure.CloudMachine.CloudMachineInfrastructure infra, string? infraDirectory = null) { } } - public partial class RestClient + public static partial class CloudMachineClientExtensions { - public RestClient() { } - public RestClient(System.ClientModel.Primitives.PipelinePolicy auth) { } - public static Azure.RestClient Shared { get { throw null; } } - public System.ClientModel.Primitives.PipelineMessage Create(string method, System.Uri uri) { throw null; } - public System.ClientModel.Primitives.PipelineResponse Get(string uri, System.ClientModel.Primitives.RequestOptions? options = null) { throw null; } - public System.ClientModel.Primitives.PipelineResponse Patch(string uri, System.ClientModel.BinaryContent content, System.ClientModel.Primitives.RequestOptions? options = null) { throw null; } - public System.ClientModel.Primitives.PipelineResponse Post(string uri, System.ClientModel.BinaryContent content, System.ClientModel.Primitives.RequestOptions? options = null) { throw null; } - public System.ClientModel.Primitives.PipelineResponse Put(string uri, System.ClientModel.BinaryContent content, System.ClientModel.Primitives.RequestOptions? options = null) { throw null; } - public System.ClientModel.Primitives.PipelineResponse Send(System.ClientModel.Primitives.PipelineMessage message, System.ClientModel.Primitives.RequestOptions? options = null) { throw null; } + public static T AddFeature(this Azure.CloudMachine.CloudMachineClient client, T feature) where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } + public static void Configure(this Azure.CloudMachine.CloudMachineClient client, System.Action? configure = null) { } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public static Azure.CloudMachine.CloudMachineInfrastructure GetInfrastructure(this Azure.CloudMachine.CloudMachineClient client) { throw null; } } - public partial class RestClientOptions : System.ClientModel.Primitives.ClientPipelineOptions + public static partial class CloudMachineCommands { - public RestClientOptions() { } - } -} -namespace Azure.CloudMachine -{ - public partial class CloudMachineCommands - { - public CloudMachineCommands() { } public static bool Execute(string[] args, System.Action? configure = null, bool exitProcessIfHandled = true) { throw null; } } public partial class CloudMachineInfrastructure { - public CloudMachineInfrastructure(string cmId) { } + public CloudMachineInfrastructure(string? cmId = null) { } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.CloudMachine.ConnectionCollection Connections { get { throw null; } } public Azure.CloudMachine.FeatureCollection Features { get { throw null; } } public string Id { get { throw null; } } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public Azure.Provisioning.Roles.UserAssignedIdentity Identity { get { throw null; } } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public Azure.Provisioning.ProvisioningParameter PrincipalIdParameter { get { throw null; } } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public void AddConstruct(Azure.Provisioning.Primitives.NamedProvisionableConstruct resource) { } public void AddEndpoints() { } - public void AddFeature(Azure.Provisioning.CloudMachine.CloudMachineFeature feature) { } - public void AddResource(Azure.Provisioning.Primitives.NamedProvisionableConstruct resource) { } + public T AddFeature(T feature) where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public Azure.Provisioning.ProvisioningPlan Build(Azure.Provisioning.ProvisioningBuildOptions? context = null) { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override bool Equals(object? obj) { throw null; } + public Azure.CloudMachine.CloudMachineClient GetClient() { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override int GetHashCode() { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override string ToString() { throw null; } } - public partial class EventGridSystemTopicFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public EventGridSystemTopicFeature(string name, Azure.Provisioning.CloudMachine.CloudMachineFeature source) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } - } - public partial class FeatureCollection + public partial class FeatureCollection : System.Collections.Generic.IEnumerable, System.Collections.IEnumerable { - public FeatureCollection() { } + internal FeatureCollection() { } public System.Collections.Generic.IEnumerable FindAll() where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } + public System.Collections.Generic.IEnumerator GetEnumerator() { throw null; } + System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator() { throw null; } } public partial class ServiceBusNamespaceFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature { public ServiceBusNamespaceFeature(string name, Azure.Provisioning.ServiceBus.ServiceBusSkuName sku = Azure.Provisioning.ServiceBus.ServiceBusSkuName.Standard, Azure.Provisioning.ServiceBus.ServiceBusSkuTier tier = Azure.Provisioning.ServiceBus.ServiceBusSkuTier.Standard) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } } public partial class ServiceBusSubscriptionFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature { public ServiceBusSubscriptionFeature(string name, Azure.CloudMachine.ServiceBusTopicFeature parent) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } } public partial class ServiceBusTopicFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature { public ServiceBusTopicFeature(string name, Azure.CloudMachine.ServiceBusNamespaceFeature parent) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } - } - public partial class StorageFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public StorageFeature(string accountName, Azure.Provisioning.Storage.StorageSkuName sku = Azure.Provisioning.Storage.StorageSkuName.StandardLrs, System.Collections.Generic.IEnumerable? containerNames = null) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } - } - public partial class SystemTopicEventSubscriptionFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public SystemTopicEventSubscriptionFeature(string name, Azure.CloudMachine.EventGridSystemTopicFeature parent, Azure.CloudMachine.ServiceBusTopicFeature destination, Azure.CloudMachine.ServiceBusNamespaceFeature parentNamespace) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + public string Name { get { throw null; } } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } } } namespace Azure.CloudMachine.AppService { public partial class AppServiceFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature { - public AppServiceFeature(Azure.Provisioning.AppService.AppServiceSkuDescription? sku = null) { } + public AppServiceFeature() { } public Azure.Provisioning.AppService.AppServiceSkuDescription Sku { get { throw null; } set { } } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } } } namespace Azure.CloudMachine.KeyVault @@ -91,7 +83,8 @@ public partial class KeyVaultFeature : Azure.Provisioning.CloudMachine.CloudMach { public KeyVaultFeature(Azure.Provisioning.KeyVault.KeyVaultSku? sku = null) { } public Azure.Provisioning.KeyVault.KeyVaultSku Sku { get { throw null; } set { } } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } } } namespace Azure.CloudMachine.OpenAI @@ -101,13 +94,14 @@ public enum AIModelKind Chat = 0, Embedding = 1, } - public partial class OpenAIModel : Azure.Provisioning.CloudMachine.CloudMachineFeature + public partial class OpenAIModelFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature { - public OpenAIModel(string model, string modelVersion, Azure.CloudMachine.OpenAI.AIModelKind kind = Azure.CloudMachine.OpenAI.AIModelKind.Chat) { } + public OpenAIModelFeature(string model, string modelVersion, Azure.CloudMachine.OpenAI.AIModelKind kind = Azure.CloudMachine.OpenAI.AIModelKind.Chat) { } public string Model { get { throw null; } } public string ModelVersion { get { throw null; } } - public override void AddTo(Azure.CloudMachine.CloudMachineInfrastructure cm) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure cm) { throw null; } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure cm) { throw null; } + protected internal override void EmitFeatures(Azure.CloudMachine.FeatureCollection features, string cmId) { } } } namespace Azure.Provisioning.CloudMachine @@ -116,14 +110,18 @@ public abstract partial class CloudMachineFeature { protected CloudMachineFeature() { } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public Azure.Provisioning.Primitives.ProvisionableResource Emitted { get { throw null; } protected set { } } + public Azure.Provisioning.Primitives.ProvisionableResource Emitted { get { throw null; } } protected internal System.Collections.Generic.Dictionary RequiredSystemRoles { get { throw null; } } + protected internal virtual void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected abstract Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure cm); + protected internal virtual void EmitFeatures(Azure.CloudMachine.FeatureCollection features, string cmId) { } + protected static T EnsureEmits(Azure.Provisioning.CloudMachine.CloudMachineFeature feature) { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override bool Equals(object? obj) { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public virtual void AddTo(Azure.CloudMachine.CloudMachineInfrastructure cm) { } + public override int GetHashCode() { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public void Emit(Azure.CloudMachine.CloudMachineInfrastructure cm) { } - protected abstract Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure cm); - protected static T ValidateIsOfType(Azure.Provisioning.CloudMachine.CloudMachineFeature resource) { throw null; } + public override string ToString() { throw null; } } } namespace System.ClientModel.TypeSpec diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs index dedf171961694..6e7ba8f19b01b 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs @@ -1,88 +1,80 @@ -namespace Azure +namespace Azure.CloudMachine { - public partial class RestCallFailedException : System.Exception + public static partial class Azd { - public RestCallFailedException(string message, System.ClientModel.Primitives.PipelineResponse response) { } + public static void Init(Azure.CloudMachine.CloudMachineClient client, string? infraDirectory = null) { } + public static void Init(Azure.CloudMachine.CloudMachineInfrastructure infra, string? infraDirectory = null) { } } - public partial class RestClient + public static partial class CloudMachineClientExtensions { - public RestClient() { } - public RestClient(System.ClientModel.Primitives.PipelinePolicy auth) { } - public static Azure.RestClient Shared { get { throw null; } } - public System.ClientModel.Primitives.PipelineMessage Create(string method, System.Uri uri) { throw null; } - public System.ClientModel.Primitives.PipelineResponse Get(string uri, System.ClientModel.Primitives.RequestOptions? options = null) { throw null; } - public System.ClientModel.Primitives.PipelineResponse Patch(string uri, System.ClientModel.BinaryContent content, System.ClientModel.Primitives.RequestOptions? options = null) { throw null; } - public System.ClientModel.Primitives.PipelineResponse Post(string uri, System.ClientModel.BinaryContent content, System.ClientModel.Primitives.RequestOptions? options = null) { throw null; } - public System.ClientModel.Primitives.PipelineResponse Put(string uri, System.ClientModel.BinaryContent content, System.ClientModel.Primitives.RequestOptions? options = null) { throw null; } - public System.ClientModel.Primitives.PipelineResponse Send(System.ClientModel.Primitives.PipelineMessage message, System.ClientModel.Primitives.RequestOptions? options = null) { throw null; } + public static T AddFeature(this Azure.CloudMachine.CloudMachineClient client, T feature) where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } + public static void Configure(this Azure.CloudMachine.CloudMachineClient client, System.Action? configure = null) { } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public static Azure.CloudMachine.CloudMachineInfrastructure GetInfrastructure(this Azure.CloudMachine.CloudMachineClient client) { throw null; } } - public partial class RestClientOptions : System.ClientModel.Primitives.ClientPipelineOptions + public static partial class CloudMachineCommands { - public RestClientOptions() { } - } -} -namespace Azure.CloudMachine -{ - public partial class CloudMachineCommands - { - public CloudMachineCommands() { } public static bool Execute(string[] args, System.Action? configure = null, bool exitProcessIfHandled = true) { throw null; } } public partial class CloudMachineInfrastructure { - public CloudMachineInfrastructure(string cmId) { } + public CloudMachineInfrastructure(string? cmId = null) { } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.CloudMachine.ConnectionCollection Connections { get { throw null; } } public Azure.CloudMachine.FeatureCollection Features { get { throw null; } } public string Id { get { throw null; } } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public Azure.Provisioning.Roles.UserAssignedIdentity Identity { get { throw null; } } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public Azure.Provisioning.ProvisioningParameter PrincipalIdParameter { get { throw null; } } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public void AddConstruct(Azure.Provisioning.Primitives.NamedProvisionableConstruct resource) { } public void AddEndpoints() { } - public void AddFeature(Azure.Provisioning.CloudMachine.CloudMachineFeature feature) { } - public void AddResource(Azure.Provisioning.Primitives.NamedProvisionableConstruct resource) { } + public T AddFeature(T feature) where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public Azure.Provisioning.ProvisioningPlan Build(Azure.Provisioning.ProvisioningBuildOptions? context = null) { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override bool Equals(object? obj) { throw null; } + public Azure.CloudMachine.CloudMachineClient GetClient() { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override int GetHashCode() { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override string ToString() { throw null; } } - public partial class EventGridSystemTopicFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public EventGridSystemTopicFeature(string name, Azure.Provisioning.CloudMachine.CloudMachineFeature source) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } - } - public partial class FeatureCollection + public partial class FeatureCollection : System.Collections.Generic.IEnumerable, System.Collections.IEnumerable { - public FeatureCollection() { } + internal FeatureCollection() { } public System.Collections.Generic.IEnumerable FindAll() where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } + public System.Collections.Generic.IEnumerator GetEnumerator() { throw null; } + System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator() { throw null; } } public partial class ServiceBusNamespaceFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature { public ServiceBusNamespaceFeature(string name, Azure.Provisioning.ServiceBus.ServiceBusSkuName sku = Azure.Provisioning.ServiceBus.ServiceBusSkuName.Standard, Azure.Provisioning.ServiceBus.ServiceBusSkuTier tier = Azure.Provisioning.ServiceBus.ServiceBusSkuTier.Standard) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } } public partial class ServiceBusSubscriptionFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature { public ServiceBusSubscriptionFeature(string name, Azure.CloudMachine.ServiceBusTopicFeature parent) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } } public partial class ServiceBusTopicFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature { public ServiceBusTopicFeature(string name, Azure.CloudMachine.ServiceBusNamespaceFeature parent) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } - } - public partial class StorageFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public StorageFeature(string accountName, Azure.Provisioning.Storage.StorageSkuName sku = Azure.Provisioning.Storage.StorageSkuName.StandardLrs, System.Collections.Generic.IEnumerable? containerNames = null) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } - } - public partial class SystemTopicEventSubscriptionFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public SystemTopicEventSubscriptionFeature(string name, Azure.CloudMachine.EventGridSystemTopicFeature parent, Azure.CloudMachine.ServiceBusTopicFeature destination, Azure.CloudMachine.ServiceBusNamespaceFeature parentNamespace) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + public string Name { get { throw null; } } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } } } namespace Azure.CloudMachine.AppService { public partial class AppServiceFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature { - public AppServiceFeature(Azure.Provisioning.AppService.AppServiceSkuDescription? sku = null) { } + public AppServiceFeature() { } public Azure.Provisioning.AppService.AppServiceSkuDescription Sku { get { throw null; } set { } } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } } } namespace Azure.CloudMachine.KeyVault @@ -91,7 +83,8 @@ public partial class KeyVaultFeature : Azure.Provisioning.CloudMachine.CloudMach { public KeyVaultFeature(Azure.Provisioning.KeyVault.KeyVaultSku? sku = null) { } public Azure.Provisioning.KeyVault.KeyVaultSku Sku { get { throw null; } set { } } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } } } namespace Azure.CloudMachine.OpenAI @@ -101,13 +94,14 @@ public enum AIModelKind Chat = 0, Embedding = 1, } - public partial class OpenAIModel : Azure.Provisioning.CloudMachine.CloudMachineFeature + public partial class OpenAIModelFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature { - public OpenAIModel(string model, string modelVersion, Azure.CloudMachine.OpenAI.AIModelKind kind = Azure.CloudMachine.OpenAI.AIModelKind.Chat) { } + public OpenAIModelFeature(string model, string modelVersion, Azure.CloudMachine.OpenAI.AIModelKind kind = Azure.CloudMachine.OpenAI.AIModelKind.Chat) { } public string Model { get { throw null; } } public string ModelVersion { get { throw null; } } - public override void AddTo(Azure.CloudMachine.CloudMachineInfrastructure cm) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure cm) { throw null; } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure cm) { throw null; } + protected internal override void EmitFeatures(Azure.CloudMachine.FeatureCollection features, string cmId) { } } } namespace Azure.Provisioning.CloudMachine @@ -116,14 +110,18 @@ public abstract partial class CloudMachineFeature { protected CloudMachineFeature() { } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public Azure.Provisioning.Primitives.ProvisionableResource Emitted { get { throw null; } protected set { } } + public Azure.Provisioning.Primitives.ProvisionableResource Emitted { get { throw null; } } protected internal System.Collections.Generic.Dictionary RequiredSystemRoles { get { throw null; } } + protected internal virtual void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected abstract Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure cm); + protected internal virtual void EmitFeatures(Azure.CloudMachine.FeatureCollection features, string cmId) { } + protected static T EnsureEmits(Azure.Provisioning.CloudMachine.CloudMachineFeature feature) { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override bool Equals(object? obj) { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public virtual void AddTo(Azure.CloudMachine.CloudMachineInfrastructure cm) { } + public override int GetHashCode() { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public void Emit(Azure.CloudMachine.CloudMachineInfrastructure cm) { } - protected abstract Azure.Provisioning.Primitives.ProvisionableResource EmitCore(Azure.CloudMachine.CloudMachineInfrastructure cm); - protected static T ValidateIsOfType(Azure.Provisioning.CloudMachine.CloudMachineFeature resource) { throw null; } + public override string ToString() { throw null; } } } namespace System.ClientModel.TypeSpec From a835511e15b5170d1bf8a19ddbf4b3d9d2f87d2e Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Mon, 2 Dec 2024 08:53:53 -0800 Subject: [PATCH 19/24] made tests run sequentially --- .../Azure.Provisioning.CloudMachine/tests/ProvisioningTests.cs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/ProvisioningTests.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/ProvisioningTests.cs index 56ed939ceed94..3645cf2c7ccf3 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/ProvisioningTests.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/tests/ProvisioningTests.cs @@ -11,6 +11,8 @@ using Azure.CloudMachine.OpenAI; using NUnit.Framework; +[assembly: NonParallelizable] + namespace Azure.CloudMachine.Tests; public class ProvisioningTests From 14f58b35c5202cebdca07f313be3f00a7887d0be Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Mon, 2 Dec 2024 09:54:24 -0800 Subject: [PATCH 20/24] made tests run sequentially --- sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs b/sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs index df30d955cabf2..33e28a6dbde7c 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/tests/ConnectionTests.cs @@ -12,6 +12,8 @@ using NUnit.Framework; using OpenAI.Chat; +[assembly: NonParallelizable] + namespace Azure.CloudMachine.Tests; public class ConnectionTests From 9bee34a584fcfaf8f322d10a628a3fa7d7a50d16 Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Mon, 2 Dec 2024 10:26:35 -0800 Subject: [PATCH 21/24] removed duplicated code --- .../api/Azure.CloudMachine.net8.0.cs | 2 + .../api/Azure.CloudMachine.netstandard2.0.cs | 2 + .../src/AppConfigHelpers.cs | 87 ++++++++++++++ .../src/CloudMachineWorkspace.cs | 86 ++------------ .../CloudMachineInfrastructure.cs | 2 +- .../src/Tooling/AppConfigHelpers.cs | 108 ------------------ .../src/Tooling/Azd.cs | 2 +- .../src/Tooling/CloudMachineCommands.cs | 2 +- 8 files changed, 103 insertions(+), 188 deletions(-) create mode 100644 sdk/cloudmachine/Azure.CloudMachine/src/AppConfigHelpers.cs delete mode 100644 sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/AppConfigHelpers.cs diff --git a/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.net8.0.cs b/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.net8.0.cs index 09f4d6544fdc0..8e90eb23698cb 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.net8.0.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.net8.0.cs @@ -53,6 +53,8 @@ public CloudMachineWorkspace(Azure.Core.TokenCredential credential = null, Micro [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public override int GetHashCode() { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public static string ReadOrCreateCmid() { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public override string ToString() { throw null; } } public partial class ConnectionCollection : System.Collections.ObjectModel.KeyedCollection diff --git a/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.netstandard2.0.cs b/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.netstandard2.0.cs index 09f4d6544fdc0..8e90eb23698cb 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.netstandard2.0.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.netstandard2.0.cs @@ -53,6 +53,8 @@ public CloudMachineWorkspace(Azure.Core.TokenCredential credential = null, Micro [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public override int GetHashCode() { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public static string ReadOrCreateCmid() { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public override string ToString() { throw null; } } public partial class ConnectionCollection : System.Collections.ObjectModel.KeyedCollection diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/AppConfigHelpers.cs b/sdk/cloudmachine/Azure.CloudMachine/src/AppConfigHelpers.cs new file mode 100644 index 0000000000000..64c9a256bbde3 --- /dev/null +++ b/sdk/cloudmachine/Azure.CloudMachine/src/AppConfigHelpers.cs @@ -0,0 +1,87 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +using System.IO; +using System.Text.Json.Nodes; +using System.Text.Json; +using System; + +namespace Azure.CloudMachine; + +internal static class AppConfigHelpers +{ + internal static string ReadOrCreateCmid() + { + string appsettings = Path.Combine(".", "appsettings.json"); + + string cmid; + if (!File.Exists(appsettings)) + { + cmid = GenerateCloudMachineId(); + + using FileStream file = File.OpenWrite(appsettings); + Utf8JsonWriter writer = new(file); + writer.WriteStartObject(); + writer.WritePropertyName("CloudMachine"u8); + writer.WriteStartObject(); + writer.WriteString("ID"u8, cmid); + writer.WriteEndObject(); + writer.WriteEndObject(); + writer.Flush(); + file.Close(); + return cmid; + } + + using FileStream json = new FileStream(appsettings, FileMode.Open, FileAccess.Read, FileShare.Read); + using JsonDocument jd = JsonDocument.Parse(json); + JsonElement je = jd.RootElement; + // attempt to read CM configuration from existing configuration file + if (je.TryGetProperty("CloudMachine"u8, out JsonElement cm)) + { + if (!cm.TryGetProperty("ID"u8, out JsonElement id)) + { + throw new NotImplementedException(); + } + cmid = id.GetString(); + if (cmid == null) + throw new NotImplementedException(); + return cmid; + } + else + { // add CM configuration to existing file + json.Seek(0, SeekOrigin.Begin); + JsonNode root = JsonNode.Parse(json); + json.Close(); + if (root is null || root is not JsonObject obj) + throw new InvalidOperationException("Existing appsettings.json is not a valid JSON object"); + + var cmProperties = new JsonObject(); + cmid = GenerateCloudMachineId(); + cmProperties.Add("ID", cmid); + obj.Add("CloudMachine", cmProperties); + + using FileStream file = new FileStream(appsettings, FileMode.Open, FileAccess.Write, FileShare.None); + JsonWriterOptions writerOptions = new() + { + Indented = true, + }; + Utf8JsonWriter writer = new(file, writerOptions); + JsonSerializerOptions options = new() + { + WriteIndented = true, + }; + root.WriteTo(writer, options); + writer.Flush(); + } + + return cmid; + } + + private static string GenerateCloudMachineId() + { + var guid = Guid.NewGuid(); + var guidString = guid.ToString("N"); + var cnId = "cm" + guidString.Substring(0, 15); // we can increase it to 20, but the template name cannot be that long + return cnId; + } +} diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs index 8121d6339681c..bcc67e99fe5f4 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs @@ -11,6 +11,7 @@ using System.Text.Json.Nodes; using Azure.Core; using Azure.Identity; +using Azure.Messaging.EventGrid.SystemEvents; using Microsoft.Extensions.Configuration; namespace Azure.CloudMachine; @@ -50,7 +51,7 @@ public CloudMachineWorkspace(TokenCredential credential = default, IConfiguratio Id = configuration switch { - null => ReadOrCreateCmid(), + null => AppConfigHelpers.ReadOrCreateCmid(), _ => configuration["CloudMachine:ID"] ?? throw new Exception("CloudMachine:ID configuration value missing") }; } @@ -82,6 +83,13 @@ public override ClientConnection GetConnectionOptions(string connectionId) return Connections[connectionId]; } + /// + /// Reads or creates the cloud machine ID. + /// + /// + [EditorBrowsable(EditorBrowsableState.Never)] + public static string ReadOrCreateCmid() => AppConfigHelpers.ReadOrCreateCmid(); + /// [EditorBrowsable(EditorBrowsableState.Never)] public override bool Equals(object obj) => base.Equals(obj); @@ -93,80 +101,4 @@ public override ClientConnection GetConnectionOptions(string connectionId) /// [EditorBrowsable(EditorBrowsableState.Never)] public override string ToString() => Id; - - // TODO: Decide if this should live here. - internal static string ReadOrCreateCmid() - { - string appsettings = Path.Combine(".", "appsettings.json"); - - string cmid; - if (!File.Exists(appsettings)) - { - cmid = GenerateCloudMachineId(); - - using FileStream file = File.OpenWrite(appsettings); - Utf8JsonWriter writer = new(file); - writer.WriteStartObject(); - writer.WritePropertyName("CloudMachine"u8); - writer.WriteStartObject(); - writer.WriteString("ID"u8, cmid); - writer.WriteEndObject(); - writer.WriteEndObject(); - writer.Flush(); - file.Close(); - return cmid; - } - - using FileStream json = new FileStream(appsettings, FileMode.Open, FileAccess.Read, FileShare.Read); - using JsonDocument jd = JsonDocument.Parse(json); - JsonElement je = jd.RootElement; - // attempt to read CM configuration from existing configuration file - if (je.TryGetProperty("CloudMachine"u8, out JsonElement cm)) - { - if (!cm.TryGetProperty("ID"u8, out JsonElement id)) - { - throw new NotImplementedException(); - } - cmid = id.GetString(); - if (cmid == null) - throw new NotImplementedException(); - return cmid; - } - else - { // add CM configuration to existing file - json.Seek(0, SeekOrigin.Begin); - JsonNode root = JsonNode.Parse(json); - json.Close(); - if (root is null || root is not JsonObject obj) - throw new InvalidOperationException("Existing appsettings.json is not a valid JSON object"); - - var cmProperties = new JsonObject(); - cmid = GenerateCloudMachineId(); - cmProperties.Add("ID", cmid); - obj.Add("CloudMachine", cmProperties); - - using FileStream file = new FileStream(appsettings, FileMode.Open, FileAccess.Write, FileShare.None); - JsonWriterOptions writerOptions = new() - { - Indented = true, - }; - Utf8JsonWriter writer = new(file, writerOptions); - JsonSerializerOptions options = new() - { - WriteIndented = true, - }; - root.WriteTo(writer, options); - writer.Flush(); - } - - return cmid; - - static string GenerateCloudMachineId() - { - var guid = Guid.NewGuid(); - var guidString = guid.ToString("N"); - var cnId = "cm" + guidString.Substring(0, 15); // we can increase it to 20, but the template name cannot be that long - return cnId; - } - } } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs index abf9fa78fd6ca..c4c3c020d9abb 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs @@ -43,7 +43,7 @@ public CloudMachineInfrastructure(string? cmId = default) { if (cmId == default) { - cmId = AppConfigHelpers.ReadOrCreateCmid(); + cmId = CloudMachineWorkspace.ReadOrCreateCmid(); } Id = cmId; diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/AppConfigHelpers.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/AppConfigHelpers.cs deleted file mode 100644 index aca1a23358140..0000000000000 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/AppConfigHelpers.cs +++ /dev/null @@ -1,108 +0,0 @@ -// Copyright (c) Microsoft Corporation. All rights reserved. -// Licensed under the MIT License. - -using System.IO; -using System.Text.Json.Nodes; -using System.Text.Json; -using System; - -namespace Azure.CloudMachine; - -internal static class AppConfigHelpers -{ - public static string GenerateCloudMachineId() - { - var guid = Guid.NewGuid(); - var guidString = guid.ToString("N"); - var cnId = "cm" + guidString.Substring(0, 15); // we can increase it to 20, but the template name cannot be that long - return cnId; - } - - internal static string ReadOrCreateCmid() - { - string appsettingsPath = Path.Combine(".", "appsettings.json"); - - if (TryReadCmId(appsettingsPath, out string? cmid)) return cmid!; - - cmid = GenerateCloudMachineId(); - - if (!File.Exists(appsettingsPath)) - { - CreateAppConfig(appsettingsPath, cmid); - } - else - { - AddCloudMachineId(appsettingsPath, cmid); - } - - return cmid; - } - - private static void CreateAppConfig(string appsettingsPath, string? cmid) - { - FileStream file = File.OpenWrite(appsettingsPath); - Utf8JsonWriter writer = new(file); - writer.WriteStartObject(); - writer.WritePropertyName("CloudMachine"u8); - writer.WriteStartObject(); - writer.WriteString("ID"u8, cmid); - writer.WriteEndObject(); - writer.WriteEndObject(); - writer.Flush(); - } - - private static bool TryReadCmId(string appsettings, out string? cmid) - { - if (!File.Exists(appsettings)) - { - cmid = null; - return false; - } - using FileStream json = File.OpenRead(appsettings); - using JsonDocument jd = JsonDocument.Parse(json); - JsonElement je = jd.RootElement; - if (je.TryGetProperty("CloudMachine"u8, out JsonElement cm)) - { - if (!cm.TryGetProperty("ID"u8, out JsonElement id)) - { - throw new NotImplementedException(); - } - cmid = id.GetString(); - if (cmid == null) throw new InvalidOperationException($"CloudMachine:ID in {appsettings} is invalid"); - return true; - } - else - { - cmid = null; - return false; - } - } - - private static void AddCloudMachineId(string appsettings, string cmid) - { - FileStream json = File.OpenRead(appsettings); - - JsonNode? root = JsonNode.Parse(json); - json.Close(); - if (root is null || root is not JsonObject obj) - throw new InvalidOperationException("Existing appsettings.json is not a valid JSON object"); - - var cmProperties = new JsonObject(); - cmid = GenerateCloudMachineId(); - cmProperties.Add("ID", cmid); - obj.Add("CloudMachine", cmProperties); - - using FileStream file = File.OpenWrite(appsettings); - JsonWriterOptions writerOptions = new() - { - Indented = true, - }; - Utf8JsonWriter writer = new(file, writerOptions); - JsonSerializerOptions options = new() - { - WriteIndented = true, - }; - root.WriteTo(writer, options); - writer.Flush(); - } -} diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/Azd.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/Azd.cs index f1a94f21ca235..d81c906aa07e1 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/Azd.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/Azd.cs @@ -32,7 +32,7 @@ public static void Init(CloudMachineInfrastructure infra, string? infraDirectory Directory.CreateDirectory(infraDirectory); infra.Build().Save(infraDirectory); - var cmid = AppConfigHelpers.ReadOrCreateCmid(); + var cmid = CloudMachineWorkspace.ReadOrCreateCmid(); // main.bicep var location = new ProvisioningParameter("location", typeof(string)); diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/CloudMachineCommands.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/CloudMachineCommands.cs index 1e80f6405bcd2..334dd97911e80 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/CloudMachineCommands.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/CloudMachineCommands.cs @@ -20,7 +20,7 @@ public static bool Execute(string[] args, Action? co if (args.Length < 1) return false; - string cmid = AppConfigHelpers.ReadOrCreateCmid(); + string cmid = CloudMachineWorkspace.ReadOrCreateCmid(); CloudMachineInfrastructure cmi = new(cmid); if (configure != default) { From 329c88dd67614d2a66e3d42e66a9af7827fc418f Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Mon, 2 Dec 2024 10:41:19 -0800 Subject: [PATCH 22/24] fixed solution file --- sdk/provisioning/Azure.Provisioning.sln | 6 ------ 1 file changed, 6 deletions(-) diff --git a/sdk/provisioning/Azure.Provisioning.sln b/sdk/provisioning/Azure.Provisioning.sln index b75af01e6b88c..e8220a9e9bee9 100644 --- a/sdk/provisioning/Azure.Provisioning.sln +++ b/sdk/provisioning/Azure.Provisioning.sln @@ -106,12 +106,6 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.Provisioning.WebPubSu EndProject Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.Provisioning.WebPubSub.Tests", "Azure.Provisioning.WebPubSub\tests\Azure.Provisioning.WebPubSub.Tests.csproj", "{015670AB-881C-464F-8545-E686C9A92C4F}" EndProject -Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Azure.Provisioning.CloudMachine", "Azure.Provisioning.CloudMachine", "{DB2FC0CB-7103-4D3C-A737-8C6AB61168F8}" -EndProject -Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.Provisioning.CloudMachine", "Azure.Provisioning.CloudMachine\src\Azure.Provisioning.CloudMachine.csproj", "{F54CA64F-3BB8-49D5-AE3B-3408AE324632}" -EndProject -Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.Provisioning.CloudMachine.Tests", "Azure.Provisioning.CloudMachine\tests\Azure.Provisioning.CloudMachine.Tests.csproj", "{94642992-7CC0-4649-B58F-8E5E3F134C8A}" -EndProject Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.Provisioning.Deployment", "Azure.Provisioning.Deployment\src\Azure.Provisioning.Deployment.csproj", "{0BC64290-A91B-4EF2-BAF2-01AE91F81AED}" EndProject Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.Provisioning.Deployment.Tests", "Azure.Provisioning.Deployment\tests\Azure.Provisioning.Deployment.Tests.csproj", "{0D2C464F-C246-4069-90B2-33D637BE18BF}" From 6f4131be704d998fa5efbe4110d4fbf044da9479 Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Mon, 2 Dec 2024 10:54:41 -0800 Subject: [PATCH 23/24] fixed misspelling --- .../api/Azure.CloudMachine.Web.net8.0.cs | 8 -- .../api/Azure.CloudMachine.net8.0.cs | 2 +- .../api/Azure.CloudMachine.netstandard2.0.cs | 2 +- .../src/AppConfigHelpers.cs | 2 +- .../src/CloudMachineWorkspace.cs | 4 +- .../Azure.Provisioning.CloudMachine.net8.0.cs | 136 ------------------ ...rovisioning.CloudMachine.netstandard2.0.cs | 136 ------------------ .../CloudMachineInfrastructure.cs | 2 +- .../src/Tooling/Azd.cs | 2 +- .../src/Tooling/CloudMachineCommands.cs | 2 +- 10 files changed, 8 insertions(+), 288 deletions(-) delete mode 100644 sdk/cloudmachine/Azure.CloudMachine.Web/api/Azure.CloudMachine.Web.net8.0.cs delete mode 100644 sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs delete mode 100644 sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs diff --git a/sdk/cloudmachine/Azure.CloudMachine.Web/api/Azure.CloudMachine.Web.net8.0.cs b/sdk/cloudmachine/Azure.CloudMachine.Web/api/Azure.CloudMachine.Web.net8.0.cs deleted file mode 100644 index 097b70d30b9b7..0000000000000 --- a/sdk/cloudmachine/Azure.CloudMachine.Web/api/Azure.CloudMachine.Web.net8.0.cs +++ /dev/null @@ -1,8 +0,0 @@ -namespace Azure.CloudMachine -{ - public static partial class CloudMachineExtensions - { - public static void Map(this Microsoft.AspNetCore.Routing.IEndpointRouteBuilder routeBuilder, T serviceImplementation) where T : class { } - public static System.Threading.Tasks.Task UploadFormAsync(this Azure.CloudMachine.StorageServices storage, Microsoft.AspNetCore.Http.HttpRequest multiPartFormData) { throw null; } - } -} diff --git a/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.net8.0.cs b/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.net8.0.cs index 8e90eb23698cb..1e6b182bb3f11 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.net8.0.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.net8.0.cs @@ -53,7 +53,7 @@ public CloudMachineWorkspace(Azure.Core.TokenCredential credential = null, Micro [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public override int GetHashCode() { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public static string ReadOrCreateCmid() { throw null; } + public static string ReadOrCreateCloudMachineId() { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public override string ToString() { throw null; } } diff --git a/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.netstandard2.0.cs b/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.netstandard2.0.cs index 8e90eb23698cb..1e6b182bb3f11 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.netstandard2.0.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/api/Azure.CloudMachine.netstandard2.0.cs @@ -53,7 +53,7 @@ public CloudMachineWorkspace(Azure.Core.TokenCredential credential = null, Micro [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public override int GetHashCode() { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public static string ReadOrCreateCmid() { throw null; } + public static string ReadOrCreateCloudMachineId() { throw null; } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public override string ToString() { throw null; } } diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/AppConfigHelpers.cs b/sdk/cloudmachine/Azure.CloudMachine/src/AppConfigHelpers.cs index 64c9a256bbde3..fa4be22d4f3ec 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/AppConfigHelpers.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/AppConfigHelpers.cs @@ -10,7 +10,7 @@ namespace Azure.CloudMachine; internal static class AppConfigHelpers { - internal static string ReadOrCreateCmid() + internal static string ReadOrCreateCloudMachineId() { string appsettings = Path.Combine(".", "appsettings.json"); diff --git a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs index bcc67e99fe5f4..500365b346fcd 100644 --- a/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs +++ b/sdk/cloudmachine/Azure.CloudMachine/src/CloudMachineWorkspace.cs @@ -51,7 +51,7 @@ public CloudMachineWorkspace(TokenCredential credential = default, IConfiguratio Id = configuration switch { - null => AppConfigHelpers.ReadOrCreateCmid(), + null => AppConfigHelpers.ReadOrCreateCloudMachineId(), _ => configuration["CloudMachine:ID"] ?? throw new Exception("CloudMachine:ID configuration value missing") }; } @@ -88,7 +88,7 @@ public override ClientConnection GetConnectionOptions(string connectionId) /// /// [EditorBrowsable(EditorBrowsableState.Never)] - public static string ReadOrCreateCmid() => AppConfigHelpers.ReadOrCreateCmid(); + public static string ReadOrCreateCloudMachineId() => AppConfigHelpers.ReadOrCreateCloudMachineId(); /// [EditorBrowsable(EditorBrowsableState.Never)] diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs deleted file mode 100644 index 6e7ba8f19b01b..0000000000000 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs +++ /dev/null @@ -1,136 +0,0 @@ -namespace Azure.CloudMachine -{ - public static partial class Azd - { - public static void Init(Azure.CloudMachine.CloudMachineClient client, string? infraDirectory = null) { } - public static void Init(Azure.CloudMachine.CloudMachineInfrastructure infra, string? infraDirectory = null) { } - } - public static partial class CloudMachineClientExtensions - { - public static T AddFeature(this Azure.CloudMachine.CloudMachineClient client, T feature) where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } - public static void Configure(this Azure.CloudMachine.CloudMachineClient client, System.Action? configure = null) { } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public static Azure.CloudMachine.CloudMachineInfrastructure GetInfrastructure(this Azure.CloudMachine.CloudMachineClient client) { throw null; } - } - public static partial class CloudMachineCommands - { - public static bool Execute(string[] args, System.Action? configure = null, bool exitProcessIfHandled = true) { throw null; } - } - public partial class CloudMachineInfrastructure - { - public CloudMachineInfrastructure(string? cmId = null) { } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public Azure.CloudMachine.ConnectionCollection Connections { get { throw null; } } - public Azure.CloudMachine.FeatureCollection Features { get { throw null; } } - public string Id { get { throw null; } } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public Azure.Provisioning.Roles.UserAssignedIdentity Identity { get { throw null; } } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public Azure.Provisioning.ProvisioningParameter PrincipalIdParameter { get { throw null; } } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public void AddConstruct(Azure.Provisioning.Primitives.NamedProvisionableConstruct resource) { } - public void AddEndpoints() { } - public T AddFeature(T feature) where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public Azure.Provisioning.ProvisioningPlan Build(Azure.Provisioning.ProvisioningBuildOptions? context = null) { throw null; } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public override bool Equals(object? obj) { throw null; } - public Azure.CloudMachine.CloudMachineClient GetClient() { throw null; } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public override int GetHashCode() { throw null; } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public override string ToString() { throw null; } - } - public partial class FeatureCollection : System.Collections.Generic.IEnumerable, System.Collections.IEnumerable - { - internal FeatureCollection() { } - public System.Collections.Generic.IEnumerable FindAll() where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } - public System.Collections.Generic.IEnumerator GetEnumerator() { throw null; } - System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator() { throw null; } - } - public partial class ServiceBusNamespaceFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public ServiceBusNamespaceFeature(string name, Azure.Provisioning.ServiceBus.ServiceBusSkuName sku = Azure.Provisioning.ServiceBus.ServiceBusSkuName.Standard, Azure.Provisioning.ServiceBus.ServiceBusSkuTier tier = Azure.Provisioning.ServiceBus.ServiceBusSkuTier.Standard) { } - protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } - } - public partial class ServiceBusSubscriptionFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public ServiceBusSubscriptionFeature(string name, Azure.CloudMachine.ServiceBusTopicFeature parent) { } - protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } - } - public partial class ServiceBusTopicFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public ServiceBusTopicFeature(string name, Azure.CloudMachine.ServiceBusNamespaceFeature parent) { } - public string Name { get { throw null; } } - protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } - } -} -namespace Azure.CloudMachine.AppService -{ - public partial class AppServiceFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public AppServiceFeature() { } - public Azure.Provisioning.AppService.AppServiceSkuDescription Sku { get { throw null; } set { } } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } - } -} -namespace Azure.CloudMachine.KeyVault -{ - public partial class KeyVaultFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public KeyVaultFeature(Azure.Provisioning.KeyVault.KeyVaultSku? sku = null) { } - public Azure.Provisioning.KeyVault.KeyVaultSku Sku { get { throw null; } set { } } - protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } - } -} -namespace Azure.CloudMachine.OpenAI -{ - public enum AIModelKind - { - Chat = 0, - Embedding = 1, - } - public partial class OpenAIModelFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public OpenAIModelFeature(string model, string modelVersion, Azure.CloudMachine.OpenAI.AIModelKind kind = Azure.CloudMachine.OpenAI.AIModelKind.Chat) { } - public string Model { get { throw null; } } - public string ModelVersion { get { throw null; } } - protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure cm) { throw null; } - protected internal override void EmitFeatures(Azure.CloudMachine.FeatureCollection features, string cmId) { } - } -} -namespace Azure.Provisioning.CloudMachine -{ - public abstract partial class CloudMachineFeature - { - protected CloudMachineFeature() { } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public Azure.Provisioning.Primitives.ProvisionableResource Emitted { get { throw null; } } - protected internal System.Collections.Generic.Dictionary RequiredSystemRoles { get { throw null; } } - protected internal virtual void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } - protected abstract Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure cm); - protected internal virtual void EmitFeatures(Azure.CloudMachine.FeatureCollection features, string cmId) { } - protected static T EnsureEmits(Azure.Provisioning.CloudMachine.CloudMachineFeature feature) { throw null; } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public override bool Equals(object? obj) { throw null; } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public override int GetHashCode() { throw null; } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public override string ToString() { throw null; } - } -} -namespace System.ClientModel.TypeSpec -{ - public static partial class TypeSpecWriter - { - public static void WriteModel(System.IO.Stream output, System.Type model) { } - public static void WriteModel(System.IO.Stream output) { } - public static void WriteServer(System.IO.Stream output, System.Type service) { } - public static void WriteServer(System.IO.Stream output) { } - } -} diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs deleted file mode 100644 index 6e7ba8f19b01b..0000000000000 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs +++ /dev/null @@ -1,136 +0,0 @@ -namespace Azure.CloudMachine -{ - public static partial class Azd - { - public static void Init(Azure.CloudMachine.CloudMachineClient client, string? infraDirectory = null) { } - public static void Init(Azure.CloudMachine.CloudMachineInfrastructure infra, string? infraDirectory = null) { } - } - public static partial class CloudMachineClientExtensions - { - public static T AddFeature(this Azure.CloudMachine.CloudMachineClient client, T feature) where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } - public static void Configure(this Azure.CloudMachine.CloudMachineClient client, System.Action? configure = null) { } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public static Azure.CloudMachine.CloudMachineInfrastructure GetInfrastructure(this Azure.CloudMachine.CloudMachineClient client) { throw null; } - } - public static partial class CloudMachineCommands - { - public static bool Execute(string[] args, System.Action? configure = null, bool exitProcessIfHandled = true) { throw null; } - } - public partial class CloudMachineInfrastructure - { - public CloudMachineInfrastructure(string? cmId = null) { } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public Azure.CloudMachine.ConnectionCollection Connections { get { throw null; } } - public Azure.CloudMachine.FeatureCollection Features { get { throw null; } } - public string Id { get { throw null; } } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public Azure.Provisioning.Roles.UserAssignedIdentity Identity { get { throw null; } } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public Azure.Provisioning.ProvisioningParameter PrincipalIdParameter { get { throw null; } } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public void AddConstruct(Azure.Provisioning.Primitives.NamedProvisionableConstruct resource) { } - public void AddEndpoints() { } - public T AddFeature(T feature) where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public Azure.Provisioning.ProvisioningPlan Build(Azure.Provisioning.ProvisioningBuildOptions? context = null) { throw null; } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public override bool Equals(object? obj) { throw null; } - public Azure.CloudMachine.CloudMachineClient GetClient() { throw null; } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public override int GetHashCode() { throw null; } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public override string ToString() { throw null; } - } - public partial class FeatureCollection : System.Collections.Generic.IEnumerable, System.Collections.IEnumerable - { - internal FeatureCollection() { } - public System.Collections.Generic.IEnumerable FindAll() where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } - public System.Collections.Generic.IEnumerator GetEnumerator() { throw null; } - System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator() { throw null; } - } - public partial class ServiceBusNamespaceFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public ServiceBusNamespaceFeature(string name, Azure.Provisioning.ServiceBus.ServiceBusSkuName sku = Azure.Provisioning.ServiceBus.ServiceBusSkuName.Standard, Azure.Provisioning.ServiceBus.ServiceBusSkuTier tier = Azure.Provisioning.ServiceBus.ServiceBusSkuTier.Standard) { } - protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } - } - public partial class ServiceBusSubscriptionFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public ServiceBusSubscriptionFeature(string name, Azure.CloudMachine.ServiceBusTopicFeature parent) { } - protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } - } - public partial class ServiceBusTopicFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public ServiceBusTopicFeature(string name, Azure.CloudMachine.ServiceBusNamespaceFeature parent) { } - public string Name { get { throw null; } } - protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } - } -} -namespace Azure.CloudMachine.AppService -{ - public partial class AppServiceFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public AppServiceFeature() { } - public Azure.Provisioning.AppService.AppServiceSkuDescription Sku { get { throw null; } set { } } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } - } -} -namespace Azure.CloudMachine.KeyVault -{ - public partial class KeyVaultFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public KeyVaultFeature(Azure.Provisioning.KeyVault.KeyVaultSku? sku = null) { } - public Azure.Provisioning.KeyVault.KeyVaultSku Sku { get { throw null; } set { } } - protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } - } -} -namespace Azure.CloudMachine.OpenAI -{ - public enum AIModelKind - { - Chat = 0, - Embedding = 1, - } - public partial class OpenAIModelFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature - { - public OpenAIModelFeature(string model, string modelVersion, Azure.CloudMachine.OpenAI.AIModelKind kind = Azure.CloudMachine.OpenAI.AIModelKind.Chat) { } - public string Model { get { throw null; } } - public string ModelVersion { get { throw null; } } - protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } - protected override Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure cm) { throw null; } - protected internal override void EmitFeatures(Azure.CloudMachine.FeatureCollection features, string cmId) { } - } -} -namespace Azure.Provisioning.CloudMachine -{ - public abstract partial class CloudMachineFeature - { - protected CloudMachineFeature() { } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public Azure.Provisioning.Primitives.ProvisionableResource Emitted { get { throw null; } } - protected internal System.Collections.Generic.Dictionary RequiredSystemRoles { get { throw null; } } - protected internal virtual void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } - protected abstract Azure.Provisioning.Primitives.ProvisionableResource EmitConstructs(Azure.CloudMachine.CloudMachineInfrastructure cm); - protected internal virtual void EmitFeatures(Azure.CloudMachine.FeatureCollection features, string cmId) { } - protected static T EnsureEmits(Azure.Provisioning.CloudMachine.CloudMachineFeature feature) { throw null; } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public override bool Equals(object? obj) { throw null; } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public override int GetHashCode() { throw null; } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public override string ToString() { throw null; } - } -} -namespace System.ClientModel.TypeSpec -{ - public static partial class TypeSpecWriter - { - public static void WriteModel(System.IO.Stream output, System.Type model) { } - public static void WriteModel(System.IO.Stream output) { } - public static void WriteServer(System.IO.Stream output, System.Type service) { } - public static void WriteServer(System.IO.Stream output) { } - } -} diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs index c4c3c020d9abb..7fd0408e8025b 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs @@ -43,7 +43,7 @@ public CloudMachineInfrastructure(string? cmId = default) { if (cmId == default) { - cmId = CloudMachineWorkspace.ReadOrCreateCmid(); + cmId = CloudMachineWorkspace.ReadOrCreateCloudMachineId(); } Id = cmId; diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/Azd.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/Azd.cs index d81c906aa07e1..9f5fcc7a1727d 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/Azd.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/Azd.cs @@ -32,7 +32,7 @@ public static void Init(CloudMachineInfrastructure infra, string? infraDirectory Directory.CreateDirectory(infraDirectory); infra.Build().Save(infraDirectory); - var cmid = CloudMachineWorkspace.ReadOrCreateCmid(); + var cmid = CloudMachineWorkspace.ReadOrCreateCloudMachineId(); // main.bicep var location = new ProvisioningParameter("location", typeof(string)); diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/CloudMachineCommands.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/CloudMachineCommands.cs index 334dd97911e80..4685dda5683e5 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/CloudMachineCommands.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/Tooling/CloudMachineCommands.cs @@ -20,7 +20,7 @@ public static bool Execute(string[] args, Action? co if (args.Length < 1) return false; - string cmid = CloudMachineWorkspace.ReadOrCreateCmid(); + string cmid = CloudMachineWorkspace.ReadOrCreateCloudMachineId(); CloudMachineInfrastructure cmi = new(cmid); if (configure != default) { From 4c0cd039c1ded692a46f7a13106a86295c3090ce Mon Sep 17 00:00:00 2001 From: Krzysztof Cwalina Date: Mon, 2 Dec 2024 13:57:17 -0800 Subject: [PATCH 24/24] PR feedback --- .../api/Azure.CloudMachine.Web.net8.0.cs | 8 ++ .../Azure.Provisioning.CloudMachine.net8.0.cs | 136 ++++++++++++++++++ ...rovisioning.CloudMachine.netstandard2.0.cs | 136 ++++++++++++++++++ .../CloudMachineFeature.cs | 12 +- .../CloudMachineInfrastructure.cs | 2 +- .../EventGridSystemTopicFeature.cs | 4 +- .../ServiceBusNamespaceFeature.cs | 6 +- .../ServiceBusSubscriptionFeature.cs | 4 +- .../FeaturesBuiltIn/ServiceBusTopicFeature.cs | 4 +- .../FeaturesBuiltIn/StorageAccountFeature.cs | 25 ++-- .../SystemTopicEventSubscriptionFeature.cs | 6 +- .../FeaturesExtensions/AppServiceFeature.cs | 6 +- .../src/FeaturesExtensions/KeyVaultFeature.cs | 4 +- .../src/FeaturesExtensions/OpenAIFeature.cs | 4 +- .../FeaturesExtensions/OpenAIModelFeature.cs | 12 +- 15 files changed, 325 insertions(+), 44 deletions(-) create mode 100644 sdk/cloudmachine/Azure.CloudMachine.Web/api/Azure.CloudMachine.Web.net8.0.cs create mode 100644 sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs create mode 100644 sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs diff --git a/sdk/cloudmachine/Azure.CloudMachine.Web/api/Azure.CloudMachine.Web.net8.0.cs b/sdk/cloudmachine/Azure.CloudMachine.Web/api/Azure.CloudMachine.Web.net8.0.cs new file mode 100644 index 0000000000000..097b70d30b9b7 --- /dev/null +++ b/sdk/cloudmachine/Azure.CloudMachine.Web/api/Azure.CloudMachine.Web.net8.0.cs @@ -0,0 +1,8 @@ +namespace Azure.CloudMachine +{ + public static partial class CloudMachineExtensions + { + public static void Map(this Microsoft.AspNetCore.Routing.IEndpointRouteBuilder routeBuilder, T serviceImplementation) where T : class { } + public static System.Threading.Tasks.Task UploadFormAsync(this Azure.CloudMachine.StorageServices storage, Microsoft.AspNetCore.Http.HttpRequest multiPartFormData) { throw null; } + } +} diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs new file mode 100644 index 0000000000000..a526056e2055f --- /dev/null +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.net8.0.cs @@ -0,0 +1,136 @@ +namespace Azure.CloudMachine +{ + public static partial class Azd + { + public static void Init(Azure.CloudMachine.CloudMachineClient client, string? infraDirectory = null) { } + public static void Init(Azure.CloudMachine.CloudMachineInfrastructure infra, string? infraDirectory = null) { } + } + public static partial class CloudMachineClientExtensions + { + public static T AddFeature(this Azure.CloudMachine.CloudMachineClient client, T feature) where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } + public static void Configure(this Azure.CloudMachine.CloudMachineClient client, System.Action? configure = null) { } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public static Azure.CloudMachine.CloudMachineInfrastructure GetInfrastructure(this Azure.CloudMachine.CloudMachineClient client) { throw null; } + } + public static partial class CloudMachineCommands + { + public static bool Execute(string[] args, System.Action? configure = null, bool exitProcessIfHandled = true) { throw null; } + } + public partial class CloudMachineInfrastructure + { + public CloudMachineInfrastructure(string? cmId = null) { } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.CloudMachine.ConnectionCollection Connections { get { throw null; } } + public Azure.CloudMachine.FeatureCollection Features { get { throw null; } } + public string Id { get { throw null; } } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.Provisioning.Roles.UserAssignedIdentity Identity { get { throw null; } } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.Provisioning.ProvisioningParameter PrincipalIdParameter { get { throw null; } } + public void AddEndpoints() { } + public T AddFeature(T feature) where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public void AddResource(Azure.Provisioning.Primitives.NamedProvisionableConstruct resource) { } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.Provisioning.ProvisioningPlan Build(Azure.Provisioning.ProvisioningBuildOptions? context = null) { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override bool Equals(object? obj) { throw null; } + public Azure.CloudMachine.CloudMachineClient GetClient() { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override int GetHashCode() { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override string ToString() { throw null; } + } + public partial class FeatureCollection : System.Collections.Generic.IEnumerable, System.Collections.IEnumerable + { + internal FeatureCollection() { } + public System.Collections.Generic.IEnumerable FindAll() where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } + public System.Collections.Generic.IEnumerator GetEnumerator() { throw null; } + System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator() { throw null; } + } + public partial class ServiceBusNamespaceFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature + { + public ServiceBusNamespaceFeature(string name, Azure.Provisioning.ServiceBus.ServiceBusSkuName sku = Azure.Provisioning.ServiceBus.ServiceBusSkuName.Standard, Azure.Provisioning.ServiceBus.ServiceBusSkuTier tier = Azure.Provisioning.ServiceBus.ServiceBusSkuTier.Standard) { } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitResources(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + } + public partial class ServiceBusSubscriptionFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature + { + public ServiceBusSubscriptionFeature(string name, Azure.CloudMachine.ServiceBusTopicFeature parent) { } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitResources(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + } + public partial class ServiceBusTopicFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature + { + public ServiceBusTopicFeature(string name, Azure.CloudMachine.ServiceBusNamespaceFeature parent) { } + public string Name { get { throw null; } } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitResources(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + } +} +namespace Azure.CloudMachine.AppService +{ + public partial class AppServiceFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature + { + public AppServiceFeature() { } + public Azure.Provisioning.AppService.AppServiceSkuDescription Sku { get { throw null; } set { } } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitResources(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + } +} +namespace Azure.CloudMachine.KeyVault +{ + public partial class KeyVaultFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature + { + public KeyVaultFeature(Azure.Provisioning.KeyVault.KeyVaultSku? sku = null) { } + public Azure.Provisioning.KeyVault.KeyVaultSku Sku { get { throw null; } set { } } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitResources(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + } +} +namespace Azure.CloudMachine.OpenAI +{ + public enum AIModelKind + { + Chat = 0, + Embedding = 1, + } + public partial class OpenAIModelFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature + { + public OpenAIModelFeature(string model, string modelVersion, Azure.CloudMachine.OpenAI.AIModelKind kind = Azure.CloudMachine.OpenAI.AIModelKind.Chat) { } + public string Model { get { throw null; } } + public string ModelVersion { get { throw null; } } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected internal override void EmitFeatures(Azure.CloudMachine.FeatureCollection features, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitResources(Azure.CloudMachine.CloudMachineInfrastructure cm) { throw null; } + } +} +namespace Azure.Provisioning.CloudMachine +{ + public abstract partial class CloudMachineFeature + { + protected CloudMachineFeature() { } + protected internal System.Collections.Generic.Dictionary RequiredSystemRoles { get { throw null; } } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.Provisioning.Primitives.ProvisionableResource Resource { get { throw null; } } + protected internal virtual void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected internal virtual void EmitFeatures(Azure.CloudMachine.FeatureCollection features, string cmId) { } + protected abstract Azure.Provisioning.Primitives.ProvisionableResource EmitResources(Azure.CloudMachine.CloudMachineInfrastructure cm); + protected static T EnsureEmits(Azure.Provisioning.CloudMachine.CloudMachineFeature feature) { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override bool Equals(object? obj) { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override int GetHashCode() { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override string ToString() { throw null; } + } +} +namespace System.ClientModel.TypeSpec +{ + public static partial class TypeSpecWriter + { + public static void WriteModel(System.IO.Stream output, System.Type model) { } + public static void WriteModel(System.IO.Stream output) { } + public static void WriteServer(System.IO.Stream output, System.Type service) { } + public static void WriteServer(System.IO.Stream output) { } + } +} diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs new file mode 100644 index 0000000000000..a526056e2055f --- /dev/null +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/api/Azure.Provisioning.CloudMachine.netstandard2.0.cs @@ -0,0 +1,136 @@ +namespace Azure.CloudMachine +{ + public static partial class Azd + { + public static void Init(Azure.CloudMachine.CloudMachineClient client, string? infraDirectory = null) { } + public static void Init(Azure.CloudMachine.CloudMachineInfrastructure infra, string? infraDirectory = null) { } + } + public static partial class CloudMachineClientExtensions + { + public static T AddFeature(this Azure.CloudMachine.CloudMachineClient client, T feature) where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } + public static void Configure(this Azure.CloudMachine.CloudMachineClient client, System.Action? configure = null) { } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public static Azure.CloudMachine.CloudMachineInfrastructure GetInfrastructure(this Azure.CloudMachine.CloudMachineClient client) { throw null; } + } + public static partial class CloudMachineCommands + { + public static bool Execute(string[] args, System.Action? configure = null, bool exitProcessIfHandled = true) { throw null; } + } + public partial class CloudMachineInfrastructure + { + public CloudMachineInfrastructure(string? cmId = null) { } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.CloudMachine.ConnectionCollection Connections { get { throw null; } } + public Azure.CloudMachine.FeatureCollection Features { get { throw null; } } + public string Id { get { throw null; } } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.Provisioning.Roles.UserAssignedIdentity Identity { get { throw null; } } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.Provisioning.ProvisioningParameter PrincipalIdParameter { get { throw null; } } + public void AddEndpoints() { } + public T AddFeature(T feature) where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public void AddResource(Azure.Provisioning.Primitives.NamedProvisionableConstruct resource) { } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.Provisioning.ProvisioningPlan Build(Azure.Provisioning.ProvisioningBuildOptions? context = null) { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override bool Equals(object? obj) { throw null; } + public Azure.CloudMachine.CloudMachineClient GetClient() { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override int GetHashCode() { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override string ToString() { throw null; } + } + public partial class FeatureCollection : System.Collections.Generic.IEnumerable, System.Collections.IEnumerable + { + internal FeatureCollection() { } + public System.Collections.Generic.IEnumerable FindAll() where T : Azure.Provisioning.CloudMachine.CloudMachineFeature { throw null; } + public System.Collections.Generic.IEnumerator GetEnumerator() { throw null; } + System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator() { throw null; } + } + public partial class ServiceBusNamespaceFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature + { + public ServiceBusNamespaceFeature(string name, Azure.Provisioning.ServiceBus.ServiceBusSkuName sku = Azure.Provisioning.ServiceBus.ServiceBusSkuName.Standard, Azure.Provisioning.ServiceBus.ServiceBusSkuTier tier = Azure.Provisioning.ServiceBus.ServiceBusSkuTier.Standard) { } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitResources(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + } + public partial class ServiceBusSubscriptionFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature + { + public ServiceBusSubscriptionFeature(string name, Azure.CloudMachine.ServiceBusTopicFeature parent) { } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitResources(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + } + public partial class ServiceBusTopicFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature + { + public ServiceBusTopicFeature(string name, Azure.CloudMachine.ServiceBusNamespaceFeature parent) { } + public string Name { get { throw null; } } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitResources(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + } +} +namespace Azure.CloudMachine.AppService +{ + public partial class AppServiceFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature + { + public AppServiceFeature() { } + public Azure.Provisioning.AppService.AppServiceSkuDescription Sku { get { throw null; } set { } } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitResources(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + } +} +namespace Azure.CloudMachine.KeyVault +{ + public partial class KeyVaultFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature + { + public KeyVaultFeature(Azure.Provisioning.KeyVault.KeyVaultSku? sku = null) { } + public Azure.Provisioning.KeyVault.KeyVaultSku Sku { get { throw null; } set { } } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitResources(Azure.CloudMachine.CloudMachineInfrastructure infrastructure) { throw null; } + } +} +namespace Azure.CloudMachine.OpenAI +{ + public enum AIModelKind + { + Chat = 0, + Embedding = 1, + } + public partial class OpenAIModelFeature : Azure.Provisioning.CloudMachine.CloudMachineFeature + { + public OpenAIModelFeature(string model, string modelVersion, Azure.CloudMachine.OpenAI.AIModelKind kind = Azure.CloudMachine.OpenAI.AIModelKind.Chat) { } + public string Model { get { throw null; } } + public string ModelVersion { get { throw null; } } + protected internal override void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected internal override void EmitFeatures(Azure.CloudMachine.FeatureCollection features, string cmId) { } + protected override Azure.Provisioning.Primitives.ProvisionableResource EmitResources(Azure.CloudMachine.CloudMachineInfrastructure cm) { throw null; } + } +} +namespace Azure.Provisioning.CloudMachine +{ + public abstract partial class CloudMachineFeature + { + protected CloudMachineFeature() { } + protected internal System.Collections.Generic.Dictionary RequiredSystemRoles { get { throw null; } } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public Azure.Provisioning.Primitives.ProvisionableResource Resource { get { throw null; } } + protected internal virtual void EmitConnections(Azure.CloudMachine.ConnectionCollection connections, string cmId) { } + protected internal virtual void EmitFeatures(Azure.CloudMachine.FeatureCollection features, string cmId) { } + protected abstract Azure.Provisioning.Primitives.ProvisionableResource EmitResources(Azure.CloudMachine.CloudMachineInfrastructure cm); + protected static T EnsureEmits(Azure.Provisioning.CloudMachine.CloudMachineFeature feature) { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override bool Equals(object? obj) { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override int GetHashCode() { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + public override string ToString() { throw null; } + } +} +namespace System.ClientModel.TypeSpec +{ + public static partial class TypeSpecWriter + { + public static void WriteModel(System.IO.Stream output, System.Type model) { } + public static void WriteModel(System.IO.Stream output) { } + public static void WriteServer(System.IO.Stream output, System.Type service) { } + public static void WriteServer(System.IO.Stream output) { } + } +} diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineFeature.cs index 31b2f30600a2a..a43077e5db0b2 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineFeature.cs @@ -13,7 +13,7 @@ public abstract class CloudMachineFeature { private ProvisionableResource? _resource; - protected abstract ProvisionableResource EmitConstructs(CloudMachineInfrastructure cm); + protected abstract ProvisionableResource EmitResources(CloudMachineInfrastructure cm); protected internal virtual void EmitConnections(ConnectionCollection connections, string cmId) { } protected internal virtual void EmitFeatures(FeatureCollection features, string cmId) => features.Add(this); @@ -22,14 +22,14 @@ internal ProvisionableResource Emit(CloudMachineInfrastructure cm) { if (_resource == null) { - ProvisionableResource provisionable = EmitConstructs(cm); - _resource = provisionable; + ProvisionableResource namedResource = EmitResources(cm); + _resource = namedResource; } - return Emitted; + return Resource; } [EditorBrowsable(EditorBrowsableState.Never)] - public ProvisionableResource Emitted { + public ProvisionableResource Resource { get { if (_resource == null) @@ -44,7 +44,7 @@ public ProvisionableResource Emitted { protected static T EnsureEmits(CloudMachineFeature feature) { - if (feature.Emitted is T typed) + if (feature.Resource is T typed) return typed; throw new ArgumentException($"Expected resource of type {typeof(T).Name}, but got {feature.GetType().Name}"); } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs index 7fd0408e8025b..2e3c0adb91383 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/CloudMachineInfrastructure/CloudMachineInfrastructure.cs @@ -81,7 +81,7 @@ public void AddEndpoints() } [EditorBrowsable(EditorBrowsableState.Never)] - public void AddConstruct(NamedProvisionableConstruct resource) + public void AddResource(NamedProvisionableConstruct resource) { _constrcuts.Add(resource); } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/EventGridSystemTopicFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/EventGridSystemTopicFeature.cs index 15eb89af792f9..5f331be310e44 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/EventGridSystemTopicFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/EventGridSystemTopicFeature.cs @@ -13,7 +13,7 @@ namespace Azure.CloudMachine; internal class EventGridSystemTopicFeature(string topicName, CloudMachineFeature source, string topicType) : CloudMachineFeature { - protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitResources(CloudMachineInfrastructure infrastructure) { var topic = new SystemTopic("cm_eventgrid_topic", InternalConstants.EventGridTopicVersion) { @@ -27,7 +27,7 @@ protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructu Name = topicName }; - infrastructure.AddConstruct(topic); + infrastructure.AddResource(topic); return topic; } } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusNamespaceFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusNamespaceFeature.cs index 1da3accf1756b..749b82feadb4a 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusNamespaceFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusNamespaceFeature.cs @@ -11,7 +11,7 @@ namespace Azure.CloudMachine; public class ServiceBusNamespaceFeature(string name, ServiceBusSkuName sku = ServiceBusSkuName.Standard, ServiceBusSkuTier tier = ServiceBusSkuTier.Standard) : CloudMachineFeature { - protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitResources(CloudMachineInfrastructure infrastructure) { var _serviceBusNamespace = new ServiceBusNamespace("cm_servicebus") { @@ -22,8 +22,8 @@ protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructu }, Name = name, }; - infrastructure.AddConstruct(_serviceBusNamespace); - infrastructure.AddConstruct( + infrastructure.AddResource(_serviceBusNamespace); + infrastructure.AddResource( new ServiceBusNamespaceAuthorizationRule("cm_servicebus_auth_rule", "2021-11-01") { Parent = _serviceBusNamespace, diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusSubscriptionFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusSubscriptionFeature.cs index 71d069294a545..dd757020c1844 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusSubscriptionFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusSubscriptionFeature.cs @@ -11,7 +11,7 @@ namespace Azure.CloudMachine; public class ServiceBusSubscriptionFeature(string name, ServiceBusTopicFeature parent) : CloudMachineFeature { - protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitResources(CloudMachineInfrastructure infrastructure) { var subscription = new ServiceBusSubscription(name, "2021-11-01") { @@ -28,7 +28,7 @@ protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructu Status = ServiceBusMessagingEntityStatus.Active }; - infrastructure.AddConstruct(subscription); + infrastructure.AddResource(subscription); return subscription; } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusTopicFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusTopicFeature.cs index de468816cee6f..e547085e41375 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusTopicFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/ServiceBusTopicFeature.cs @@ -17,7 +17,7 @@ public ServiceBusTopicFeature(string name, ServiceBusNamespaceFeature parent) _parent = parent; } - protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitResources(CloudMachineInfrastructure infrastructure) { var topic = new ServiceBusTopic(Name, "2021-11-01") { @@ -31,7 +31,7 @@ protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructu Status = ServiceBusMessagingEntityStatus.Active }; - infrastructure.AddConstruct(topic); + infrastructure.AddResource(topic); return topic; } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/StorageAccountFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/StorageAccountFeature.cs index bb97b2393aa71..772ba9c7eaeb5 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/StorageAccountFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/StorageAccountFeature.cs @@ -17,18 +17,19 @@ namespace Azure.CloudMachine; internal class StorageAccountFeature : CloudMachineFeature { private readonly StorageSkuName _skuName; - private readonly string _name; + public string Name { get; } + public StorageAccountFeature(string accountName, StorageSkuName sku = StorageSkuName.StandardLrs) { _skuName = sku; - _name = accountName; + Name = accountName; } - protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitResources(CloudMachineInfrastructure infrastructure) { var storage = new StorageAccount("cm_storage", StorageAccount.ResourceVersions.V2023_01_01) { - Name = _name, + Name = Name, Kind = StorageKind.StorageV2, Sku = new StorageSku { Name = _skuName }, IsHnsEnabled = true, @@ -39,7 +40,7 @@ protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructu UserAssignedIdentities = { { BicepFunction.Interpolate($"{infrastructure.Identity.Id}").Compile().ToString(), new UserAssignedIdentityDetails() } } } }; - infrastructure.AddConstruct(storage); + infrastructure.AddResource(storage); RequiredSystemRoles.Add(storage, [ @@ -62,14 +63,14 @@ public BlobContainerFeature(BlobServiceFeature parent, string? containerName = d ContainerName = containerName; Parent = parent; } - protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure cm) + protected override ProvisionableResource EmitResources(CloudMachineInfrastructure cm) { BlobContainer container = new($"cm_storage_blobs_container_{ContainerName}", "2023-01-01") { - Parent = (BlobService)Parent.Emitted, + Parent = (BlobService)Parent.Resource, Name = ContainerName }; - cm.AddConstruct(container); + cm.AddResource(container); return container; } @@ -77,7 +78,7 @@ protected internal override void EmitConnections(ConnectionCollection connection { ClientConnection connection = new( $"Azure.Storage.Blobs.BlobContainerClient@{ContainerName}", - $"https://{cmId}.blob.core.windows.net/{ContainerName}" + $"https://{Parent.Account.Name}.blob.core.windows.net/{ContainerName}" ); connections.Add(connection); } @@ -91,13 +92,13 @@ public BlobServiceFeature(StorageAccountFeature account) { Account = account; } - protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure cm) + protected override ProvisionableResource EmitResources(CloudMachineInfrastructure cm) { BlobService blobs = new("cm_storage_blobs") { - Parent = (StorageAccount)Account.Emitted, + Parent = (StorageAccount)Account.Resource, }; - cm.AddConstruct(blobs); + cm.AddResource(blobs); return blobs; } } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/SystemTopicEventSubscriptionFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/SystemTopicEventSubscriptionFeature.cs index 467159d98f0d8..ffc8fe92c4ee0 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/SystemTopicEventSubscriptionFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesBuiltIn/SystemTopicEventSubscriptionFeature.cs @@ -12,7 +12,7 @@ namespace Azure.CloudMachine; internal class SystemTopicEventSubscriptionFeature(string name, EventGridSystemTopicFeature parent, ServiceBusTopicFeature destination, ServiceBusNamespaceFeature parentNamespace) : CloudMachineFeature { - protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitResources(CloudMachineInfrastructure infrastructure) { ServiceBusNamespace serviceBusNamespace = EnsureEmits(parentNamespace); @@ -62,8 +62,8 @@ protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructu }; subscription.DependsOn.Add(roleAssignment); - infrastructure.AddConstruct(subscription); - infrastructure.AddConstruct(roleAssignment); + infrastructure.AddResource(subscription); + infrastructure.AddResource(roleAssignment); return subscription; } } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/AppServiceFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/AppServiceFeature.cs index a0c823686ae44..83d771f27482c 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/AppServiceFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/AppServiceFeature.cs @@ -18,7 +18,7 @@ public AppServiceFeature() Sku = new AppServiceSkuDescription { Tier = "Free", Name = "F1" }; } - protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitResources(CloudMachineInfrastructure infrastructure) { //Add a App Service to the CloudMachine infrastructure. AppServicePlan hostingPlan = new("cm_hosting_plan") @@ -27,7 +27,7 @@ protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructu Sku = Sku, Kind = "app" }; - infrastructure.AddConstruct(hostingPlan); + infrastructure.AddResource(hostingPlan); WebSite appService = new("cm_website") { @@ -59,7 +59,7 @@ protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructu ] } }; - infrastructure.AddConstruct(appService); + infrastructure.AddResource(appService); return appService; } diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/KeyVaultFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/KeyVaultFeature.cs index 92e09a5b50a92..1e52c9d79ce2a 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/KeyVaultFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/KeyVaultFeature.cs @@ -25,7 +25,7 @@ protected internal override void EmitConnections(ConnectionCollection connection connections.Add(new ClientConnection("Azure.Security.KeyVault.Secrets.SecretClient", $"https://{cmId}.vault.azure.net/")); } - protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure infrastructure) + protected override ProvisionableResource EmitResources(CloudMachineInfrastructure infrastructure) { // Add a KeyVault to the CloudMachine infrastructure. KeyVaultService keyVaultResource = new("cm_kv") @@ -48,7 +48,7 @@ protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructu ] }, }; - infrastructure.AddConstruct(keyVaultResource); + infrastructure.AddResource(keyVaultResource); RequiredSystemRoles.Add(keyVaultResource, [(KeyVaultBuiltInRole.GetBuiltInRoleName(KeyVaultBuiltInRole.KeyVaultAdministrator), KeyVaultBuiltInRole.KeyVaultAdministrator.ToString())]); return keyVaultResource; diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIFeature.cs index 4a5911e77b5f6..8ce12b0bf57a5 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIFeature.cs @@ -14,10 +14,10 @@ internal class OpenAIFeature : CloudMachineFeature public OpenAIFeature() { } - protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure cloudMachine) + protected override ProvisionableResource EmitResources(CloudMachineInfrastructure cloudMachine) { CognitiveServicesAccount cognitiveServices = CreateOpenAIAccount(cloudMachine); - cloudMachine.AddConstruct(cognitiveServices); + cloudMachine.AddResource(cognitiveServices); RequiredSystemRoles.Add(cognitiveServices, [(CognitiveServicesBuiltInRole.GetBuiltInRoleName(CognitiveServicesBuiltInRole.CognitiveServicesOpenAIContributor) ,CognitiveServicesBuiltInRole.CognitiveServicesOpenAIContributor.ToString())]); diff --git a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIModelFeature.cs b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIModelFeature.cs index ce5346dd4141e..6efcbd3b98924 100644 --- a/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIModelFeature.cs +++ b/sdk/cloudmachine/Azure.Provisioning.CloudMachine/src/FeaturesExtensions/OpenAIModelFeature.cs @@ -55,10 +55,10 @@ protected internal override void EmitConnections(ConnectionCollection connection } } - protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructure cm) + protected override ProvisionableResource EmitResources(CloudMachineInfrastructure cm) { if (Account == null) throw new InvalidOperationException("Account must be set before emitting"); - if (Account.Emitted == null) throw new InvalidOperationException("Account must be emitted before emitting"); + if (Account.Resource == null) throw new InvalidOperationException("Account must be emitted before emitting"); string name = Kind switch { @@ -67,7 +67,7 @@ protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructu _ => throw new NotImplementedException() }; - CognitiveServicesAccount parent = (CognitiveServicesAccount)Account.Emitted; + CognitiveServicesAccount parent = (CognitiveServicesAccount)Account.Resource; CognitiveServicesAccountDeployment deployment = new($"openai_{name}", "2024-06-01-preview") { Parent = parent, @@ -94,12 +94,12 @@ protected override ProvisionableResource EmitConstructs(CloudMachineInfrastructu OpenAIModelFeature? previous = FindPrevious(cm, this); if (previous != null) { - if (previous.Emitted == null) throw new InvalidOperationException("Previous must be emitted"); - CognitiveServicesAccountDeployment previousDeployment = (CognitiveServicesAccountDeployment)previous.Emitted; + if (previous.Resource == null) throw new InvalidOperationException("Previous must be emitted"); + CognitiveServicesAccountDeployment previousDeployment = (CognitiveServicesAccountDeployment)previous.Resource; deployment.DependsOn.Add(previousDeployment); } - cm.AddConstruct(deployment); + cm.AddResource(deployment); return deployment; OpenAIModelFeature? FindPrevious(CloudMachineInfrastructure cm, OpenAIModelFeature current)