File: Settings\TemplateInfo.cs
Web Access
Project: src\src\sdk\src\TemplateEngine\Microsoft.TemplateEngine.Edge\Microsoft.TemplateEngine.Edge.csproj (Microsoft.TemplateEngine.Edge)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.

using System.Text.Json.Nodes;
using System.Text.Json.Serialization;
using Microsoft.TemplateEngine.Abstractions;
using Microsoft.TemplateEngine.Abstractions.Constraints;
using Microsoft.TemplateEngine.Abstractions.Parameters;
using Microsoft.TemplateEngine.Utils;

namespace Microsoft.TemplateEngine.Edge.Settings
{
    internal partial class TemplateInfo : ITemplateInfo, ITemplateInfoHostJsonCache
    {
        internal const string CurrentVersion = "1.0.0.7";

#pragma warning disable CS0618 // Type or member is obsolete
        private IReadOnlyDictionary<string, ICacheTag>? _tags;
        private IReadOnlyDictionary<string, ICacheParameter>? _cacheParameters;
#pragma warning restore CS0618 // Type or member is obsolete

        internal TemplateInfo(string identity, string name, IEnumerable<string> shortNames, string mountPointUri, string configPlace)
        {
            if (string.IsNullOrWhiteSpace(identity))
            {
                throw new ArgumentException($"'{nameof(identity)}' cannot be null or whitespace.", nameof(identity));
            }

            if (string.IsNullOrWhiteSpace(name))
            {
                throw new ArgumentException($"'{nameof(name)}' cannot be null or whitespace.", nameof(name));
            }

            if (shortNames is null)
            {
                throw new ArgumentNullException(nameof(shortNames));
            }
            if (!shortNames.Any())
            {
                throw new ArgumentException($"'{nameof(shortNames)}' should contain at least one entry.", nameof(shortNames));
            }
            if (shortNames.Any(string.IsNullOrWhiteSpace))
            {
                throw new ArgumentException($"'{nameof(shortNames)}' should not contain empty values.", nameof(shortNames));
            }

            if (string.IsNullOrWhiteSpace(mountPointUri))
            {
                throw new ArgumentException($"'{nameof(mountPointUri)}' cannot be null or whitespace.", nameof(mountPointUri));
            }

            if (string.IsNullOrWhiteSpace(configPlace))
            {
                throw new ArgumentException($"'{nameof(configPlace)}' cannot be null or whitespace.", nameof(configPlace));
            }

            Identity = identity;
            Name = name;
            MountPointUri = mountPointUri;
            ConfigPlace = configPlace;
            ShortNameList = shortNames.ToList();
        }

        /// <summary>
        /// Localization copy-constructor.
        /// </summary>
        /// <param name="template">unlocalized template.</param>
        /// <param name="localizationInfo">localization information.</param>
        /// <param name="hostConfig">host config information.</param>
        internal TemplateInfo(IScanTemplateInfo template, ILocalizationLocator? localizationInfo, (string Path, JsonObject? Content)? hostConfig)
        {
            if (template is null)
            {
                throw new ArgumentNullException(nameof(template));
            }
            GeneratorId = template.GeneratorId;
            ConfigPlace = template.ConfigPlace;
            MountPointUri = template.MountPointUri;
            TagsCollection = template.TagsCollection;
            Classifications = template.Classifications;
            GroupIdentity = template.GroupIdentity;
            Precedence = template.Precedence;
            Identity = template.Identity;
            DefaultName = template.DefaultName;
            PreferDefaultName = template.PreferDefaultName;
            HostConfigPlace = hostConfig?.Path;
            ThirdPartyNotices = template.ThirdPartyNotices;
            BaselineInfo = template.BaselineInfo;
            ShortNameList = template.ShortNameList;
            PostActions = template.PostActions;
            Constraints = template.Constraints;

            LocaleConfigPlace = localizationInfo?.ConfigPlace;

            Author = localizationInfo?.Author ?? template.Author;
            Description = localizationInfo?.Description ?? template.Description;

            Name = localizationInfo?.Name ?? template.Name;
            ParameterDefinitions = LocalizeParameters(template, localizationInfo);
            HostData = hostConfig?.Content?.ToJsonString();
        }

#pragma warning disable CS0618 // Type or member is obsolete
        [JsonPropertyName("Parameters")]
#pragma warning restore CS0618 // Type or member is obsolete
        public IParameterDefinitionSet ParameterDefinitions { get; private set; } = ParameterDefinitionSet.Empty;

        [JsonIgnore]
        [Obsolete("Use ParameterDefinitionSet instead.")]
        public IReadOnlyList<ITemplateParameter> Parameters => ParameterDefinitions;

        [JsonPropertyName("MountPointUri")]
        public string MountPointUri { get; }

        [JsonPropertyName("Author")]
        public string? Author { get; private set; }

        [JsonPropertyName("Classifications")]
        public IReadOnlyList<string> Classifications { get; private set; } = new List<string>();

        [JsonPropertyName("DefaultName")]
        public string? DefaultName { get; private set; }

        [JsonPropertyName("Description")]
        public string? Description { get; private set; }

        [JsonPropertyName("Identity")]
        public string Identity { get; }

        [JsonPropertyName("GeneratorId")]
        public Guid GeneratorId { get; private set; }

        [JsonPropertyName("GroupIdentity")]
        public string? GroupIdentity { get; private set; }

        [JsonPropertyName("Precedence")]
        public int Precedence { get; private set; }

        [JsonPropertyName("Name")]
        public string Name { get; }

        [JsonIgnore]
        [Obsolete]
        string ITemplateInfo.ShortName
        {
            get
            {
                if (ShortNameList.Count > 0)
                {
                    return ShortNameList[0];
                }

                return string.Empty;
            }
        }

        public IReadOnlyList<string> ShortNameList { get; } = new List<string>();

        [JsonPropertyName("PreferDefaultName")]
        public bool PreferDefaultName { get; private set; }

        [JsonIgnore]
        [Obsolete]
        IReadOnlyDictionary<string, ICacheTag> ITemplateInfo.Tags
        {
            get
            {
                if (_tags == null)
                {
                    Dictionary<string, ICacheTag> tags = new Dictionary<string, ICacheTag>();
                    foreach (KeyValuePair<string, string> tag in TagsCollection)
                    {
                        tags[tag.Key] = new CacheTag(null, null, new Dictionary<string, ParameterChoice> { { tag.Value, new ParameterChoice(null, null) } }, tag.Value);
                    }
                    foreach (ITemplateParameter parameter in ParameterDefinitions.Where(TemplateParameterExtensions.IsChoice))
                    {
                        IReadOnlyDictionary<string, ParameterChoice> choices = parameter.Choices ?? new Dictionary<string, ParameterChoice>();
                        tags[parameter.Name] = new CacheTag(parameter.DisplayName, parameter.Documentation, choices, parameter.DefaultValue);
                    }
                    return _tags = tags;
                }
                return _tags;
            }
        }

        [JsonIgnore]
        [Obsolete]
        IReadOnlyDictionary<string, ICacheParameter> ITemplateInfo.CacheParameters
        {
            get
            {
                if (_cacheParameters == null)
                {
                    Dictionary<string, ICacheParameter> cacheParameters = new Dictionary<string, ICacheParameter>();
                    foreach (ITemplateParameter parameter in ParameterDefinitions.Where(p => !p.IsChoice()))
                    {
                        cacheParameters[parameter.Name] = new CacheParameter()
                        {
                            DataType = parameter.DataType,
                            DefaultValue = parameter.DefaultValue,
                            Description = parameter.Documentation,
                            DefaultIfOptionWithoutValue = parameter.DefaultIfOptionWithoutValue,
                            DisplayName = parameter.DisplayName

                        };
                    }
                    return _cacheParameters = cacheParameters;
                }
                return _cacheParameters;
            }
        }

        [JsonPropertyName("ConfigPlace")]
        public string ConfigPlace { get; }

        [JsonPropertyName("LocaleConfigPlace")]
        public string? LocaleConfigPlace { get; private set; }

        [JsonPropertyName("HostConfigPlace")]
        public string? HostConfigPlace { get; private set; }

        [JsonPropertyName("ThirdPartyNotices")]
        public string? ThirdPartyNotices { get; private set; }

        [JsonPropertyName("BaselineInfo")]
        public IReadOnlyDictionary<string, IBaselineInfo> BaselineInfo { get; private set; } = new Dictionary<string, IBaselineInfo>();

        [JsonPropertyName("TagsCollection")]
        public IReadOnlyDictionary<string, string> TagsCollection { get; private set; } = new Dictionary<string, string>();

        [JsonIgnore]
        bool ITemplateInfo.HasScriptRunningPostActions { get; set; }

        public string? HostData { get; private set; }

        [JsonPropertyName("PostActions")]
        public IReadOnlyList<Guid> PostActions { get; private set; } = [];

        [JsonPropertyName("Constraints")]
        public IReadOnlyList<TemplateConstraintInfo> Constraints { get; private set; } = [];

        public static TemplateInfo FromJObject(JsonObject entry)
        {
            return TemplateInfoReader.FromJObject(entry);
        }

        private static IParameterDefinitionSet LocalizeParameters(IScanTemplateInfo template, ILocalizationLocator? localizationInfo)
        {
            //we would like to copy the parameters to format supported for serialization as we cannot be sure that ITemplateInfo supports serialization in needed format.
            List<ITemplateParameter> localizedParameters = new List<ITemplateParameter>();

            foreach (ITemplateParameter parameter in template.ParameterDefinitions)
            {
                IParameterSymbolLocalizationModel? localization = null;
                Dictionary<string, ParameterChoice>? localizedChoices = null;
                if (localizationInfo != null)
                {
                    if (!localizationInfo.ParameterSymbols.TryGetValue(parameter.Name, out localization))
                    {
                        // There is no localization for this symbol. Use the symbol as is.
                        localizedParameters.Add(parameter);
                        continue;
                    }
                    if (parameter.IsChoice() && parameter.Choices != null)
                    {
                        localizedChoices = new Dictionary<string, ParameterChoice>();
                        foreach (KeyValuePair<string, ParameterChoice> templateChoice in parameter.Choices)
                        {
                            ParameterChoice localizedChoice = new ParameterChoice(
                                templateChoice.Value.DisplayName,
                                templateChoice.Value.Description);

                            if (localization.Choices.TryGetValue(templateChoice.Key, out ParameterChoiceLocalizationModel locModel))
                            {
                                localizedChoice.Localize(locModel);
                            }
                            localizedChoices.Add(templateChoice.Key, localizedChoice);
                        }
                    }
                }

                TemplateParameter localizedParameter = new TemplateParameter(
                    name: parameter.Name,
                    displayName: localization?.DisplayName ?? parameter.DisplayName,
                    description: localization?.Description ?? parameter.Description,
                    defaultValue: parameter.DefaultValue,
                    defaultIfOptionWithoutValue: parameter.DefaultIfOptionWithoutValue,
                    datatype: parameter.DataType,
                    precedence: parameter.Precedence,
                    type: parameter.Type,
                    allowMultipleValues: parameter.AllowMultipleValues,
                    choices: localizedChoices ?? parameter.Choices);

                localizedParameters.Add(localizedParameter);
            }
            return new ParameterDefinitionSet(localizedParameters);
        }
    }
}