mirror of
https://github.com/MindWorkAI/AI-Studio.git
synced 2026-05-21 12:32:15 +00:00
Using modern C#
This commit is contained in:
parent
c490e0c668
commit
f4eb78159c
@ -88,7 +88,7 @@ public sealed class ProviderAlibabaCloud() : BaseProvider(LLMProviders.ALIBABA_C
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <inhertidoc />
|
/// <inhertidoc />
|
||||||
public override async Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Provider.Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
public override async Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
||||||
{
|
{
|
||||||
var requestedSecret = await RUST_SERVICE.GetAPIKey(this, SecretStoreType.EMBEDDING_PROVIDER);
|
var requestedSecret = await RUST_SERVICE.GetAPIKey(this, SecretStoreType.EMBEDDING_PROVIDER);
|
||||||
return await this.PerformStandardTextEmbeddingRequest(requestedSecret, embeddingModel, token: token, texts: texts);
|
return await this.PerformStandardTextEmbeddingRequest(requestedSecret, embeddingModel, token: token, texts: texts);
|
||||||
|
|||||||
@ -115,9 +115,9 @@ public sealed class ProviderAnthropic() : BaseProvider(LLMProviders.ANTHROPIC, "
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <inhertidoc />
|
/// <inhertidoc />
|
||||||
public override Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Provider.Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
public override Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
||||||
{
|
{
|
||||||
return Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>(Array.Empty<IReadOnlyList<float>>());
|
return Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>([]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
|
|||||||
@ -668,8 +668,8 @@ public abstract class BaseProvider : IProvider, ISecretId
|
|||||||
input = texts,
|
input = texts,
|
||||||
encoding_format = "float"
|
encoding_format = "float"
|
||||||
};
|
};
|
||||||
|
|
||||||
var embeddingRequest = JsonSerializer.Serialize(payload, JSON_SERIALIZER_OPTIONS);
|
var embeddingRequest = JsonSerializer.Serialize(payload, JSON_SERIALIZER_OPTIONS);
|
||||||
|
|
||||||
using var request = new HttpRequestMessage(HttpMethod.Post, host.EmbeddingURL());
|
using var request = new HttpRequestMessage(HttpMethod.Post, host.EmbeddingURL());
|
||||||
|
|
||||||
// Handle the authorization header based on the provider:
|
// Handle the authorization header based on the provider:
|
||||||
@ -685,7 +685,7 @@ public abstract class BaseProvider : IProvider, ISecretId
|
|||||||
if(!requestedSecret.Success)
|
if(!requestedSecret.Success)
|
||||||
{
|
{
|
||||||
this.logger.LogError("No valid API key available for embedding request.");
|
this.logger.LogError("No valid API key available for embedding request.");
|
||||||
return Array.Empty<IReadOnlyList<float>>();
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
request.Headers.Authorization = new AuthenticationHeaderValue("Bearer", await requestedSecret.Secret.Decrypt(ENCRYPTION));
|
request.Headers.Authorization = new AuthenticationHeaderValue("Bearer", await requestedSecret.Secret.Decrypt(ENCRYPTION));
|
||||||
@ -694,34 +694,33 @@ public abstract class BaseProvider : IProvider, ISecretId
|
|||||||
|
|
||||||
// Set the content:
|
// Set the content:
|
||||||
request.Content = new StringContent(embeddingRequest, Encoding.UTF8, "application/json");
|
request.Content = new StringContent(embeddingRequest, Encoding.UTF8, "application/json");
|
||||||
|
|
||||||
using var response = await this.httpClient.SendAsync(request, token);
|
using var response = await this.httpClient.SendAsync(request, token);
|
||||||
var responseBody = response.Content.ReadAsStringAsync(token).Result;
|
var responseBody = response.Content.ReadAsStringAsync(token).Result;
|
||||||
|
|
||||||
if (!response.IsSuccessStatusCode)
|
if (!response.IsSuccessStatusCode)
|
||||||
{
|
{
|
||||||
this.logger.LogError("Embedding request failed with status code {ResponseStatusCode} and body: '{ResponseBody}'.", response.StatusCode, responseBody);
|
this.logger.LogError("Embedding request failed with status code {ResponseStatusCode} and body: '{ResponseBody}'.", response.StatusCode, responseBody);
|
||||||
return Array.Empty<IReadOnlyList<float>>();
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
var embeddingResponse = JsonSerializer.Deserialize<EmbeddingResponse>(responseBody, JSON_SERIALIZER_OPTIONS);
|
var embeddingResponse = JsonSerializer.Deserialize<EmbeddingResponse>(responseBody, JSON_SERIALIZER_OPTIONS);
|
||||||
if (embeddingResponse is { Data: not null })
|
if (embeddingResponse is { Data: not null })
|
||||||
{
|
{
|
||||||
return embeddingResponse.Data
|
return embeddingResponse.Data
|
||||||
.Select(d => d.Embedding?.ToArray() ?? Array.Empty<float>())
|
.Select(d => d.Embedding?.ToArray() ?? [])
|
||||||
.Cast<IReadOnlyList<float>>()
|
.Cast<IReadOnlyList<float>>()
|
||||||
.ToArray();
|
.ToArray();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
this.logger.LogError("Was not able to deserialize the embedding response.");
|
this.logger.LogError("Was not able to deserialize the embedding response.");
|
||||||
return Array.Empty<IReadOnlyList<float>>();
|
return [];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
this.logger.LogError("Failed to perform embedding request: '{Message}'.", e.Message);
|
this.logger.LogError("Failed to perform embedding request: '{Message}'.", e.Message);
|
||||||
return Array.Empty<IReadOnlyList<float>>();
|
return [];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -88,9 +88,9 @@ public sealed class ProviderDeepSeek() : BaseProvider(LLMProviders.DEEP_SEEK, "h
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <inhertidoc />
|
/// <inhertidoc />
|
||||||
public override Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Provider.Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
public override Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
||||||
{
|
{
|
||||||
return Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>(Array.Empty<IReadOnlyList<float>>());
|
return Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>([]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
|
|||||||
@ -90,9 +90,9 @@ public class ProviderFireworks() : BaseProvider(LLMProviders.FIREWORKS, "https:/
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <inhertidoc />
|
/// <inhertidoc />
|
||||||
public override Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Provider.Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
public override Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
||||||
{
|
{
|
||||||
return Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>(Array.Empty<IReadOnlyList<float>>());
|
return Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>([]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
|
|||||||
@ -89,9 +89,9 @@ public sealed class ProviderGWDG() : BaseProvider(LLMProviders.GWDG, "https://ch
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <inhertidoc />
|
/// <inhertidoc />
|
||||||
public override Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Provider.Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
public override Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
||||||
{
|
{
|
||||||
return Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>(Array.Empty<IReadOnlyList<float>>());
|
return Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>([]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
|
|||||||
@ -98,7 +98,7 @@ public class ProviderGoogle() : BaseProvider(LLMProviders.GOOGLE, "https://gener
|
|||||||
if (string.IsNullOrWhiteSpace(modelName))
|
if (string.IsNullOrWhiteSpace(modelName))
|
||||||
{
|
{
|
||||||
LOGGER.LogError("No model name provided for embedding request.");
|
LOGGER.LogError("No model name provided for embedding request.");
|
||||||
return Array.Empty<IReadOnlyList<float>>();
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (modelName.StartsWith("models/", StringComparison.OrdinalIgnoreCase))
|
if (modelName.StartsWith("models/", StringComparison.OrdinalIgnoreCase))
|
||||||
@ -107,7 +107,7 @@ public class ProviderGoogle() : BaseProvider(LLMProviders.GOOGLE, "https://gener
|
|||||||
if (!requestedSecret.Success)
|
if (!requestedSecret.Success)
|
||||||
{
|
{
|
||||||
LOGGER.LogError("No valid API key available for embedding request.");
|
LOGGER.LogError("No valid API key available for embedding request.");
|
||||||
return Array.Empty<IReadOnlyList<float>>();
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
// Prepare the Google Gemini embedding request:
|
// Prepare the Google Gemini embedding request:
|
||||||
@ -117,10 +117,11 @@ public class ProviderGoogle() : BaseProvider(LLMProviders.GOOGLE, "https://gener
|
|||||||
{
|
{
|
||||||
parts = texts.Select(text => new { text }).ToArray()
|
parts = texts.Select(text => new { text }).ToArray()
|
||||||
},
|
},
|
||||||
|
|
||||||
taskType = "SEMANTIC_SIMILARITY"
|
taskType = "SEMANTIC_SIMILARITY"
|
||||||
};
|
};
|
||||||
|
|
||||||
var embeddingRequest = JsonSerializer.Serialize(payload, JSON_SERIALIZER_OPTIONS);
|
var embeddingRequest = JsonSerializer.Serialize(payload, JSON_SERIALIZER_OPTIONS);
|
||||||
|
|
||||||
var embedUrl = $"https://generativelanguage.googleapis.com/v1beta/models/{modelName}:embedContent";
|
var embedUrl = $"https://generativelanguage.googleapis.com/v1beta/models/{modelName}:embedContent";
|
||||||
using var request = new HttpRequestMessage(HttpMethod.Post, embedUrl);
|
using var request = new HttpRequestMessage(HttpMethod.Post, embedUrl);
|
||||||
request.Headers.Add("x-goog-api-key", await requestedSecret.Secret.Decrypt(ENCRYPTION));
|
request.Headers.Add("x-goog-api-key", await requestedSecret.Secret.Decrypt(ENCRYPTION));
|
||||||
@ -134,29 +135,28 @@ public class ProviderGoogle() : BaseProvider(LLMProviders.GOOGLE, "https://gener
|
|||||||
if (!response.IsSuccessStatusCode)
|
if (!response.IsSuccessStatusCode)
|
||||||
{
|
{
|
||||||
LOGGER.LogError("Embedding request failed with status code {ResponseStatusCode} and body: '{ResponseBody}'.", response.StatusCode, responseBody);
|
LOGGER.LogError("Embedding request failed with status code {ResponseStatusCode} and body: '{ResponseBody}'.", response.StatusCode, responseBody);
|
||||||
return Array.Empty<IReadOnlyList<float>>();
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
var embeddingResponse = JsonSerializer.Deserialize<GoogleEmbeddingResponse>(responseBody, JSON_SERIALIZER_OPTIONS);
|
var embeddingResponse = JsonSerializer.Deserialize<GoogleEmbeddingResponse>(responseBody, JSON_SERIALIZER_OPTIONS);
|
||||||
|
|
||||||
if (embeddingResponse is { Embedding: not null })
|
if (embeddingResponse is { Embedding: not null })
|
||||||
{
|
{
|
||||||
return embeddingResponse.Embedding
|
return embeddingResponse.Embedding
|
||||||
.Select(d => d.Values?.ToArray() ?? Array.Empty<float>())
|
.Select(d => d.Values?.ToArray() ?? [])
|
||||||
.Cast<IReadOnlyList<float>>()
|
.Cast<IReadOnlyList<float>>()
|
||||||
.ToArray();
|
.ToArray();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
LOGGER.LogError("Was not able to deserialize the embedding response.");
|
LOGGER.LogError("Was not able to deserialize the embedding response.");
|
||||||
return Array.Empty<IReadOnlyList<float>>();
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
catch (Exception e)
|
catch (Exception e)
|
||||||
{
|
{
|
||||||
LOGGER.LogError("Failed to perform embedding request: '{Message}'.", e.Message);
|
LOGGER.LogError("Failed to perform embedding request: '{Message}'.", e.Message);
|
||||||
return Array.Empty<IReadOnlyList<float>>();
|
return [];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -89,9 +89,9 @@ public class ProviderGroq() : BaseProvider(LLMProviders.GROQ, "https://api.groq.
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <inhertidoc />
|
/// <inhertidoc />
|
||||||
public override Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Provider.Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
public override Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
||||||
{
|
{
|
||||||
return Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>(Array.Empty<IReadOnlyList<float>>());
|
return Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>([]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
|
|||||||
@ -88,7 +88,7 @@ public sealed class ProviderHelmholtz() : BaseProvider(LLMProviders.HELMHOLTZ, "
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <inhertidoc />
|
/// <inhertidoc />
|
||||||
public override async Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Provider.Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
public override async Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
||||||
{
|
{
|
||||||
var requestedSecret = await RUST_SERVICE.GetAPIKey(this, SecretStoreType.EMBEDDING_PROVIDER);
|
var requestedSecret = await RUST_SERVICE.GetAPIKey(this, SecretStoreType.EMBEDDING_PROVIDER);
|
||||||
return await this.PerformStandardTextEmbeddingRequest(requestedSecret, embeddingModel, token: token, texts: texts);
|
return await this.PerformStandardTextEmbeddingRequest(requestedSecret, embeddingModel, token: token, texts: texts);
|
||||||
|
|||||||
@ -93,9 +93,9 @@ public sealed class ProviderHuggingFace : BaseProvider
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <inhertidoc />
|
/// <inhertidoc />
|
||||||
public override Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Provider.Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
public override Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
||||||
{
|
{
|
||||||
return Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>(Array.Empty<IReadOnlyList<float>>());
|
return Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>([]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
|
|||||||
@ -40,7 +40,7 @@ public class NoProvider : IProvider
|
|||||||
|
|
||||||
public Task<string> TranscribeAudioAsync(Model transcriptionModel, string audioFilePath, SettingsManager settingsManager, CancellationToken token = default) => Task.FromResult(string.Empty);
|
public Task<string> TranscribeAudioAsync(Model transcriptionModel, string audioFilePath, SettingsManager settingsManager, CancellationToken token = default) => Task.FromResult(string.Empty);
|
||||||
|
|
||||||
public Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts) => Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>(Array.Empty<IReadOnlyList<float>>());
|
public Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts) => Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>([]);
|
||||||
|
|
||||||
public IReadOnlyCollection<Capability> GetModelCapabilities(Model model) => [ Capability.NONE ];
|
public IReadOnlyCollection<Capability> GetModelCapabilities(Model model) => [ Capability.NONE ];
|
||||||
|
|
||||||
|
|||||||
@ -226,7 +226,7 @@ public sealed class ProviderOpenAI() : BaseProvider(LLMProviders.OPEN_AI, "https
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <inhertidoc />
|
/// <inhertidoc />
|
||||||
public override async Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Provider.Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
public override async Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
||||||
{
|
{
|
||||||
var requestedSecret = await RUST_SERVICE.GetAPIKey(this, SecretStoreType.EMBEDDING_PROVIDER);
|
var requestedSecret = await RUST_SERVICE.GetAPIKey(this, SecretStoreType.EMBEDDING_PROVIDER);
|
||||||
return await this.PerformStandardTextEmbeddingRequest(requestedSecret, embeddingModel, token: token, texts: texts);
|
return await this.PerformStandardTextEmbeddingRequest(requestedSecret, embeddingModel, token: token, texts: texts);
|
||||||
|
|||||||
@ -96,7 +96,7 @@ public sealed class ProviderOpenRouter() : BaseProvider(LLMProviders.OPEN_ROUTER
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <inhertidoc />
|
/// <inhertidoc />
|
||||||
public override async Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Provider.Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
public override async Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
||||||
{
|
{
|
||||||
var requestedSecret = await RUST_SERVICE.GetAPIKey(this, SecretStoreType.EMBEDDING_PROVIDER);
|
var requestedSecret = await RUST_SERVICE.GetAPIKey(this, SecretStoreType.EMBEDDING_PROVIDER);
|
||||||
return await this.PerformStandardTextEmbeddingRequest(requestedSecret, embeddingModel, token: token, texts: texts);
|
return await this.PerformStandardTextEmbeddingRequest(requestedSecret, embeddingModel, token: token, texts: texts);
|
||||||
|
|||||||
@ -96,9 +96,9 @@ public sealed class ProviderPerplexity() : BaseProvider(LLMProviders.PERPLEXITY,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <inhertidoc />
|
/// <inhertidoc />
|
||||||
public override Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Provider.Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
public override Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
||||||
{
|
{
|
||||||
return Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>(Array.Empty<IReadOnlyList<float>>());
|
return Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>([]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
|
|||||||
@ -89,9 +89,9 @@ public sealed class ProviderX() : BaseProvider(LLMProviders.X, "https://api.x.ai
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <inhertidoc />
|
/// <inhertidoc />
|
||||||
public override Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Provider.Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
public override Task<IReadOnlyList<IReadOnlyList<float>>> EmbedTextAsync(Model embeddingModel, SettingsManager settingsManager, CancellationToken token = default, params List<string> texts)
|
||||||
{
|
{
|
||||||
return Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>(Array.Empty<IReadOnlyList<float>>());
|
return Task.FromResult<IReadOnlyList<IReadOnlyList<float>>>([]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user