2024-07-12 17:12:24 +02:00
|
|
|
using System.Text.RegularExpressions;
|
2024-08-22 15:13:46 +02:00
|
|
|
using Foxnouns.Backend.Database;
|
|
|
|
using Foxnouns.Backend.Database.Models;
|
2024-07-12 17:12:24 +02:00
|
|
|
|
|
|
|
namespace Foxnouns.Backend.Utils;
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Static methods for validating user input (mostly making sure it's not too short or too long)
|
|
|
|
/// </summary>
|
|
|
|
public static class ValidationUtils
|
|
|
|
{
|
|
|
|
private static readonly Regex UsernameRegex = new("^[\\w-.]{2,40}$", RegexOptions.IgnoreCase);
|
|
|
|
|
2024-07-14 21:25:23 +02:00
|
|
|
private static readonly Regex MemberRegex =
|
|
|
|
new("^[^@\\?!#/\\\\[\\]\"\\{\\}'$%&()+<=>^|~`,\\*]{1,100}$", RegexOptions.IgnoreCase);
|
|
|
|
|
2024-07-12 17:12:24 +02:00
|
|
|
private static readonly string[] InvalidUsernames =
|
|
|
|
[
|
|
|
|
"..",
|
|
|
|
"admin",
|
|
|
|
"administrator",
|
|
|
|
"mod",
|
|
|
|
"moderator",
|
|
|
|
"api",
|
|
|
|
"page",
|
|
|
|
"pronouns",
|
|
|
|
"settings",
|
|
|
|
"pronouns.cc",
|
|
|
|
"pronounscc"
|
|
|
|
];
|
2024-07-14 21:25:23 +02:00
|
|
|
|
|
|
|
private static readonly string[] InvalidMemberNames =
|
|
|
|
[
|
|
|
|
// these break routing outright
|
|
|
|
".",
|
|
|
|
"..",
|
|
|
|
// the user edit page lives at `/@{username}/edit`, so a member named "edit" would be inaccessible
|
|
|
|
"edit"
|
|
|
|
];
|
|
|
|
|
2024-07-14 16:44:41 +02:00
|
|
|
public static ValidationError? ValidateUsername(string username)
|
2024-07-12 17:12:24 +02:00
|
|
|
{
|
|
|
|
if (!UsernameRegex.IsMatch(username))
|
2024-07-14 16:44:41 +02:00
|
|
|
return username.Length switch
|
2024-07-12 17:12:24 +02:00
|
|
|
{
|
2024-07-14 16:44:41 +02:00
|
|
|
< 2 => ValidationError.LengthError("Username is too short", 2, 40, username.Length),
|
|
|
|
> 40 => ValidationError.LengthError("Username is too long", 2, 40, username.Length),
|
|
|
|
_ => ValidationError.GenericValidationError(
|
2024-07-14 21:25:23 +02:00
|
|
|
"Username is invalid, can only contain alphanumeric characters, dashes, underscores, and periods",
|
|
|
|
username)
|
2024-07-12 17:12:24 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
if (InvalidUsernames.Any(u => string.Equals(u, username, StringComparison.InvariantCultureIgnoreCase)))
|
2024-07-14 16:44:41 +02:00
|
|
|
return ValidationError.GenericValidationError("Username is not allowed", username);
|
|
|
|
return null;
|
2024-07-12 17:12:24 +02:00
|
|
|
}
|
|
|
|
|
2024-07-14 21:25:23 +02:00
|
|
|
public static ValidationError? ValidateMemberName(string memberName)
|
|
|
|
{
|
2024-09-03 16:29:51 +02:00
|
|
|
if (!MemberRegex.IsMatch(memberName))
|
2024-07-14 21:25:23 +02:00
|
|
|
return memberName.Length switch
|
|
|
|
{
|
2024-09-03 16:29:51 +02:00
|
|
|
< 1 => ValidationError.LengthError("Name is too short", 1, 100, memberName.Length),
|
|
|
|
> 100 => ValidationError.LengthError("Name is too long", 1, 100, memberName.Length),
|
2024-07-14 21:25:23 +02:00
|
|
|
_ => ValidationError.GenericValidationError(
|
|
|
|
"Member name cannot contain any of the following: " +
|
|
|
|
" @, ?, !, #, /, \\, [, ], \", ', $, %, &, (, ), {, }, +, <, =, >, ^, |, ~, `, , " +
|
|
|
|
"and cannot be one or two periods",
|
|
|
|
memberName)
|
|
|
|
};
|
|
|
|
|
|
|
|
if (InvalidMemberNames.Any(u => string.Equals(u, memberName, StringComparison.InvariantCultureIgnoreCase)))
|
|
|
|
return ValidationError.GenericValidationError("Name is not allowed", memberName);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2024-07-14 16:44:41 +02:00
|
|
|
public static void Validate(IEnumerable<(string, ValidationError?)> errors)
|
2024-07-12 17:12:24 +02:00
|
|
|
{
|
2024-07-14 16:44:41 +02:00
|
|
|
errors = errors.Where(e => e.Item2 != null).ToList();
|
|
|
|
if (!errors.Any()) return;
|
|
|
|
|
|
|
|
var errorDict = new Dictionary<string, IEnumerable<ValidationError>>();
|
|
|
|
foreach (var error in errors)
|
2024-07-12 17:12:24 +02:00
|
|
|
{
|
2024-07-14 16:44:41 +02:00
|
|
|
if (errorDict.TryGetValue(error.Item1, out var value)) errorDict[error.Item1] = value.Append(error.Item2!);
|
|
|
|
errorDict.Add(error.Item1, [error.Item2!]);
|
2024-07-12 17:12:24 +02:00
|
|
|
}
|
2024-07-14 16:44:41 +02:00
|
|
|
|
|
|
|
throw new ApiError.BadRequest("Error validating input", errorDict);
|
2024-07-12 17:12:24 +02:00
|
|
|
}
|
|
|
|
|
2024-07-14 16:44:41 +02:00
|
|
|
public static ValidationError? ValidateDisplayName(string? displayName)
|
2024-07-12 17:12:24 +02:00
|
|
|
{
|
2024-07-14 16:44:41 +02:00
|
|
|
return displayName?.Length switch
|
2024-07-12 17:12:24 +02:00
|
|
|
{
|
2024-07-14 16:44:41 +02:00
|
|
|
0 => ValidationError.LengthError("Display name is too short", 1, 100, displayName.Length),
|
|
|
|
> 100 => ValidationError.LengthError("Display name is too long", 1, 100, displayName.Length),
|
|
|
|
_ => null
|
|
|
|
};
|
2024-07-12 17:12:24 +02:00
|
|
|
}
|
|
|
|
|
2024-09-27 15:29:33 +02:00
|
|
|
private const int MaxLinks = 25;
|
|
|
|
private const int MaxLinkLength = 256;
|
|
|
|
|
|
|
|
public static IEnumerable<(string, ValidationError?)> ValidateLinks(string[]? links)
|
|
|
|
{
|
|
|
|
if (links == null) return [];
|
|
|
|
if (links.Length > MaxLinks)
|
|
|
|
return [("links", ValidationError.LengthError("Too many links", 0, MaxLinks, links.Length))];
|
|
|
|
|
|
|
|
var errors = new List<(string, ValidationError?)>();
|
|
|
|
foreach (var (link, idx) in links.Select((l, i) => (l, i)))
|
|
|
|
{
|
|
|
|
switch (link.Length)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
errors.Add(($"links.{idx}",
|
|
|
|
ValidationError.LengthError("Link cannot be empty", 1, 256, 0)));
|
|
|
|
break;
|
|
|
|
case > MaxLinkLength:
|
|
|
|
errors.Add(($"links.{idx}",
|
|
|
|
ValidationError.LengthError("Link is too long", 1, MaxLinkLength, link.Length)));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return errors;
|
|
|
|
}
|
|
|
|
|
2024-07-14 16:44:41 +02:00
|
|
|
public static ValidationError? ValidateBio(string? bio)
|
2024-07-12 17:12:24 +02:00
|
|
|
{
|
2024-07-14 16:44:41 +02:00
|
|
|
return bio?.Length switch
|
|
|
|
{
|
|
|
|
0 => ValidationError.LengthError("Bio is too short", 1, 1024, bio.Length),
|
|
|
|
> 1024 => ValidationError.LengthError("Bio is too long", 1, 1024, bio.Length),
|
|
|
|
_ => null
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
public static ValidationError? ValidateAvatar(string? avatar)
|
|
|
|
{
|
|
|
|
return avatar?.Length switch
|
|
|
|
{
|
|
|
|
0 => ValidationError.GenericValidationError("Avatar cannot be empty", null),
|
2024-08-22 15:13:46 +02:00
|
|
|
> 1_500_000 => ValidationError.GenericValidationError("Avatar is too large", null),
|
2024-07-14 16:44:41 +02:00
|
|
|
_ => null
|
|
|
|
};
|
2024-07-12 17:12:24 +02:00
|
|
|
}
|
2024-08-22 15:13:46 +02:00
|
|
|
|
2024-09-03 16:29:51 +02:00
|
|
|
|
2024-08-22 15:13:46 +02:00
|
|
|
private static readonly string[] DefaultStatusOptions =
|
|
|
|
[
|
|
|
|
"favourite",
|
|
|
|
"okay",
|
|
|
|
"jokingly",
|
|
|
|
"friends_only",
|
|
|
|
"avoid"
|
|
|
|
];
|
|
|
|
|
|
|
|
public static IEnumerable<(string, ValidationError?)> ValidateFields(List<Field>? fields,
|
|
|
|
IReadOnlyDictionary<Snowflake, User.CustomPreference> customPreferences)
|
|
|
|
{
|
|
|
|
if (fields == null) return [];
|
|
|
|
|
|
|
|
var errors = new List<(string, ValidationError?)>();
|
|
|
|
if (fields.Count > 25)
|
2024-09-03 16:29:51 +02:00
|
|
|
errors.Add(("fields", ValidationError.LengthError("Too many fields", 0, Limits.FieldLimit, fields.Count)));
|
2024-08-22 15:13:46 +02:00
|
|
|
// No overwhelming this function, thank you
|
|
|
|
if (fields.Count > 100) return errors;
|
|
|
|
|
|
|
|
foreach (var (field, index) in fields.Select((field, index) => (field, index)))
|
|
|
|
{
|
|
|
|
switch (field.Name.Length)
|
|
|
|
{
|
2024-09-03 16:29:51 +02:00
|
|
|
case > Limits.FieldNameLimit:
|
2024-08-22 15:13:46 +02:00
|
|
|
errors.Add(($"fields.{index}.name",
|
2024-09-04 14:25:44 +02:00
|
|
|
ValidationError.LengthError("Field name is too long", 1, Limits.FieldNameLimit,
|
|
|
|
field.Name.Length)));
|
2024-08-22 15:13:46 +02:00
|
|
|
break;
|
|
|
|
case < 1:
|
|
|
|
errors.Add(($"fields.{index}.name",
|
2024-09-04 14:25:44 +02:00
|
|
|
ValidationError.LengthError("Field name is too short", 1, Limits.FieldNameLimit,
|
|
|
|
field.Name.Length)));
|
2024-08-22 15:13:46 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2024-09-25 19:48:05 +02:00
|
|
|
errors = errors.Concat(ValidateFieldEntries(field.Entries, customPreferences, $"fields.{index}.entries"))
|
|
|
|
.ToList();
|
2024-08-22 15:13:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return errors;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static IEnumerable<(string, ValidationError?)> ValidateFieldEntries(FieldEntry[]? entries,
|
|
|
|
IReadOnlyDictionary<Snowflake, User.CustomPreference> customPreferences, string errorPrefix = "fields")
|
|
|
|
{
|
|
|
|
if (entries == null || entries.Length == 0) return [];
|
|
|
|
var errors = new List<(string, ValidationError?)>();
|
|
|
|
|
2024-09-03 16:29:51 +02:00
|
|
|
if (entries.Length > Limits.FieldEntriesLimit)
|
2024-09-24 20:56:10 +02:00
|
|
|
errors.Add((errorPrefix,
|
2024-09-03 16:29:51 +02:00
|
|
|
ValidationError.LengthError("Field has too many entries", 0, Limits.FieldEntriesLimit,
|
2024-08-22 15:13:46 +02:00
|
|
|
entries.Length)));
|
|
|
|
|
|
|
|
// Same as above, no overwhelming this function with a ridiculous amount of entries
|
2024-09-03 16:29:51 +02:00
|
|
|
if (entries.Length > Limits.FieldEntriesLimit + 50) return errors;
|
2024-08-22 15:13:46 +02:00
|
|
|
|
|
|
|
foreach (var (entry, entryIdx) in entries.Select((entry, entryIdx) => (entry, entryIdx)))
|
|
|
|
{
|
|
|
|
switch (entry.Value.Length)
|
|
|
|
{
|
2024-09-03 16:29:51 +02:00
|
|
|
case > Limits.FieldEntryTextLimit:
|
2024-09-24 20:56:10 +02:00
|
|
|
errors.Add(($"{errorPrefix}.{entryIdx}.value",
|
2024-09-03 16:29:51 +02:00
|
|
|
ValidationError.LengthError("Field value is too long", 1, Limits.FieldEntryTextLimit,
|
2024-08-22 15:13:46 +02:00
|
|
|
entry.Value.Length)));
|
|
|
|
break;
|
|
|
|
case < 1:
|
2024-09-24 20:56:10 +02:00
|
|
|
errors.Add(($"{errorPrefix}.{entryIdx}.value",
|
2024-09-03 16:29:51 +02:00
|
|
|
ValidationError.LengthError("Field value is too short", 1, Limits.FieldEntryTextLimit,
|
2024-08-22 15:13:46 +02:00
|
|
|
entry.Value.Length)));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
var customPreferenceIds = customPreferences?.Keys.Select(id => id.ToString()) ?? [];
|
|
|
|
|
|
|
|
if (!DefaultStatusOptions.Contains(entry.Status) && !customPreferenceIds.Contains(entry.Status))
|
2024-09-24 20:56:10 +02:00
|
|
|
errors.Add(($"{errorPrefix}.{entryIdx}.status",
|
|
|
|
ValidationError.GenericValidationError("Invalid status", entry.Status)));
|
|
|
|
}
|
|
|
|
|
|
|
|
return errors;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static IEnumerable<(string, ValidationError?)> ValidatePronouns(Pronoun[]? entries,
|
|
|
|
IReadOnlyDictionary<Snowflake, User.CustomPreference> customPreferences, string errorPrefix = "pronouns")
|
|
|
|
{
|
|
|
|
if (entries == null || entries.Length == 0) return [];
|
|
|
|
var errors = new List<(string, ValidationError?)>();
|
|
|
|
|
|
|
|
if (entries.Length > Limits.FieldEntriesLimit)
|
|
|
|
errors.Add((errorPrefix,
|
|
|
|
ValidationError.LengthError("Too many pronouns", 0, Limits.FieldEntriesLimit,
|
|
|
|
entries.Length)));
|
|
|
|
|
|
|
|
// Same as above, no overwhelming this function with a ridiculous amount of entries
|
|
|
|
if (entries.Length > Limits.FieldEntriesLimit + 50) return errors;
|
|
|
|
|
|
|
|
foreach (var (entry, entryIdx) in entries.Select((entry, entryIdx) => (entry, entryIdx)))
|
|
|
|
{
|
|
|
|
switch (entry.Value.Length)
|
|
|
|
{
|
|
|
|
case > Limits.FieldEntryTextLimit:
|
|
|
|
errors.Add(($"{errorPrefix}.{entryIdx}.value",
|
|
|
|
ValidationError.LengthError("Pronoun value is too long", 1, Limits.FieldEntryTextLimit,
|
|
|
|
entry.Value.Length)));
|
|
|
|
break;
|
|
|
|
case < 1:
|
|
|
|
errors.Add(($"{errorPrefix}.{entryIdx}.value",
|
|
|
|
ValidationError.LengthError("Pronoun value is too short", 1, Limits.FieldEntryTextLimit,
|
|
|
|
entry.Value.Length)));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (entry.DisplayText != null)
|
|
|
|
{
|
|
|
|
switch (entry.DisplayText.Length)
|
|
|
|
{
|
|
|
|
case > Limits.FieldEntryTextLimit:
|
|
|
|
errors.Add(($"{errorPrefix}.{entryIdx}.value",
|
2024-09-25 19:48:05 +02:00
|
|
|
ValidationError.LengthError("Pronoun display text is too long", 1,
|
|
|
|
Limits.FieldEntryTextLimit,
|
2024-09-24 20:56:10 +02:00
|
|
|
entry.Value.Length)));
|
|
|
|
break;
|
|
|
|
case < 1:
|
|
|
|
errors.Add(($"{errorPrefix}.{entryIdx}.value",
|
2024-09-25 19:48:05 +02:00
|
|
|
ValidationError.LengthError("Pronoun display text is too short", 1,
|
|
|
|
Limits.FieldEntryTextLimit,
|
2024-09-24 20:56:10 +02:00
|
|
|
entry.Value.Length)));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var customPreferenceIds = customPreferences?.Keys.Select(id => id.ToString()) ?? [];
|
|
|
|
|
|
|
|
if (!DefaultStatusOptions.Contains(entry.Status) && !customPreferenceIds.Contains(entry.Status))
|
|
|
|
errors.Add(($"{errorPrefix}.{entryIdx}.status",
|
2024-08-22 15:13:46 +02:00
|
|
|
ValidationError.GenericValidationError("Invalid status", entry.Status)));
|
|
|
|
}
|
|
|
|
|
|
|
|
return errors;
|
|
|
|
}
|
2024-07-12 17:12:24 +02:00
|
|
|
}
|