feat: refactor external input handling code
- refactor import/export code - make import/export use the same JSON parsing as API - make Patch.AssertIsValid actually useful
This commit is contained in:
		@@ -31,14 +31,14 @@ namespace PluralKit.Core
 | 
			
		||||
            .With("list_privacy", ListPrivacy)
 | 
			
		||||
            .With("visibility", Visibility);
 | 
			
		||||
 | 
			
		||||
        public new void CheckIsValid()
 | 
			
		||||
        public new void AssertIsValid()
 | 
			
		||||
        {
 | 
			
		||||
            if (Icon.Value != null && !MiscUtils.TryMatchUri(Icon.Value, out var avatarUri))
 | 
			
		||||
                throw new InvalidPatchException("icon");
 | 
			
		||||
                throw new ValidationError("icon");
 | 
			
		||||
            if (BannerImage.Value != null && !MiscUtils.TryMatchUri(BannerImage.Value, out var bannerImage))
 | 
			
		||||
                throw new InvalidPatchException("banner");
 | 
			
		||||
                throw new ValidationError("banner");
 | 
			
		||||
            if (Color.Value != null && (!Regex.IsMatch(Color.Value, "^[0-9a-fA-F]{6}$")))
 | 
			
		||||
                throw new InvalidPatchException("color");
 | 
			
		||||
                throw new ValidationError("color");
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
 
 | 
			
		||||
@@ -53,14 +53,28 @@ namespace PluralKit.Core
 | 
			
		||||
            .With("avatar_privacy", AvatarPrivacy)
 | 
			
		||||
            .With("metadata_privacy", MetadataPrivacy);
 | 
			
		||||
 | 
			
		||||
        public new void CheckIsValid()
 | 
			
		||||
        public new void AssertIsValid()
 | 
			
		||||
        {
 | 
			
		||||
            if (AvatarUrl.Value != null && !MiscUtils.TryMatchUri(AvatarUrl.Value, out var avatarUri))
 | 
			
		||||
                throw new InvalidPatchException("avatar_url");
 | 
			
		||||
            if (BannerImage.Value != null && !MiscUtils.TryMatchUri(BannerImage.Value, out var bannerImage))
 | 
			
		||||
                throw new InvalidPatchException("banner");
 | 
			
		||||
            if (Color.Value != null && (!Regex.IsMatch(Color.Value, "^[0-9a-fA-F]{6}$")))
 | 
			
		||||
                throw new InvalidPatchException("color");
 | 
			
		||||
            if (Name.IsPresent)
 | 
			
		||||
                AssertValid(Name.Value, "display_name", Limits.MaxMemberNameLength);
 | 
			
		||||
            if (DisplayName.Value != null)
 | 
			
		||||
                AssertValid(DisplayName.Value, "display_name", Limits.MaxMemberNameLength);
 | 
			
		||||
            if (AvatarUrl.Value != null)
 | 
			
		||||
                AssertValid(AvatarUrl.Value, "avatar_url", Limits.MaxUriLength,
 | 
			
		||||
                    s => MiscUtils.TryMatchUri(s, out var avatarUri));
 | 
			
		||||
            if (BannerImage.Value != null)
 | 
			
		||||
                AssertValid(BannerImage.Value, "banner", Limits.MaxUriLength,
 | 
			
		||||
                    s => MiscUtils.TryMatchUri(s, out var bannerUri));
 | 
			
		||||
            if (Color.Value != null)
 | 
			
		||||
                AssertValid(Color.Value, "color", "^[0-9a-fA-F]{6}$");
 | 
			
		||||
            if (Pronouns.Value != null)
 | 
			
		||||
                AssertValid(Pronouns.Value, "pronouns", Limits.MaxPronounsLength);
 | 
			
		||||
            if (Description.Value != null)
 | 
			
		||||
                AssertValid(Description.Value, "description", Limits.MaxDescriptionLength);
 | 
			
		||||
            if (ProxyTags.IsPresent && (ProxyTags.Value.Length > 100 ||
 | 
			
		||||
                                        ProxyTags.Value.Any(tag => tag.ProxyString.IsLongerThan(100))))
 | 
			
		||||
                // todo: have a better error for this
 | 
			
		||||
                throw new ValidationError("proxy_tags");
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
#nullable disable
 | 
			
		||||
@@ -70,13 +84,13 @@ namespace PluralKit.Core
 | 
			
		||||
            var patch = new MemberPatch();
 | 
			
		||||
 | 
			
		||||
            if (o.ContainsKey("name") && o["name"].Type == JTokenType.Null) 
 | 
			
		||||
                throw new JsonModelParseError("Member name can not be set to null.");
 | 
			
		||||
                throw new ValidationError("Member name can not be set to null.");
 | 
			
		||||
            
 | 
			
		||||
            if (o.ContainsKey("name")) patch.Name = o.Value<string>("name").BoundsCheckField(Limits.MaxMemberNameLength, "Member name");
 | 
			
		||||
            if (o.ContainsKey("name")) patch.Name = o.Value<string>("name");
 | 
			
		||||
            if (o.ContainsKey("color")) patch.Color = o.Value<string>("color").NullIfEmpty()?.ToLower();
 | 
			
		||||
            if (o.ContainsKey("display_name")) patch.DisplayName = o.Value<string>("display_name").NullIfEmpty().BoundsCheckField(Limits.MaxMemberNameLength, "Member display name");
 | 
			
		||||
            if (o.ContainsKey("avatar_url")) patch.AvatarUrl = o.Value<string>("avatar_url").NullIfEmpty().BoundsCheckField(Limits.MaxUriLength, "Member avatar URL");
 | 
			
		||||
            if (o.ContainsKey("banner")) patch.BannerImage = o.Value<string>("banner").NullIfEmpty().BoundsCheckField(Limits.MaxUriLength, "Member banner URL");
 | 
			
		||||
            if (o.ContainsKey("display_name")) patch.DisplayName = o.Value<string>("display_name").NullIfEmpty();
 | 
			
		||||
            if (o.ContainsKey("avatar_url")) patch.AvatarUrl = o.Value<string>("avatar_url").NullIfEmpty();
 | 
			
		||||
            if (o.ContainsKey("banner")) patch.BannerImage = o.Value<string>("banner").NullIfEmpty();
 | 
			
		||||
 | 
			
		||||
            if (o.ContainsKey("birthday"))
 | 
			
		||||
            {
 | 
			
		||||
@@ -84,26 +98,25 @@ namespace PluralKit.Core
 | 
			
		||||
                var res = DateTimeFormats.DateExportFormat.Parse(str);
 | 
			
		||||
                if (res.Success) patch.Birthday = res.Value;
 | 
			
		||||
                else if (str == null) patch.Birthday = null;
 | 
			
		||||
                else throw new JsonModelParseError("Could not parse member birthday.");
 | 
			
		||||
                else throw new ValidationError("birthday");
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            if (o.ContainsKey("pronouns")) patch.Pronouns = o.Value<string>("pronouns").NullIfEmpty().BoundsCheckField(Limits.MaxPronounsLength, "Member pronouns");
 | 
			
		||||
            if (o.ContainsKey("description")) patch.Description = o.Value<string>("description").NullIfEmpty().BoundsCheckField(Limits.MaxDescriptionLength, "Member descriptoin");
 | 
			
		||||
            if (o.ContainsKey("pronouns")) patch.Pronouns = o.Value<string>("pronouns").NullIfEmpty();
 | 
			
		||||
            if (o.ContainsKey("description")) patch.Description = o.Value<string>("description").NullIfEmpty();
 | 
			
		||||
            if (o.ContainsKey("keep_proxy")) patch.KeepProxy = o.Value<bool>("keep_proxy");
 | 
			
		||||
 | 
			
		||||
            // legacy: used in old export files and APIv1
 | 
			
		||||
            // todo: should we parse `proxy_tags` first?
 | 
			
		||||
            if (o.ContainsKey("prefix") || o.ContainsKey("suffix") && !o.ContainsKey("proxy_tags"))
 | 
			
		||||
                patch.ProxyTags = new[] {new ProxyTag(o.Value<string>("prefix"), o.Value<string>("suffix"))};
 | 
			
		||||
            else if (o.ContainsKey("proxy_tags"))
 | 
			
		||||
            {
 | 
			
		||||
                patch.ProxyTags = o.Value<JArray>("proxy_tags")
 | 
			
		||||
                    .OfType<JObject>().Select(o => new ProxyTag(o.Value<string>("prefix"), o.Value<string>("suffix")))
 | 
			
		||||
                    .Where(p => p.Valid)
 | 
			
		||||
                    .ToArray();
 | 
			
		||||
            }
 | 
			
		||||
            
 | 
			
		||||
            if(o.ContainsKey("privacy")) //TODO: Deprecate this completely in api v2
 | 
			
		||||
            {
 | 
			
		||||
                var plevel = o.Value<string>("privacy").ParsePrivacy("member");
 | 
			
		||||
                var plevel = o.ParsePrivacy("privacy");
 | 
			
		||||
                                
 | 
			
		||||
                patch.Visibility = plevel;
 | 
			
		||||
                patch.NamePrivacy = plevel;
 | 
			
		||||
@@ -116,14 +129,14 @@ namespace PluralKit.Core
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                if (o.ContainsKey("visibility")) patch.Visibility = o.Value<string>("visibility").ParsePrivacy("member");
 | 
			
		||||
                if (o.ContainsKey("name_privacy")) patch.NamePrivacy = o.Value<string>("name_privacy").ParsePrivacy("member");
 | 
			
		||||
                if (o.ContainsKey("description_privacy")) patch.DescriptionPrivacy = o.Value<string>("description_privacy").ParsePrivacy("member");
 | 
			
		||||
                if (o.ContainsKey("avatar_privacy")) patch.AvatarPrivacy = o.Value<string>("avatar_privacy").ParsePrivacy("member");
 | 
			
		||||
                if (o.ContainsKey("birthday_privacy")) patch.BirthdayPrivacy = o.Value<string>("birthday_privacy").ParsePrivacy("member");
 | 
			
		||||
                if (o.ContainsKey("pronoun_privacy")) patch.PronounPrivacy = o.Value<string>("pronoun_privacy").ParsePrivacy("member");
 | 
			
		||||
                // if (o.ContainsKey("color_privacy")) member.ColorPrivacy = o.Value<string>("color_privacy").ParsePrivacy("member");
 | 
			
		||||
                if (o.ContainsKey("metadata_privacy")) patch.MetadataPrivacy = o.Value<string>("metadata_privacy").ParsePrivacy("member");
 | 
			
		||||
                if (o.ContainsKey("visibility")) patch.Visibility = o.ParsePrivacy("visibility");
 | 
			
		||||
                if (o.ContainsKey("name_privacy")) patch.NamePrivacy = o.ParsePrivacy("name_privacy");
 | 
			
		||||
                if (o.ContainsKey("description_privacy")) patch.DescriptionPrivacy = o.ParsePrivacy("description_privacy");
 | 
			
		||||
                if (o.ContainsKey("avatar_privacy")) patch.AvatarPrivacy = o.ParsePrivacy("avatar_privacy");
 | 
			
		||||
                if (o.ContainsKey("birthday_privacy")) patch.BirthdayPrivacy = o.ParsePrivacy("birthday_privacy");
 | 
			
		||||
                if (o.ContainsKey("pronoun_privacy")) patch.PronounPrivacy = o.ParsePrivacy("pronoun_privacy");
 | 
			
		||||
                // if (o.ContainsKey("color_privacy")) member.ColorPrivacy = o.ParsePrivacy("member");
 | 
			
		||||
                if (o.ContainsKey("metadata_privacy")) patch.MetadataPrivacy = o.ParsePrivacy("metadata_privacy");
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            return patch;
 | 
			
		||||
 
 | 
			
		||||
@@ -1,17 +1,48 @@
 | 
			
		||||
using System;
 | 
			
		||||
using System.Text.RegularExpressions;
 | 
			
		||||
 | 
			
		||||
namespace PluralKit.Core
 | 
			
		||||
{
 | 
			
		||||
 | 
			
		||||
    public class InvalidPatchException : Exception
 | 
			
		||||
    {
 | 
			
		||||
        public InvalidPatchException(string message) : base(message) {}
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public abstract class PatchObject
 | 
			
		||||
    {
 | 
			
		||||
        public abstract UpdateQueryBuilder Apply(UpdateQueryBuilder b);
 | 
			
		||||
 | 
			
		||||
        public void CheckIsValid() {}
 | 
			
		||||
        public void AssertIsValid() {}
 | 
			
		||||
 | 
			
		||||
        protected bool AssertValid(string input, string name, int maxLength, Func<string, bool>? validate = null)
 | 
			
		||||
        {
 | 
			
		||||
            if (input.Length > maxLength)
 | 
			
		||||
                throw new FieldTooLongError(name, maxLength, input.Length);
 | 
			
		||||
            if (validate != null && !validate(input))
 | 
			
		||||
                throw new ValidationError(name);
 | 
			
		||||
            return true;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        protected bool AssertValid(string input, string name, string pattern)
 | 
			
		||||
        {
 | 
			
		||||
            if (!Regex.IsMatch(input, pattern))
 | 
			
		||||
                throw new ValidationError(name);
 | 
			
		||||
            return true;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public class ValidationError: Exception
 | 
			
		||||
    {
 | 
			
		||||
        public ValidationError(string message): base(message) { }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public class FieldTooLongError: ValidationError
 | 
			
		||||
    {
 | 
			
		||||
        public string Name;
 | 
			
		||||
        public int MaxLength;
 | 
			
		||||
        public int ActualLength;
 | 
			
		||||
 | 
			
		||||
        public FieldTooLongError(string name, int maxLength, int actualLength):
 | 
			
		||||
            base($"{name} too long ({actualLength} > {maxLength})")
 | 
			
		||||
        {
 | 
			
		||||
            Name = name;
 | 
			
		||||
            MaxLength = maxLength;
 | 
			
		||||
            ActualLength = actualLength;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,8 +1,11 @@
 | 
			
		||||
#nullable enable
 | 
			
		||||
using System;
 | 
			
		||||
using System.Text.RegularExpressions;
 | 
			
		||||
 | 
			
		||||
using Newtonsoft.Json.Linq;
 | 
			
		||||
 | 
			
		||||
using NodaTime;
 | 
			
		||||
 | 
			
		||||
namespace PluralKit.Core
 | 
			
		||||
{
 | 
			
		||||
    public class SystemPatch: PatchObject
 | 
			
		||||
@@ -46,34 +49,44 @@ namespace PluralKit.Core
 | 
			
		||||
            .With("member_limit_override", MemberLimitOverride)
 | 
			
		||||
            .With("group_limit_override", GroupLimitOverride);
 | 
			
		||||
 | 
			
		||||
        public new void CheckIsValid()
 | 
			
		||||
        public new void AssertIsValid()
 | 
			
		||||
        {
 | 
			
		||||
            if (AvatarUrl.Value != null && !MiscUtils.TryMatchUri(AvatarUrl.Value, out var avatarUri))
 | 
			
		||||
                throw new InvalidPatchException("avatar_url");
 | 
			
		||||
            if (BannerImage.Value != null && !MiscUtils.TryMatchUri(BannerImage.Value, out var bannerImage))
 | 
			
		||||
                throw new InvalidPatchException("banner");
 | 
			
		||||
            if (Color.Value != null && (!Regex.IsMatch(Color.Value, "^[0-9a-fA-F]{6}$")))
 | 
			
		||||
                throw new InvalidPatchException("color");
 | 
			
		||||
            if (Name.Value != null)
 | 
			
		||||
                AssertValid(Name.Value, "name", Limits.MaxSystemNameLength);
 | 
			
		||||
            if (Description.Value != null)
 | 
			
		||||
                AssertValid(Description.Value, "description", Limits.MaxDescriptionLength);
 | 
			
		||||
            if (Tag.Value != null)
 | 
			
		||||
                AssertValid(Tag.Value, "tag", Limits.MaxSystemTagLength);
 | 
			
		||||
            if (AvatarUrl.Value != null)
 | 
			
		||||
                AssertValid(AvatarUrl.Value, "avatar_url", Limits.MaxUriLength,
 | 
			
		||||
                    s => MiscUtils.TryMatchUri(s, out var avatarUri));
 | 
			
		||||
            if (BannerImage.Value != null)
 | 
			
		||||
                AssertValid(BannerImage.Value, "banner", Limits.MaxUriLength,
 | 
			
		||||
                    s => MiscUtils.TryMatchUri(s, out var bannerUri));
 | 
			
		||||
            if (Color.Value != null)
 | 
			
		||||
                AssertValid(Color.Value, "color", "^[0-9a-fA-F]{6}$");
 | 
			
		||||
            if (UiTz.IsPresent && DateTimeZoneProviders.Tzdb.GetZoneOrNull(UiTz.Value) == null)
 | 
			
		||||
                throw new ValidationError("avatar_url");
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        public static SystemPatch FromJSON(JObject o)
 | 
			
		||||
        {
 | 
			
		||||
            var patch = new SystemPatch();
 | 
			
		||||
            if (o.ContainsKey("name")) patch.Name = o.Value<string>("name").NullIfEmpty().BoundsCheckField(Limits.MaxSystemNameLength, "System name");
 | 
			
		||||
            if (o.ContainsKey("description")) patch.Description = o.Value<string>("description").NullIfEmpty().BoundsCheckField(Limits.MaxDescriptionLength, "System description");
 | 
			
		||||
            if (o.ContainsKey("tag")) patch.Tag = o.Value<string>("tag").NullIfEmpty().BoundsCheckField(Limits.MaxSystemTagLength, "System tag");
 | 
			
		||||
            if (o.ContainsKey("avatar_url")) patch.AvatarUrl = o.Value<string>("avatar_url").NullIfEmpty().BoundsCheckField(Limits.MaxUriLength, "System avatar URL");
 | 
			
		||||
            if (o.ContainsKey("banner")) patch.BannerImage = o.Value<string>("banner").NullIfEmpty().BoundsCheckField(Limits.MaxUriLength, "System banner URL");
 | 
			
		||||
            if (o.ContainsKey("name")) patch.Name = o.Value<string>("name").NullIfEmpty();
 | 
			
		||||
            if (o.ContainsKey("description")) patch.Description = o.Value<string>("description").NullIfEmpty();
 | 
			
		||||
            if (o.ContainsKey("tag")) patch.Tag = o.Value<string>("tag").NullIfEmpty();
 | 
			
		||||
            if (o.ContainsKey("avatar_url")) patch.AvatarUrl = o.Value<string>("avatar_url").NullIfEmpty();
 | 
			
		||||
            if (o.ContainsKey("banner")) patch.BannerImage = o.Value<string>("banner").NullIfEmpty();
 | 
			
		||||
            if (o.ContainsKey("timezone")) patch.UiTz = o.Value<string>("tz") ?? "UTC";
 | 
			
		||||
 | 
			
		||||
            // legacy: APIv1 uses "tz" instead of "timezone"
 | 
			
		||||
            // todo: remove in APIv2
 | 
			
		||||
            if (o.ContainsKey("tz")) patch.UiTz = o.Value<string>("tz") ?? "UTC";
 | 
			
		||||
            
 | 
			
		||||
            if (o.ContainsKey("description_privacy")) patch.DescriptionPrivacy = o.Value<string>("description_privacy").ParsePrivacy("description");
 | 
			
		||||
            if (o.ContainsKey("member_list_privacy")) patch.MemberListPrivacy = o.Value<string>("member_list_privacy").ParsePrivacy("member list");
 | 
			
		||||
            if (o.ContainsKey("front_privacy")) patch.FrontPrivacy = o.Value<string>("front_privacy").ParsePrivacy("front");
 | 
			
		||||
            if (o.ContainsKey("front_history_privacy")) patch.FrontHistoryPrivacy = o.Value<string>("front_history_privacy").ParsePrivacy("front history");
 | 
			
		||||
            if (o.ContainsKey("description_privacy")) patch.DescriptionPrivacy = o.ParsePrivacy("description_privacy");
 | 
			
		||||
            if (o.ContainsKey("member_list_privacy")) patch.MemberListPrivacy = o.ParsePrivacy("member_list_privacy");
 | 
			
		||||
            if (o.ContainsKey("front_privacy")) patch.FrontPrivacy = o.ParsePrivacy("front_privacy");
 | 
			
		||||
            if (o.ContainsKey("front_history_privacy")) patch.FrontHistoryPrivacy = o.ParsePrivacy("front_history_privacy");
 | 
			
		||||
            return patch;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user