feat(backend): make field limits configurable

This commit is contained in:
sam 2025-02-28 16:37:15 +01:00
parent 7ea6c62d67
commit 218c756a70
Signed by: sam
GPG key ID: B4EF20DDE721CAA1
7 changed files with 133 additions and 149 deletions

View file

@ -99,6 +99,11 @@ public class Config
{ {
public int MaxMemberCount { get; init; } = 1000; public int MaxMemberCount { get; init; } = 1000;
public int MaxFields { get; init; } = 25;
public int MaxFieldNameLength { get; init; } = 100;
public int MaxFieldEntryTextLength { get; init; } = 100;
public int MaxFieldEntries { get; init; } = 100;
public int MaxUsernameLength { get; init; } = 40; public int MaxUsernameLength { get; init; } = 40;
public int MaxMemberNameLength { get; init; } = 100; public int MaxMemberNameLength { get; init; } = 100;
public int MaxDisplayNameLength { get; init; } = 100; public int MaxDisplayNameLength { get; init; } = 100;

View file

@ -81,13 +81,13 @@ public class MembersController(
("display_name", validationService.ValidateDisplayName(req.DisplayName)), ("display_name", validationService.ValidateDisplayName(req.DisplayName)),
("bio", validationService.ValidateBio(req.Bio)), ("bio", validationService.ValidateBio(req.Bio)),
("avatar", validationService.ValidateAvatar(req.Avatar)), ("avatar", validationService.ValidateAvatar(req.Avatar)),
.. ValidationUtils.ValidateFields(req.Fields, CurrentUser!.CustomPreferences), .. validationService.ValidateFields(req.Fields, CurrentUser!.CustomPreferences),
.. ValidationUtils.ValidateFieldEntries( .. validationService.ValidateFieldEntries(
req.Names?.ToArray(), req.Names?.ToArray(),
CurrentUser!.CustomPreferences, CurrentUser!.CustomPreferences,
"names" "names"
), ),
.. ValidationUtils.ValidatePronouns( .. validationService.ValidatePronouns(
req.Pronouns?.ToArray(), req.Pronouns?.ToArray(),
CurrentUser!.CustomPreferences CurrentUser!.CustomPreferences
), ),
@ -191,7 +191,7 @@ public class MembersController(
if (req.Names != null) if (req.Names != null)
{ {
errors.AddRange( errors.AddRange(
ValidationUtils.ValidateFieldEntries( validationService.ValidateFieldEntries(
req.Names, req.Names,
CurrentUser!.CustomPreferences, CurrentUser!.CustomPreferences,
"names" "names"
@ -203,7 +203,7 @@ public class MembersController(
if (req.Pronouns != null) if (req.Pronouns != null)
{ {
errors.AddRange( errors.AddRange(
ValidationUtils.ValidatePronouns(req.Pronouns, CurrentUser!.CustomPreferences) validationService.ValidatePronouns(req.Pronouns, CurrentUser!.CustomPreferences)
); );
member.Pronouns = req.Pronouns.ToList(); member.Pronouns = req.Pronouns.ToList();
} }
@ -211,7 +211,10 @@ public class MembersController(
if (req.Fields != null) if (req.Fields != null)
{ {
errors.AddRange( errors.AddRange(
ValidationUtils.ValidateFields(req.Fields.ToList(), CurrentUser!.CustomPreferences) validationService.ValidateFields(
req.Fields.ToList(),
CurrentUser!.CustomPreferences
)
); );
member.Fields = req.Fields.ToList(); member.Fields = req.Fields.ToList();
} }

View file

@ -91,7 +91,7 @@ public class UsersController(
if (req.Names != null) if (req.Names != null)
{ {
errors.AddRange( errors.AddRange(
ValidationUtils.ValidateFieldEntries( validationService.ValidateFieldEntries(
req.Names, req.Names,
CurrentUser!.CustomPreferences, CurrentUser!.CustomPreferences,
"names" "names"
@ -103,7 +103,7 @@ public class UsersController(
if (req.Pronouns != null) if (req.Pronouns != null)
{ {
errors.AddRange( errors.AddRange(
ValidationUtils.ValidatePronouns(req.Pronouns, CurrentUser!.CustomPreferences) validationService.ValidatePronouns(req.Pronouns, CurrentUser!.CustomPreferences)
); );
user.Pronouns = req.Pronouns.ToList(); user.Pronouns = req.Pronouns.ToList();
} }
@ -111,7 +111,10 @@ public class UsersController(
if (req.Fields != null) if (req.Fields != null)
{ {
errors.AddRange( errors.AddRange(
ValidationUtils.ValidateFields(req.Fields.ToList(), CurrentUser!.CustomPreferences) validationService.ValidateFields(
req.Fields.ToList(),
CurrentUser!.CustomPreferences
)
); );
user.Fields = req.Fields.ToList(); user.Fields = req.Fields.ToList();
} }

View file

@ -15,9 +15,9 @@
using Foxnouns.Backend.Database; using Foxnouns.Backend.Database;
using Foxnouns.Backend.Database.Models; using Foxnouns.Backend.Database.Models;
namespace Foxnouns.Backend.Utils; namespace Foxnouns.Backend.Services;
public static partial class ValidationUtils public partial class ValidationService
{ {
public static readonly string[] DefaultStatusOptions = public static readonly string[] DefaultStatusOptions =
[ [
@ -28,7 +28,7 @@ public static partial class ValidationUtils
"avoid", "avoid",
]; ];
public static IEnumerable<(string, ValidationError?)> ValidateFields( public IEnumerable<(string, ValidationError?)> ValidateFields(
List<Field>? fields, List<Field>? fields,
IReadOnlyDictionary<Snowflake, User.CustomPreference> customPreferences IReadOnlyDictionary<Snowflake, User.CustomPreference> customPreferences
) )
@ -37,7 +37,7 @@ public static partial class ValidationUtils
return []; return [];
var errors = new List<(string, ValidationError?)>(); var errors = new List<(string, ValidationError?)>();
if (fields.Count > 25) if (fields.Count > _limits.MaxFields)
{ {
errors.Add( errors.Add(
( (
@ -45,7 +45,7 @@ public static partial class ValidationUtils
ValidationError.LengthError( ValidationError.LengthError(
"Too many fields", "Too many fields",
0, 0,
Limits.FieldLimit, _limits.MaxFields,
fields.Count fields.Count
) )
) )
@ -53,39 +53,38 @@ public static partial class ValidationUtils
} }
// No overwhelming this function, thank you // No overwhelming this function, thank you
if (fields.Count > 100) if (fields.Count > _limits.MaxFields + 50)
return errors; return errors;
foreach ((Field? field, int index) in fields.Select((field, index) => (field, index))) foreach ((Field? field, int index) in fields.Select((field, index) => (field, index)))
{ {
switch (field.Name.Length) if (field.Name.Length > _limits.MaxFieldNameLength)
{ {
case > Limits.FieldNameLimit: errors.Add(
errors.Add( (
( $"fields.{index}.name",
$"fields.{index}.name", ValidationError.LengthError(
ValidationError.LengthError( "Field name is too long",
"Field name is too long", 1,
1, _limits.MaxFieldNameLength,
Limits.FieldNameLimit, field.Name.Length
field.Name.Length
)
) )
); )
break; );
case < 1: }
errors.Add( else if (field.Name.Length < 1)
( {
$"fields.{index}.name", errors.Add(
ValidationError.LengthError( (
"Field name is too short", $"fields.{index}.name",
1, ValidationError.LengthError(
Limits.FieldNameLimit, "Field name is too short",
field.Name.Length 1,
) _limits.MaxFieldNameLength,
field.Name.Length
) )
); )
break; );
} }
errors = errors errors = errors
@ -102,7 +101,7 @@ public static partial class ValidationUtils
return errors; return errors;
} }
public static IEnumerable<(string, ValidationError?)> ValidateFieldEntries( public IEnumerable<(string, ValidationError?)> ValidateFieldEntries(
FieldEntry[]? entries, FieldEntry[]? entries,
IReadOnlyDictionary<Snowflake, User.CustomPreference> customPreferences, IReadOnlyDictionary<Snowflake, User.CustomPreference> customPreferences,
string errorPrefix = "fields" string errorPrefix = "fields"
@ -112,7 +111,7 @@ public static partial class ValidationUtils
return []; return [];
var errors = new List<(string, ValidationError?)>(); var errors = new List<(string, ValidationError?)>();
if (entries.Length > Limits.FieldEntriesLimit) if (entries.Length > _limits.MaxFieldEntries)
{ {
errors.Add( errors.Add(
( (
@ -120,7 +119,7 @@ public static partial class ValidationUtils
ValidationError.LengthError( ValidationError.LengthError(
"Field has too many entries", "Field has too many entries",
0, 0,
Limits.FieldEntriesLimit, _limits.MaxFieldEntries,
entries.Length entries.Length
) )
) )
@ -128,7 +127,7 @@ public static partial class ValidationUtils
} }
// Same as above, no overwhelming this function with a ridiculous amount of entries // Same as above, no overwhelming this function with a ridiculous amount of entries
if (entries.Length > Limits.FieldEntriesLimit + 50) if (entries.Length > _limits.MaxFieldEntries + 50)
return errors; return errors;
string[] customPreferenceIds = customPreferences.Keys.Select(id => id.ToString()).ToArray(); string[] customPreferenceIds = customPreferences.Keys.Select(id => id.ToString()).ToArray();
@ -139,34 +138,33 @@ public static partial class ValidationUtils
) )
) )
{ {
switch (entry.Value.Length) if (entry.Value.Length > _limits.MaxFieldEntryTextLength)
{ {
case > Limits.FieldEntryTextLimit: errors.Add(
errors.Add( (
( $"{errorPrefix}.{entryIdx}.value",
$"{errorPrefix}.{entryIdx}.value", ValidationError.LengthError(
ValidationError.LengthError( "Field value is too long",
"Field value is too long", 1,
1, _limits.MaxFieldEntryTextLength,
Limits.FieldEntryTextLimit, entry.Value.Length
entry.Value.Length
)
) )
); )
break; );
case < 1: }
errors.Add( else if (entry.Value.Length < 1)
( {
$"{errorPrefix}.{entryIdx}.value", errors.Add(
ValidationError.LengthError( (
"Field value is too short", $"{errorPrefix}.{entryIdx}.value",
1, ValidationError.LengthError(
Limits.FieldEntryTextLimit, "Field value is too short",
entry.Value.Length 1,
) _limits.MaxFieldEntryTextLength,
entry.Value.Length
) )
); )
break; );
} }
if ( if (
@ -186,7 +184,7 @@ public static partial class ValidationUtils
return errors; return errors;
} }
public static IEnumerable<(string, ValidationError?)> ValidatePronouns( public IEnumerable<(string, ValidationError?)> ValidatePronouns(
Pronoun[]? entries, Pronoun[]? entries,
IReadOnlyDictionary<Snowflake, User.CustomPreference> customPreferences, IReadOnlyDictionary<Snowflake, User.CustomPreference> customPreferences,
string errorPrefix = "pronouns" string errorPrefix = "pronouns"
@ -196,7 +194,7 @@ public static partial class ValidationUtils
return []; return [];
var errors = new List<(string, ValidationError?)>(); var errors = new List<(string, ValidationError?)>();
if (entries.Length > Limits.FieldEntriesLimit) if (entries.Length > _limits.MaxFieldEntries)
{ {
errors.Add( errors.Add(
( (
@ -204,7 +202,7 @@ public static partial class ValidationUtils
ValidationError.LengthError( ValidationError.LengthError(
"Too many pronouns", "Too many pronouns",
0, 0,
Limits.FieldEntriesLimit, _limits.MaxFieldEntries,
entries.Length entries.Length
) )
) )
@ -212,7 +210,7 @@ public static partial class ValidationUtils
} }
// Same as above, no overwhelming this function with a ridiculous amount of entries // Same as above, no overwhelming this function with a ridiculous amount of entries
if (entries.Length > Limits.FieldEntriesLimit + 50) if (entries.Length > _limits.MaxFieldEntries + 50)
return errors; return errors;
string[] customPreferenceIds = customPreferences.Keys.Select(id => id.ToString()).ToArray(); string[] customPreferenceIds = customPreferences.Keys.Select(id => id.ToString()).ToArray();
@ -221,66 +219,64 @@ public static partial class ValidationUtils
(Pronoun? entry, int entryIdx) in entries.Select((entry, entryIdx) => (entry, entryIdx)) (Pronoun? entry, int entryIdx) in entries.Select((entry, entryIdx) => (entry, entryIdx))
) )
{ {
switch (entry.Value.Length) if (entry.Value.Length > _limits.MaxFieldEntryTextLength)
{ {
case > Limits.FieldEntryTextLimit: errors.Add(
errors.Add( (
( $"{errorPrefix}.{entryIdx}.value",
$"{errorPrefix}.{entryIdx}.value", ValidationError.LengthError(
ValidationError.LengthError( "Pronoun value is too long",
"Pronoun value is too long", 1,
1, _limits.MaxFieldEntryTextLength,
Limits.FieldEntryTextLimit, entry.Value.Length
entry.Value.Length
)
) )
); )
break; );
case < 1: }
errors.Add( else if (entry.Value.Length < 1)
( {
$"{errorPrefix}.{entryIdx}.value", errors.Add(
ValidationError.LengthError( (
"Pronoun value is too short", $"{errorPrefix}.{entryIdx}.value",
1, ValidationError.LengthError(
Limits.FieldEntryTextLimit, "Pronoun value is too short",
entry.Value.Length 1,
) _limits.MaxFieldEntryTextLength,
entry.Value.Length
) )
); )
break; );
} }
if (entry.DisplayText != null) if (entry.DisplayText != null)
{ {
switch (entry.DisplayText.Length) if (entry.DisplayText.Length > _limits.MaxFieldEntryTextLength)
{ {
case > Limits.FieldEntryTextLimit: errors.Add(
errors.Add( (
( $"{errorPrefix}.{entryIdx}.display_text",
$"{errorPrefix}.{entryIdx}.display_text", ValidationError.LengthError(
ValidationError.LengthError( "Pronoun display text is too long",
"Pronoun display text is too long", 1,
1, _limits.MaxFieldEntryTextLength,
Limits.FieldEntryTextLimit, entry.Value.Length
entry.Value.Length
)
) )
); )
break; );
case < 1: }
errors.Add( else if (entry.DisplayText.Length < 1)
( {
$"{errorPrefix}.{entryIdx}.display_text", errors.Add(
ValidationError.LengthError( (
"Pronoun display text is too short", $"{errorPrefix}.{entryIdx}.display_text",
1, ValidationError.LengthError(
Limits.FieldEntryTextLimit, "Pronoun display text is too short",
entry.Value.Length 1,
) _limits.MaxFieldEntryTextLength,
entry.Value.Length
) )
); )
break; );
} }
} }

View file

@ -1,23 +0,0 @@
// Copyright (C) 2023-present sam/u1f320 (vulpine.solutions)
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published
// by the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
namespace Foxnouns.Backend.Utils;
public static class Limits
{
public const int FieldLimit = 25;
public const int FieldNameLimit = 100;
public const int FieldEntryTextLimit = 100;
public const int FieldEntriesLimit = 100;
}

View file

@ -20,7 +20,7 @@ public static partial class ValidationUtils
public static ValidationError? ValidateReportContext(string? context) => public static ValidationError? ValidateReportContext(string? context) =>
context?.Length > MaximumReportContextLength context?.Length > MaximumReportContextLength
? ValidationError.GenericValidationError("Avatar is too large", null) ? ValidationError.GenericValidationError("Report context is too long", null)
: null; : null;
public const int MinimumPasswordLength = 12; public const int MinimumPasswordLength = 12;

View file

@ -2,7 +2,7 @@ using System.Diagnostics.CodeAnalysis;
using Dapper; using Dapper;
using Foxnouns.Backend.Database; using Foxnouns.Backend.Database;
using Foxnouns.Backend.Database.Models; using Foxnouns.Backend.Database.Models;
using Foxnouns.Backend.Utils; using Foxnouns.Backend.Services;
using Foxnouns.DataMigrator.Models; using Foxnouns.DataMigrator.Models;
using NodaTime.Extensions; using NodaTime.Extensions;
using Npgsql; using Npgsql;
@ -260,6 +260,6 @@ public class UserMigrator(
{ {
if (_preferenceIds.TryGetValue(id, out Snowflake preferenceId)) if (_preferenceIds.TryGetValue(id, out Snowflake preferenceId))
return preferenceId.ToString(); return preferenceId.ToString();
return ValidationUtils.DefaultStatusOptions.Contains(id) ? id : "okay"; return ValidationService.DefaultStatusOptions.Contains(id) ? id : "okay";
} }
} }