From 3345df918dba7360cbba9c31805c57f663288618 Mon Sep 17 00:00:00 2001 From: Tom Jones Date: Fri, 13 Dec 2024 12:54:22 +0000 Subject: [PATCH] EES-5047: Move string length and message definitions into constants, update naming convention for "DataSet" in validation messaging. --- .../DataArchiveValidationServiceTests.cs | 12 ++--- .../Controllers/Api/ReleasesController.cs | 5 +- .../Requests/DataBlockRequests.cs | 13 ++--- .../Requests/DataGuidanceUpdateRequest.cs | 36 +++++++++++--- .../Requests/FeaturedTableRequests.cs | 42 ++++++++++++---- .../Requests/ReleaseFileRequests.cs | 41 +++++++++++----- .../Services/DataArchiveValidationService.cs | 17 +++---- .../Validators/ValidationMessages.cs | 48 +++++++++---------- .../Constants/ValidationConstants.cs | 25 ++++++++++ .../release/data/ReleaseDataFilePage.tsx | 9 +++- .../data/components/AncillaryFileForm.tsx | 16 +++++-- .../data/components/DataFileUploadForm.tsx | 26 +++++----- .../components/ReleaseDataGuidanceSection.tsx | 7 +-- .../components/DataBlockDetailsForm.tsx | 22 ++++++--- 14 files changed, 218 insertions(+), 101 deletions(-) create mode 100644 src/GovUk.Education.ExploreEducationStatistics.Common/Constants/ValidationConstants.cs diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin.Tests/Services/DataArchiveValidationServiceTests.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin.Tests/Services/DataArchiveValidationServiceTests.cs index 27fca5a4bca..11c6f987170 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin.Tests/Services/DataArchiveValidationServiceTests.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin.Tests/Services/DataArchiveValidationServiceTests.cs @@ -238,8 +238,8 @@ public async Task ValidateBulkDataArchiveFiles_IndexFileMissing_ReturnsValidatio .AssertBadRequestWithValidationErrors([ new ErrorViewModel { - Code = ValidationMessages.BulkDataZipMustContainDatasetNamesCsv.Code, - Message = ValidationMessages.BulkDataZipMustContainDatasetNamesCsv.Message, + Code = ValidationMessages.BulkDataZipMustContainDataSetNamesCsv.Code, + Message = ValidationMessages.BulkDataZipMustContainDataSetNamesCsv.Message, } ]); } @@ -267,8 +267,8 @@ public async Task ValidateBulkDataArchiveFiles_IndexFileHasIncorrectHeaders_Retu .AssertBadRequestWithValidationErrors([ new ErrorViewModel { - Code = ValidationMessages.DatasetNamesCsvIncorrectHeaders.Code, - Message = ValidationMessages.DatasetNamesCsvIncorrectHeaders.Message, + Code = ValidationMessages.DataSetNamesCsvIncorrectHeaders.Code, + Message = ValidationMessages.DataSetNamesCsvIncorrectHeaders.Message, }, ]); } @@ -319,7 +319,7 @@ public async Task ValidateBulkDataArchiveFiles_DuplicateDataSetTitlesAndFileName .AssertLeft() .AssertBadRequestWithValidationErrors([ ValidationMessages.GenerateErrorDataSetTitleShouldBeUnique("Duplicate title"), - ValidationMessages.GenerateErrorDatasetNamesCsvFilenamesShouldBeUnique("one"), + ValidationMessages.GenerateErrorDataSetNamesCsvFilenamesShouldBeUnique("one"), ]); } } @@ -342,7 +342,7 @@ public async Task ValidateBulkDataArchiveFiles_Fail_DataSetNamesCsvFilesnamesSho result .AssertLeft() .AssertBadRequestWithValidationErrors([ - ValidationMessages.GenerateErrorDatasetNamesCsvFilenamesShouldNotEndDotCsv("one.csv") + ValidationMessages.GenerateErrorDataSetNamesCsvFilenamesShouldNotEndDotCsv("one.csv") ]); } } diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Controllers/Api/ReleasesController.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Controllers/Api/ReleasesController.cs index 75d34de36fd..7f1003b4f63 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Controllers/Api/ReleasesController.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Controllers/Api/ReleasesController.cs @@ -14,6 +14,7 @@ using System.ComponentModel.DataAnnotations; using System.Threading; using System.Threading.Tasks; +using static GovUk.Education.ExploreEducationStatistics.Common.Constants.ValidationConstants; namespace GovUk.Education.ExploreEducationStatistics.Admin.Controllers.Api { @@ -108,7 +109,7 @@ public async Task>> ReorderDataFiles(Guid releas public async Task> UploadDataSet(Guid releaseVersionId, [FromQuery(Name = "replacingFileId")] Guid? replacingFileId, [FromQuery(Name = "title")] - [MaxLength(120, ErrorMessage = "Subject title must be 120 characters or less")] + [MaxLength(SubjectTitleMaxLength, ErrorMessage = SubjectTitleMaxLengthMessage)] string title, IFormFile file, IFormFile metaFile) @@ -128,7 +129,7 @@ public async Task> UploadDataSet(Guid releaseVersionI public async Task> UploadDataSetAsZip(Guid releaseVersionId, [FromQuery(Name = "replacingFileId")] Guid? replacingFileId, [FromQuery(Name = "title")] - [MaxLength(120, ErrorMessage = "Subject title must be 120 characters or less")] + [MaxLength(SubjectTitleMaxLength, ErrorMessage = SubjectTitleMaxLengthMessage)] string title, IFormFile zipFile) { diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataBlockRequests.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataBlockRequests.cs index 88f1809d9df..6d2fe6767ce 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataBlockRequests.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataBlockRequests.cs @@ -4,6 +4,7 @@ using GovUk.Education.ExploreEducationStatistics.Common.Model.Data; using GovUk.Education.ExploreEducationStatistics.Common.Requests; using System.Collections.Generic; +using static GovUk.Education.ExploreEducationStatistics.Common.Constants.ValidationConstants; namespace GovUk.Education.ExploreEducationStatistics.Admin.Requests; @@ -17,7 +18,7 @@ public record DataBlockCreateRequest public FullTableQueryRequest Query { get; init; } = null!; - public List Charts { get; init; } = new(); + public List Charts { get; init; } = []; public TableBuilderConfiguration Table { get; init; } = null!; @@ -27,8 +28,8 @@ public Validator() { RuleFor(request => request.Heading) .NotEmpty() - .MaximumLength(120) - .WithMessage("Table title must be 120 characters or less"); + .MaximumLength(TableTitleMaxLength) + .WithMessage(TableTitleMaxLengthMessage); RuleFor(request => request.Name) .NotEmpty(); @@ -49,7 +50,7 @@ public record DataBlockUpdateRequest public FullTableQueryRequest Query { get; init; } = null!; - public List Charts { get; init; } = new(); + public List Charts { get; init; } = []; public TableBuilderConfiguration Table { get; init; } = null!; @@ -59,8 +60,8 @@ public Validator() { RuleFor(request => request.Heading) .NotEmpty() - .MaximumLength(120) - .WithMessage("Table title must be 120 characters or less"); + .MaximumLength(TableTitleMaxLength) + .WithMessage(TableTitleMaxLengthMessage); RuleFor(request => request.Name) .NotEmpty(); diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataGuidanceUpdateRequest.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataGuidanceUpdateRequest.cs index d8bf0dd14b7..b057848d576 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataGuidanceUpdateRequest.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataGuidanceUpdateRequest.cs @@ -1,25 +1,47 @@ #nullable enable +using FluentValidation; using System; using System.Collections.Generic; -using System.ComponentModel.DataAnnotations; +using static GovUk.Education.ExploreEducationStatistics.Common.Constants.ValidationConstants; namespace GovUk.Education.ExploreEducationStatistics.Admin.Requests; public record DataGuidanceUpdateRequest { - [Required] public string Content { get; init; } = string.Empty; - [MinLength(1)] - public List DataSets { get; init; } = new(); + public List DataSets { get; init; } = []; + + public class Validator : AbstractValidator + { + public Validator() + { + RuleFor(request => request.Content) + .NotEmpty(); + + RuleFor(request => request.DataSets) + .NotEmpty(); + } + } } public record DataGuidanceDataSetUpdateRequest { - [Required] public Guid FileId { get; init; } - [Required] - [MaxLength(250, ErrorMessage = "File guidance content must be 250 characters or less")] public string Content { get; init; } = string.Empty; + + public class Validator : AbstractValidator + { + public Validator() + { + RuleFor(request => request.FileId) + .NotEmpty(); + + RuleFor(request => request.Content) + .NotEmpty() + .MaximumLength(FileGuidanceContentMaxLength) + .WithMessage(FileGuidanceContentMaxLengthMessage); + } + } } diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/FeaturedTableRequests.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/FeaturedTableRequests.cs index ae9430d871e..fe6ead1a189 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/FeaturedTableRequests.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/FeaturedTableRequests.cs @@ -1,30 +1,54 @@ #nullable enable +using FluentValidation; using System; -using System.ComponentModel.DataAnnotations; +using static GovUk.Education.ExploreEducationStatistics.Common.Constants.ValidationConstants; namespace GovUk.Education.ExploreEducationStatistics.Admin.Requests; public record FeaturedTableCreateRequest { - [Required] - [MaxLength(120, ErrorMessage = "Featured table name must be 120 characters or less")] public string Name { get; init; } = string.Empty; - [Required] - [MaxLength(200, ErrorMessage = "Featured table description must be 200 characters or less")] public string Description { get; set; } = string.Empty; public Guid DataBlockId { get; set; } + public class Validator : AbstractValidator + { + public Validator() + { + RuleFor(request => request.Name) + .NotEmpty() + .MaximumLength(FeaturedTableNameMaxLength) + .WithMessage(FeaturedTableNameMaxLengthMessage); + + RuleFor(request => request.Description) + .NotEmpty() + .MaximumLength(FeaturedTableDescriptionMaxLength) + .WithMessage(FeaturedTableDescriptionMaxLengthMessage); + } + } } public record FeaturedTableUpdateRequest { - [Required] - [MaxLength(120, ErrorMessage = "Featured table name must be 120 characters or less")] public string Name { get; init; } = string.Empty; - [Required] - [MaxLength(200, ErrorMessage = "Featured table description must be 200 characters or less")] public string Description { get; set; } = string.Empty; + + public class Validator : AbstractValidator + { + public Validator() + { + RuleFor(request => request.Name) + .NotEmpty() + .MaximumLength(FeaturedTableNameMaxLength) + .WithMessage(FeaturedTableNameMaxLengthMessage); + + RuleFor(request => request.Description) + .NotEmpty() + .MaximumLength(FeaturedTableDescriptionMaxLength) + .WithMessage(FeaturedTableDescriptionMaxLengthMessage); + } + } } diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/ReleaseFileRequests.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/ReleaseFileRequests.cs index b07f72f1f95..f9bf20a2a86 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/ReleaseFileRequests.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/ReleaseFileRequests.cs @@ -1,7 +1,7 @@ #nullable enable using FluentValidation; using Microsoft.AspNetCore.Http; -using System.ComponentModel.DataAnnotations; +using static GovUk.Education.ExploreEducationStatistics.Common.Constants.ValidationConstants; namespace GovUk.Education.ExploreEducationStatistics.Admin.Requests; @@ -16,20 +16,18 @@ public class Validator : AbstractValidator public Validator() { RuleFor(request => request.Title) - .MaximumLength(120) - .WithMessage("Subject title must be 120 characters or less"); + .MaximumLength(SubjectTitleMaxLength) + .WithMessage(SubjectTitleMaxLengthMessage); } } } public record ReleaseAncillaryFileUploadRequest { - [Required] public string Title { get; set; } = string.Empty; - [Required] public string Summary { get; set; } = string.Empty; + public string Summary { get; set; } = string.Empty; - [Required] public IFormFile File { get; set; } = null!; public class Validator : AbstractValidator @@ -37,23 +35,42 @@ public class Validator : AbstractValidator public Validator() { RuleFor(request => request.Title) - .MaximumLength(120) - .WithMessage("Title must be 120 characters or less"); + .NotEmpty() + .MaximumLength(TitleMaxLength) + .WithMessage(TitleMaxLengthMessage); RuleFor(request => request.Summary) - .MaximumLength(250) - .WithMessage("Summary must be 250 characters or less"); + .NotEmpty() + .MaximumLength(SummaryMaxLength) + .WithMessage(SummaryMaxLengthMessage); + + RuleFor(request => request.File) + .NotEmpty(); } } } public record ReleaseAncillaryFileUpdateRequest { - [Required] public string Title { get; set; } = string.Empty; - [Required] public string Summary { get; set; } = string.Empty; public IFormFile? File { get; set; } + + public class Validator : AbstractValidator + { + public Validator() + { + RuleFor(request => request.Title) + .NotEmpty() + .MaximumLength(TitleMaxLength) + .WithMessage(TitleMaxLengthMessage); + + RuleFor(request => request.Summary) + .NotEmpty() + .MaximumLength(SummaryMaxLength) + .WithMessage(SummaryMaxLengthMessage); + } + } } diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Services/DataArchiveValidationService.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Services/DataArchiveValidationService.cs index cd476108e49..2731cf1b6ce 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Services/DataArchiveValidationService.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Services/DataArchiveValidationService.cs @@ -15,6 +15,7 @@ using System.IO.Compression; using System.Linq; using System.Threading.Tasks; +using static GovUk.Education.ExploreEducationStatistics.Common.Constants.ValidationConstants; namespace GovUk.Education.ExploreEducationStatistics.Admin.Services { @@ -103,8 +104,8 @@ public async Task>> ValidateBulkDa { return Common.Validators.ValidationUtils.ValidationResult(new ErrorViewModel { - Code = ValidationMessages.BulkDataZipMustContainDatasetNamesCsv.Code, - Message = ValidationMessages.BulkDataZipMustContainDatasetNamesCsv.Message, + Code = ValidationMessages.BulkDataZipMustContainDataSetNamesCsv.Code, + Message = ValidationMessages.BulkDataZipMustContainDataSetNamesCsv.Message, }); } @@ -120,8 +121,8 @@ public async Task>> ValidateBulkDa { return Common.Validators.ValidationUtils.ValidationResult(new ErrorViewModel { - Code = ValidationMessages.DatasetNamesCsvIncorrectHeaders.Code, - Message = ValidationMessages.DatasetNamesCsvIncorrectHeaders.Message, + Code = ValidationMessages.DataSetNamesCsvIncorrectHeaders.Code, + Message = ValidationMessages.DataSetNamesCsvIncorrectHeaders.Message, }); } @@ -142,9 +143,9 @@ public async Task>> ValidateBulkDa var filename = row[fileNameIndex]; var datasetName = row[datasetNameIndex].Trim(); - if (datasetName.Length > 120) + if (datasetName.Length > SubjectTitleMaxLength) { - errors.Add(ValidationMessages.GenerateErrorDatasetTitleTooLong(datasetName)); + errors.Add(ValidationMessages.GenerateErrorDataSetTitleTooLong(datasetName)); } dataSetNamesCsvEntries.Add((BaseFilename: filename, Title: datasetName)); @@ -156,7 +157,7 @@ public async Task>> ValidateBulkDa .ToList() .ForEach(baseFilename => { - errors.Add(ValidationMessages.GenerateErrorDatasetNamesCsvFilenamesShouldNotEndDotCsv(baseFilename)); + errors.Add(ValidationMessages.GenerateErrorDataSetNamesCsvFilenamesShouldNotEndDotCsv(baseFilename)); }); // Check for duplicate data set titles - because the bulk zip itself may contain duplicates! @@ -180,7 +181,7 @@ public async Task>> ValidateBulkDa .ForEach(duplicateFilename => { errors.Add(ValidationMessages - .GenerateErrorDatasetNamesCsvFilenamesShouldBeUnique(duplicateFilename)); + .GenerateErrorDataSetNamesCsvFilenamesShouldBeUnique(duplicateFilename)); }); if (errors.Count > 0) diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Validators/ValidationMessages.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Validators/ValidationMessages.cs index aedf5d4d042..9354e84ff7a 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Validators/ValidationMessages.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Validators/ValidationMessages.cs @@ -106,27 +106,27 @@ public static ErrorViewModel GenerateErrorMustBeCsvFile(string fullFilename) }; } - public static readonly LocalizableMessage BulkDataZipMustContainDatasetNamesCsv = new( - Code: nameof(BulkDataZipMustContainDatasetNamesCsv), + public static readonly LocalizableMessage BulkDataZipMustContainDataSetNamesCsv = new( + Code: nameof(BulkDataZipMustContainDataSetNamesCsv), Message: "For bulk imports, the ZIP must include dataset_names.csv" ); - public static readonly LocalizableMessage DatasetNamesCsvReaderException = new( - Code: nameof(DatasetNamesCsvReaderException), + public static readonly LocalizableMessage DataSetNamesCsvReaderException = new( + Code: nameof(DataSetNamesCsvReaderException), Message: "Failed to read dataset_names.csv. Exception: {0}" ); - public static ErrorViewModel GenerateErrorDatasetNamesCsvReaderException(string exception) + public static ErrorViewModel GenerateErrorDataSetNamesCsvReaderException(string exception) { return new ErrorViewModel { - Code = DatasetNamesCsvReaderException.Code, - Message = string.Format(DatasetNamesCsvReaderException.Message, exception), + Code = DataSetNamesCsvReaderException.Code, + Message = string.Format(DataSetNamesCsvReaderException.Message, exception), }; } - public static readonly LocalizableMessage DatasetNamesCsvIncorrectHeaders = new( - Code: nameof(DatasetNamesCsvIncorrectHeaders), + public static readonly LocalizableMessage DataSetNamesCsvIncorrectHeaders = new( + Code: nameof(DataSetNamesCsvIncorrectHeaders), Message: "dataset_names.csv has incorrect headers. It should have 'file_name' and 'dataset_name' only." ); @@ -158,22 +158,22 @@ public static ErrorViewModel GenerateErrorDataSetTitleShouldBeUnique(string dupl }; } - public static readonly LocalizableMessage DatasetNamesCsvFilenamesShouldBeUnique = new( - Code: nameof(DatasetNamesCsvFilenamesShouldBeUnique), + public static readonly LocalizableMessage DataSetNamesCsvFilenamesShouldBeUnique = new( + Code: nameof(DataSetNamesCsvFilenamesShouldBeUnique), Message: "In dataset_names.csv, all filenames should be unique. Duplicate filename: '{0}'." ); - public static ErrorViewModel GenerateErrorDatasetNamesCsvFilenamesShouldBeUnique(string duplicate) + public static ErrorViewModel GenerateErrorDataSetNamesCsvFilenamesShouldBeUnique(string duplicate) { return new ErrorViewModel { - Code = DatasetNamesCsvFilenamesShouldBeUnique.Code, - Message = string.Format(DatasetNamesCsvFilenamesShouldBeUnique.Message, duplicate), + Code = DataSetNamesCsvFilenamesShouldBeUnique.Code, + Message = string.Format(DataSetNamesCsvFilenamesShouldBeUnique.Message, duplicate), }; } - public static readonly LocalizableMessage DatasetNamesCsvFilenamesShouldNotEndDotCsv = new( - Code: nameof(DatasetNamesCsvFilenamesShouldNotEndDotCsv), + public static readonly LocalizableMessage DataSetNamesCsvFilenamesShouldNotEndDotCsv = new( + Code: nameof(DataSetNamesCsvFilenamesShouldNotEndDotCsv), Message: "Inside dataset_names.csv, file_name cell entries should not end in '.csv' i.e. should be 'filename' not 'filename.csv'. Filename found with extension: '{0}'." ); @@ -205,26 +205,26 @@ public static ErrorViewModel GenerateErrorDataReplacementAlreadyInProgress() }; } - public static readonly LocalizableMessage DatasetTitleTooLong = new( - Code: nameof(DatasetTitleTooLong), + public static readonly LocalizableMessage DataSetTitleTooLong = new( + Code: nameof(DataSetTitleTooLong), Message: "Subject title '{0}' must be 120 characters or less" ); - public static ErrorViewModel GenerateErrorDatasetTitleTooLong(string title) + public static ErrorViewModel GenerateErrorDataSetTitleTooLong(string title) { return new ErrorViewModel { - Code = DatasetTitleTooLong.Code, - Message = string.Format(DatasetTitleTooLong.Message, title), + Code = DataSetTitleTooLong.Code, + Message = string.Format(DataSetTitleTooLong.Message, title), }; } - public static ErrorViewModel GenerateErrorDatasetNamesCsvFilenamesShouldNotEndDotCsv(string filename) + public static ErrorViewModel GenerateErrorDataSetNamesCsvFilenamesShouldNotEndDotCsv(string filename) { return new ErrorViewModel { - Code = DatasetNamesCsvFilenamesShouldNotEndDotCsv.Code, - Message = string.Format(DatasetNamesCsvFilenamesShouldNotEndDotCsv.Message, filename), + Code = DataSetNamesCsvFilenamesShouldNotEndDotCsv.Code, + Message = string.Format(DataSetNamesCsvFilenamesShouldNotEndDotCsv.Message, filename), }; } diff --git a/src/GovUk.Education.ExploreEducationStatistics.Common/Constants/ValidationConstants.cs b/src/GovUk.Education.ExploreEducationStatistics.Common/Constants/ValidationConstants.cs new file mode 100644 index 00000000000..69bf57be261 --- /dev/null +++ b/src/GovUk.Education.ExploreEducationStatistics.Common/Constants/ValidationConstants.cs @@ -0,0 +1,25 @@ +namespace GovUk.Education.ExploreEducationStatistics.Common.Constants; + +public class ValidationConstants +{ + public const int TitleMaxLength = 120; + public const string TitleMaxLengthMessage = "Title must be 120 characters or less"; + + public const int SummaryMaxLength = 250; + public const string SummaryMaxLengthMessage = "Summary must be 250 characters or less"; + + public const int SubjectTitleMaxLength = 120; + public const string SubjectTitleMaxLengthMessage = "Subject title must be 120 characters or less"; + + public const int TableTitleMaxLength = 120; + public const string TableTitleMaxLengthMessage = "Table title must be 120 characters or less"; + + public const int FeaturedTableNameMaxLength = 120; + public const string FeaturedTableNameMaxLengthMessage = "Featured table name must be 120 characters or less"; + + public const int FeaturedTableDescriptionMaxLength = 200; + public const string FeaturedTableDescriptionMaxLengthMessage = "Featured table description must be 200 characters or less"; + + public const int FileGuidanceContentMaxLength = 250; + public const string FileGuidanceContentMaxLengthMessage = "File guidance content must be 250 characters or less"; +} diff --git a/src/explore-education-statistics-admin/src/pages/release/data/ReleaseDataFilePage.tsx b/src/explore-education-statistics-admin/src/pages/release/data/ReleaseDataFilePage.tsx index eb04b5c51dc..f189c26ab73 100644 --- a/src/explore-education-statistics-admin/src/pages/release/data/ReleaseDataFilePage.tsx +++ b/src/explore-education-statistics-admin/src/pages/release/data/ReleaseDataFilePage.tsx @@ -55,6 +55,8 @@ export default function ReleaseDataFilePage({ ); }; + const titleMaxLength = 120; + return ( <> ({ title: Yup.string() .required('Enter a title') - .max(120, 'Subject title must be 120 characters or less'), + .max( + titleMaxLength, + `Subject title must be ${titleMaxLength} characters or less`, + ), })} >
@@ -86,7 +91,7 @@ export default function ReleaseDataFilePage({ className="govuk-!-width-two-thirds" label="Title" name="title" - maxLength={120} + maxLength={titleMaxLength} /> diff --git a/src/explore-education-statistics-admin/src/pages/release/data/components/AncillaryFileForm.tsx b/src/explore-education-statistics-admin/src/pages/release/data/components/AncillaryFileForm.tsx index 777f8815f6a..f8b58a3595a 100644 --- a/src/explore-education-statistics-admin/src/pages/release/data/components/AncillaryFileForm.tsx +++ b/src/explore-education-statistics-admin/src/pages/release/data/components/AncillaryFileForm.tsx @@ -21,6 +21,8 @@ export interface AncillaryFileFormValues { const formId = 'ancillaryFileForm'; const MAX_FILE_SIZE = 2147483647; // 2GB +const titleMaxLength = 120; +const summaryMaxLength = 250; const errorMappings = [ mapFieldErrors({ @@ -77,10 +79,16 @@ export default function AncillaryFileForm({ ); }, }) - .max(120, 'Title must be 120 characters or less'), + .max( + titleMaxLength, + `Title must be ${titleMaxLength} characters or less`, + ), summary: Yup.string() .required('Enter a summary') - .max(250, 'Summary must be 250 characters or less'), + .max( + summaryMaxLength, + `Summary must be ${summaryMaxLength} characters or less`, + ), file: Yup.file() .minSize(0, 'Choose a file that is not empty') .maxSize(MAX_FILE_SIZE, 'Choose a file that is under 2GB') @@ -120,7 +128,7 @@ export default function AncillaryFileForm({ disabled={formState.isSubmitting} label="Title" name="title" - maxLength={120} + maxLength={titleMaxLength} /> @@ -128,7 +136,7 @@ export default function AncillaryFileForm({ disabled={formState.isSubmitting} label="Summary" name="summary" - maxLength={250} + maxLength={summaryMaxLength} /> diff --git a/src/explore-education-statistics-admin/src/pages/release/data/components/DataFileUploadForm.tsx b/src/explore-education-statistics-admin/src/pages/release/data/components/DataFileUploadForm.tsx index 9bceb74470a..6a298d3fa5a 100644 --- a/src/explore-education-statistics-admin/src/pages/release/data/components/DataFileUploadForm.tsx +++ b/src/explore-education-statistics-admin/src/pages/release/data/components/DataFileUploadForm.tsx @@ -29,6 +29,7 @@ export interface DataFileUploadFormValues { } const MAX_FILENAME_SIZE = 150; +const titleMaxLength = 120; const subjectErrorMappings = [ mapFieldErrors({ @@ -70,19 +71,19 @@ function baseErrorMappings( ...fileErrorMappings, ZipFilenameMustEndDotZip: 'ZipFilenameMustEndDotZip', MustBeZipFile: 'MustBeZipFile', - BulkDataZipMustContainDatasetNamesCsv: - 'BulkDataZipMustContainDatasetNamesCsv', - DatasetNamesCsvReaderException: 'DatasetNamesCsvReaderException', - DatasetNamesCsvIncorrectHeaders: 'DatasetNamesCsvIncorrectHeaders', - DatasetNamesCsvFilenamesShouldNotEndDotCsv: - 'DatasetNamesCsvFilenamesShouldNotEndDotCsv', - DatasetNamesCsvFilenamesShouldBeUnique: - 'DatasetNamesCsvFilenamesShouldBeUnique', + BulkDataZipMustContainDataSetNamesCsv: + 'BulkDataZipMustContainDataSetNamesCsv', + DataSetNamesCsvReaderException: 'DataSetNamesCsvReaderException', + DataSetNamesCsvIncorrectHeaders: 'DataSetNamesCsvIncorrectHeaders', + DataSetNamesCsvFilenamesShouldNotEndDotCsv: + 'DataSetNamesCsvFilenamesShouldNotEndDotCsv', + DataSetNamesCsvFilenamesShouldBeUnique: + 'DataSetNamesCsvFilenamesShouldBeUnique', FileNotFoundInZip: 'FileNotFoundInZip', ZipContainsUnusedFiles: 'ZipContainsUnusedFiles', DataReplacementAlreadyInProgress: 'Data replacement already in progress', - DatasetTitleTooLong: 'DatasetTitleTooLong', + DataSetTitleTooLong: 'DataSetTitleTooLong', }, }), ]; @@ -196,7 +197,10 @@ export default function DataFileUploadForm({ ); }, }) - .max(120, 'Subject title must be 120 characters or less'), + .max( + titleMaxLength, + `Subject title must be ${titleMaxLength} characters or less`, + ), }), }); } @@ -236,7 +240,7 @@ export default function DataFileUploadForm({ name="subjectTitle" label="Subject title" className="govuk-!-width-two-thirds" - maxLength={120} + maxLength={titleMaxLength} /> )} diff --git a/src/explore-education-statistics-admin/src/pages/release/data/components/ReleaseDataGuidanceSection.tsx b/src/explore-education-statistics-admin/src/pages/release/data/components/ReleaseDataGuidanceSection.tsx index 90cd76ab81d..80146f9349c 100644 --- a/src/explore-education-statistics-admin/src/pages/release/data/components/ReleaseDataGuidanceSection.tsx +++ b/src/explore-education-statistics-admin/src/pages/release/data/components/ReleaseDataGuidanceSection.tsx @@ -44,6 +44,7 @@ interface Props { canUpdateRelease: boolean; } +const contentMaxLength = 250; const formId = 'dataGuidanceForm'; const ReleaseDataGuidanceSection = ({ releaseId, canUpdateRelease }: Props) => { @@ -126,8 +127,8 @@ const ReleaseDataGuidanceSection = ({ releaseId, canUpdateRelease }: Props) => { return `Enter file guidance content for ${dataSet.name}`; }) .max( - 250, - 'File guidance content must be 250 characters or less', + contentMaxLength, + `File guidance content must be ${contentMaxLength} characters or less`, ), }), ), @@ -184,7 +185,7 @@ const ReleaseDataGuidanceSection = ({ releaseId, canUpdateRelease }: Props) => { label="File guidance content" name={`dataSets.${index}.content`} rows={3} - maxLength={250} + maxLength={contentMaxLength} /> ) : ( ; +const titleMaxLength = 120; +const descriptionMaxLength = 200; const formId = 'dataBlockDetailsForm'; interface Props { @@ -58,14 +60,20 @@ const DataBlockDetailsForm = ({ name: Yup.string().required('Enter a data block name'), heading: Yup.string() .required('Enter a table title') - .max(120, 'Table title must be 120 characters or less'), + .max( + titleMaxLength, + `Table title must be ${titleMaxLength} characters or less`, + ), source: Yup.string(), highlightName: Yup.string().when('isHighlight', { is: true, then: s => s .required('Enter a featured table name') - .max(120, 'Featured table name must be 120 characters or less'), + .max( + titleMaxLength, + `Featured table name must be ${titleMaxLength} characters or less`, + ), }), highlightDescription: Yup.string().when('isHighlight', { is: true, @@ -73,8 +81,8 @@ const DataBlockDetailsForm = ({ s .required('Enter a featured table description') .max( - 200, - 'Featured table description must be 200 characters or less', + descriptionMaxLength, + `Featured table description must be ${descriptionMaxLength} characters or less`, ), }), isHighlight: Yup.boolean(), @@ -111,7 +119,7 @@ const DataBlockDetailsForm = ({ onBlur={() => { onTitleChange?.(getValues('heading')); }} - maxLength={120} + maxLength={titleMaxLength} /> @@ -137,14 +145,14 @@ const DataBlockDetailsForm = ({ label="Featured table name" hint="We will show this name to table builder users as a featured table" className="govuk-!-width-two-thirds" - maxLength={120} + maxLength={titleMaxLength} /> name="highlightDescription" label="Featured table description" hint="Describe the contents of this featured table to table builder users" className="govuk-!-width-two-thirds" - maxLength={200} + maxLength={descriptionMaxLength} /> }