From bfc2fc747b92d4abf787f9beb951c5bd4cd57005 Mon Sep 17 00:00:00 2001 From: Tom Jones Date: Wed, 11 Dec 2024 13:32:17 +0000 Subject: [PATCH 1/4] EES-5047: Add character limit validation to various title and summary fields. --- .../Controllers/Api/ReleasesController.cs | 8 +++-- .../Requests/DataBlockRequests.cs | 10 ++++-- .../Requests/DataGuidanceUpdateRequest.cs | 3 +- .../Requests/FeaturedTableRequests.cs | 4 +++ .../Requests/ReleaseFileRequests.cs | 27 ++++++++++++++- .../Services/DataArchiveValidationService.cs | 9 +++-- .../Validators/ValidationMessages.cs | 14 ++++++++ .../release/data/ReleaseDataFilePage.tsx | 5 ++- .../data/components/AncillaryFileForm.tsx | 9 +++-- .../data/components/DataFileUploadForm.tsx | 33 +++++++++++-------- .../components/ReleaseDataGuidanceSection.tsx | 26 +++++++++------ .../components/DataBlockDetailsForm.tsx | 20 +++++++++-- 12 files changed, 129 insertions(+), 39 deletions(-) diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Controllers/Api/ReleasesController.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Controllers/Api/ReleasesController.cs index 331b6667e50..75d34de36fd 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Controllers/Api/ReleasesController.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Controllers/Api/ReleasesController.cs @@ -107,7 +107,9 @@ public async Task>> ReorderDataFiles(Guid releas [RequestFormLimits(ValueLengthLimit = int.MaxValue, MultipartBodyLengthLimit = int.MaxValue)] public async Task> UploadDataSet(Guid releaseVersionId, [FromQuery(Name = "replacingFileId")] Guid? replacingFileId, - [FromQuery(Name = "title")] string title, + [FromQuery(Name = "title")] + [MaxLength(120, ErrorMessage = "Subject title must be 120 characters or less")] + string title, IFormFile file, IFormFile metaFile) { @@ -125,7 +127,9 @@ public async Task> UploadDataSet(Guid releaseVersionI [RequestFormLimits(ValueLengthLimit = int.MaxValue, MultipartBodyLengthLimit = int.MaxValue)] public async Task> UploadDataSetAsZip(Guid releaseVersionId, [FromQuery(Name = "replacingFileId")] Guid? replacingFileId, - [FromQuery(Name = "title")] string title, + [FromQuery(Name = "title")] + [MaxLength(120, ErrorMessage = "Subject title must be 120 characters or less")] + string title, IFormFile zipFile) { return await _releaseDataFileService diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataBlockRequests.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataBlockRequests.cs index 947b9e36324..88f1809d9df 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataBlockRequests.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataBlockRequests.cs @@ -1,9 +1,9 @@ #nullable enable -using System.Collections.Generic; using FluentValidation; using GovUk.Education.ExploreEducationStatistics.Common.Model.Chart; using GovUk.Education.ExploreEducationStatistics.Common.Model.Data; using GovUk.Education.ExploreEducationStatistics.Common.Requests; +using System.Collections.Generic; namespace GovUk.Education.ExploreEducationStatistics.Admin.Requests; @@ -26,7 +26,9 @@ public class Validator : AbstractValidator public Validator() { RuleFor(request => request.Heading) - .NotEmpty(); + .NotEmpty() + .MaximumLength(120) + .WithMessage("Table title must be 120 characters or less"); RuleFor(request => request.Name) .NotEmpty(); @@ -56,7 +58,9 @@ public class Validator : AbstractValidator public Validator() { RuleFor(request => request.Heading) - .NotEmpty(); + .NotEmpty() + .MaximumLength(120) + .WithMessage("Table title must be 120 characters or less"); 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 1183e3dd119..d8bf0dd14b7 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataGuidanceUpdateRequest.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataGuidanceUpdateRequest.cs @@ -1,4 +1,4 @@ -#nullable enable +#nullable enable using System; using System.Collections.Generic; using System.ComponentModel.DataAnnotations; @@ -20,5 +20,6 @@ public record DataGuidanceDataSetUpdateRequest 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; } diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/FeaturedTableRequests.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/FeaturedTableRequests.cs index eb61edee5a1..ae9430d871e 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/FeaturedTableRequests.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/FeaturedTableRequests.cs @@ -7,9 +7,11 @@ 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; } @@ -19,8 +21,10 @@ public record FeaturedTableCreateRequest 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; } diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/ReleaseFileRequests.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/ReleaseFileRequests.cs index f406c0fd134..b07f72f1f95 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/ReleaseFileRequests.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/ReleaseFileRequests.cs @@ -1,6 +1,7 @@ #nullable enable -using System.ComponentModel.DataAnnotations; +using FluentValidation; using Microsoft.AspNetCore.Http; +using System.ComponentModel.DataAnnotations; namespace GovUk.Education.ExploreEducationStatistics.Admin.Requests; @@ -9,6 +10,16 @@ public record ReleaseDataFileUpdateRequest public string? Title { get; set; } public string? Summary { get; set; } + + public class Validator : AbstractValidator + { + public Validator() + { + RuleFor(request => request.Title) + .MaximumLength(120) + .WithMessage("Subject title must be 120 characters or less"); + } + } } public record ReleaseAncillaryFileUploadRequest @@ -20,6 +31,20 @@ public record ReleaseAncillaryFileUploadRequest [Required] public IFormFile File { get; set; } = null!; + + public class Validator : AbstractValidator + { + public Validator() + { + RuleFor(request => request.Title) + .MaximumLength(120) + .WithMessage("Title must be 120 characters or less"); + + RuleFor(request => request.Summary) + .MaximumLength(250) + .WithMessage("Summary must be 250 characters or less"); + } + } } public record ReleaseAncillaryFileUpdateRequest diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Services/DataArchiveValidationService.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Services/DataArchiveValidationService.cs index 71153aea010..cd476108e49 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Services/DataArchiveValidationService.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Services/DataArchiveValidationService.cs @@ -134,17 +134,22 @@ public async Task>> ValidateBulkDa rows = await CsvUtils.GetCsvRows(dataSetNamesStream); } + var errors = new List(); + var dataSetNamesCsvEntries = new List<(string BaseFilename, string Title)>(); foreach (var row in rows) { var filename = row[fileNameIndex]; var datasetName = row[datasetNameIndex].Trim(); + if (datasetName.Length > 120) + { + errors.Add(ValidationMessages.GenerateErrorDatasetTitleTooLong(datasetName)); + } + dataSetNamesCsvEntries.Add((BaseFilename: filename, Title: datasetName)); } - var errors = new List(); - dataSetNamesCsvEntries .Select(entry => entry.BaseFilename) .Where(baseFilename => baseFilename.EndsWith(".csv")) diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Validators/ValidationMessages.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Validators/ValidationMessages.cs index 46f26173e4b..aedf5d4d042 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Validators/ValidationMessages.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Validators/ValidationMessages.cs @@ -205,6 +205,20 @@ public static ErrorViewModel GenerateErrorDataReplacementAlreadyInProgress() }; } + 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) + { + return new ErrorViewModel + { + Code = DatasetTitleTooLong.Code, + Message = string.Format(DatasetTitleTooLong.Message, title), + }; + } + public static ErrorViewModel GenerateErrorDatasetNamesCsvFilenamesShouldNotEndDotCsv(string filename) { return new ErrorViewModel 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 1fbbd215c72..eb04b5c51dc 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 @@ -76,7 +76,9 @@ export default function ReleaseDataFilePage({ ({ - title: Yup.string().required('Enter a title'), + title: Yup.string() + .required('Enter a title') + .max(120, 'Subject title must be 120 characters or less'), })} >
@@ -84,6 +86,7 @@ export default function ReleaseDataFilePage({ className="govuk-!-width-two-thirds" label="Title" name="title" + maxLength={120} /> 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 e47434e8d4a..777f8815f6a 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 @@ -76,8 +76,11 @@ export default function AncillaryFileForm({ f => f.title.toUpperCase() !== value.toUpperCase(), ); }, - }), - summary: Yup.string().required('Enter a summary'), + }) + .max(120, 'Title must be 120 characters or less'), + summary: Yup.string() + .required('Enter a summary') + .max(250, 'Summary must be 250 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') @@ -117,6 +120,7 @@ export default function AncillaryFileForm({ disabled={formState.isSubmitting} label="Title" name="title" + maxLength={120} /> @@ -124,6 +128,7 @@ export default function AncillaryFileForm({ disabled={formState.isSubmitting} label="Summary" name="summary" + maxLength={250} /> 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 59df8a9c1d6..b188f279ea0 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 @@ -83,6 +83,7 @@ function baseErrorMappings( ZipContainsUnusedFiles: 'ZipContainsUnusedFiles', DataReplacementAlreadyInProgress: 'Data replacement already in progress', + DatasetTitleTooLong: 'DatasetTitleTooLong', }, }), ]; @@ -179,21 +180,24 @@ export default function DataFileUploadForm({ is: (uploadType: FileType) => uploadType === 'csv' || uploadType === 'zip', then: s => - s.required('Enter a subject title').test({ - name: 'unique', - message: 'Enter a unique subject title', - test(value: string) { - if (!value) { - return true; - } + s + .required('Enter a subject title') + .test({ + name: 'unique', + message: 'Enter a unique subject title', + test(value: string) { + if (!value) { + return true; + } - return ( - dataFiles?.find( - f => f.title.toUpperCase() === value.toUpperCase(), - ) === undefined - ); - }, - }), + return ( + dataFiles?.find( + f => f.title.toUpperCase() === value.toUpperCase(), + ) === undefined + ); + }, + }) + .max(120, 'Subject title must be 120 characters or less'), }), }); } @@ -233,6 +237,7 @@ export default function DataFileUploadForm({ name="subjectTitle" label="Subject title" className="govuk-!-width-two-thirds" + maxLength={120} /> )} 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 efcc4dc3dae..90cd76ab81d 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 @@ -114,16 +114,21 @@ const ReleaseDataGuidanceSection = ({ releaseId, canUpdateRelease }: Props) => { dataSets: Yup.array().of( Yup.object({ id: Yup.string(), - content: Yup.string().required(params => { - const [, index] = toPath(params.path); - const dataSet = dataGuidance?.dataSets[Number(index)]; - - if (!dataSet) { - return null; - } - - return `Enter file guidance content for ${dataSet.name}`; - }), + content: Yup.string() + .required(params => { + const [, index] = toPath(params.path); + const dataSet = dataGuidance?.dataSets[Number(index)]; + + if (!dataSet) { + return null; + } + + return `Enter file guidance content for ${dataSet.name}`; + }) + .max( + 250, + 'File guidance content must be 250 characters or less', + ), }), ), })} @@ -179,6 +184,7 @@ const ReleaseDataGuidanceSection = ({ releaseId, canUpdateRelease }: Props) => { label="File guidance content" name={`dataSets.${index}.content`} rows={3} + maxLength={250} /> ) : ( >(() => { return Yup.object({ name: Yup.string().required('Enter a data block name'), - heading: Yup.string().required('Enter a table title'), + heading: Yup.string() + .required('Enter a table title') + .max(120, 'Table title must be 120 characters or less'), source: Yup.string(), highlightName: Yup.string().when('isHighlight', { is: true, - then: s => s.required('Enter a featured table name'), + then: s => + s + .required('Enter a featured table name') + .max(120, 'Featured table name must be 120 characters or less'), }), highlightDescription: Yup.string().when('isHighlight', { is: true, - then: s => s.required('Enter a featured table description'), + then: s => + s + .required('Enter a featured table description') + .max( + 200, + 'Featured table description must be 200 characters or less', + ), }), isHighlight: Yup.boolean(), }); @@ -100,6 +111,7 @@ const DataBlockDetailsForm = ({ onBlur={() => { onTitleChange?.(getValues('heading')); }} + maxLength={120} /> @@ -125,12 +137,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} /> 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} /> } From 6dd4b389c866463ade20eca5cb6fef36fd60e1ac Mon Sep 17 00:00:00 2001 From: Tom Jones Date: Wed, 11 Dec 2024 13:32:49 +0000 Subject: [PATCH 2/4] EES-5047: Move input/textarea useWatch usage to parent components and update tests. --- .../src/components/form/FormFieldTextArea.tsx | 28 ++- .../components/form/FormFieldTextInput.tsx | 28 ++- .../src/components/form/FormTextArea.tsx | 32 +-- .../src/components/form/FormTextInput.tsx | 18 -- .../form/__tests__/FormFieldTextArea.test.tsx | 136 ++++++++++++ .../form/__tests__/FormTextArea.test.tsx | 210 +++--------------- 6 files changed, 221 insertions(+), 231 deletions(-) create mode 100644 src/explore-education-statistics-common/src/components/form/__tests__/FormFieldTextArea.test.tsx diff --git a/src/explore-education-statistics-common/src/components/form/FormFieldTextArea.tsx b/src/explore-education-statistics-common/src/components/form/FormFieldTextArea.tsx index 246ba3837e1..72369488537 100644 --- a/src/explore-education-statistics-common/src/components/form/FormFieldTextArea.tsx +++ b/src/explore-education-statistics-common/src/components/form/FormFieldTextArea.tsx @@ -4,15 +4,35 @@ import FormField, { import FormTextArea from '@common/components/form/FormTextArea'; import { FormTextAreaProps } from '@common/components/form/FormBaseTextArea'; import React from 'react'; -import { FieldValues } from 'react-hook-form'; +import { FieldValues, useWatch } from 'react-hook-form'; +import FormCharacterCount from '@common/components/form/FormCharacterCount'; +import FormGroup from './FormGroup'; type Props = FormFieldComponentProps< FormTextAreaProps, TFormValues >; -export default function FormFieldTextArea( - props: Props, -) { +export default function FormFieldTextArea({ + maxLength, + ...props +}: Props) { + const watchedValue = useWatch({ name: props.name }); + + if (!!maxLength && maxLength > 0) { + return ( +
+ + + + +
+ ); + } + return ; } diff --git a/src/explore-education-statistics-common/src/components/form/FormFieldTextInput.tsx b/src/explore-education-statistics-common/src/components/form/FormFieldTextInput.tsx index abc58788276..90aba48798d 100644 --- a/src/explore-education-statistics-common/src/components/form/FormFieldTextInput.tsx +++ b/src/explore-education-statistics-common/src/components/form/FormFieldTextInput.tsx @@ -5,15 +5,35 @@ import FormTextInput, { FormTextInputProps, } from '@common/components/form/FormTextInput'; import React from 'react'; -import { FieldValues } from 'react-hook-form'; +import { FieldValues, useWatch } from 'react-hook-form'; +import FormCharacterCount from '@common/components/form/FormCharacterCount'; +import FormGroup from './FormGroup'; type Props = FormFieldComponentProps< FormTextInputProps, TFormValues >; -export default function FormFieldTextInput( - props: Props, -) { +export default function FormFieldTextInput({ + maxLength, + ...props +}: Props) { + const watchedValue = useWatch({ name: props.name }); + + if (!!maxLength && maxLength > 0) { + return ( +
+ + + + +
+ ); + } + return ; } diff --git a/src/explore-education-statistics-common/src/components/form/FormTextArea.tsx b/src/explore-education-statistics-common/src/components/form/FormTextArea.tsx index 6aa385bf9bf..2ebf6b27e5f 100644 --- a/src/explore-education-statistics-common/src/components/form/FormTextArea.tsx +++ b/src/explore-education-statistics-common/src/components/form/FormTextArea.tsx @@ -1,36 +1,8 @@ -import FormGroup from '@common/components/form/FormGroup'; -import FormCharacterCount from '@common/components/form/FormCharacterCount'; import FormBaseTextArea, { FormTextAreaProps, } from '@common/components/form/FormBaseTextArea'; import React from 'react'; -import { useWatch } from 'react-hook-form'; -export default function FormTextArea({ - id, - maxLength, - name, - ...props -}: FormTextAreaProps) { - const value = useWatch({ name }); - - if (!!maxLength && maxLength > 0) { - return ( -
- - - - -
- ); - } - - return ( - - ); +export default function FormTextArea(props: FormTextAreaProps) { + return ; } diff --git a/src/explore-education-statistics-common/src/components/form/FormTextInput.tsx b/src/explore-education-statistics-common/src/components/form/FormTextInput.tsx index 81e8bc82b37..0108fa1f1e0 100644 --- a/src/explore-education-statistics-common/src/components/form/FormTextInput.tsx +++ b/src/explore-education-statistics-common/src/components/form/FormTextInput.tsx @@ -1,8 +1,6 @@ import FormBaseInput, { FormBaseInputProps, } from '@common/components/form/FormBaseInput'; -import FormCharacterCount from '@common/components/form/FormCharacterCount'; -import FormGroup from '@common/components/form/FormGroup'; import React from 'react'; export interface FormTextInputProps extends FormBaseInputProps { @@ -13,24 +11,8 @@ export interface FormTextInputProps extends FormBaseInputProps { export default function FormTextInput({ id, - maxLength, value, ...props }: FormTextInputProps) { - if (!!maxLength && maxLength > 0) { - return ( -
- - - - -
- ); - } return ; } diff --git a/src/explore-education-statistics-common/src/components/form/__tests__/FormFieldTextArea.test.tsx b/src/explore-education-statistics-common/src/components/form/__tests__/FormFieldTextArea.test.tsx new file mode 100644 index 00000000000..733ba544071 --- /dev/null +++ b/src/explore-education-statistics-common/src/components/form/__tests__/FormFieldTextArea.test.tsx @@ -0,0 +1,136 @@ +import FormFieldTextArea from '@common/components/form/FormFieldTextArea'; +import FormProvider from '@common/components/form/FormProvider'; +import { render, screen } from '@testing-library/react'; +import noop from 'lodash/noop'; +import React from 'react'; + +describe('FormFieldTextArea', () => { + describe('maxLength', () => { + test('shows a character count message when `maxLength` is above 0', () => { + render( + + + , + ); + + expect( + screen.getByText('You have 10 characters remaining'), + ).toBeInTheDocument(); + }); + + test('aria-describedby contains the character count message id when `maxLength` is above 0', () => { + render( + + + , + ); + + const ariaDescribedBy = screen + .getByLabelText('Test input') + .getAttribute('aria-describedby'); + + expect( + screen.getByText('You have 10 characters remaining'), + ).toHaveAttribute('id', 'test-input-info'); + expect(ariaDescribedBy).toContain('test-input-info'); + }); + + test('does not show a character count message when `maxLength` is below 0', () => { + render( + + + , + ); + + expect( + screen.queryByText(/You have .+ characters remaining/), + ).not.toBeInTheDocument(); + }); + + test('does not show a character count message when `maxLength` is 0', () => { + render( + + + , + ); + + expect( + screen.queryByText(/You have .+ characters remaining/), + ).not.toBeInTheDocument(); + }); + + test('shows correct character count message when difference to `maxLength` is 1', () => { + render( + + + , + ); + + expect( + screen.getByText('You have 1 character remaining'), + ).toBeInTheDocument(); + }); + + test('shows correct character count message when difference to `maxLength` is 0', () => { + render( + + + , + ); + + expect( + screen.getByText('You have 0 characters remaining'), + ).toBeInTheDocument(); + }); + + test('shows correct character count message when difference to `maxLength` is -1', () => { + render( + + + , + ); + + expect( + screen.getByText('You have 1 character too many'), + ).toBeInTheDocument(); + }); + }); +}); diff --git a/src/explore-education-statistics-common/src/components/form/__tests__/FormTextArea.test.tsx b/src/explore-education-statistics-common/src/components/form/__tests__/FormTextArea.test.tsx index 9b86fd200ab..138eb0a23d7 100644 --- a/src/explore-education-statistics-common/src/components/form/__tests__/FormTextArea.test.tsx +++ b/src/explore-education-statistics-common/src/components/form/__tests__/FormTextArea.test.tsx @@ -1,5 +1,4 @@ import FormTextArea from '@common/components/form/FormTextArea'; -import FormProvider from '@common/components/form/FormProvider'; import { render, screen } from '@testing-library/react'; import noop from 'lodash/noop'; import React from 'react'; @@ -7,9 +6,7 @@ import React from 'react'; describe('FormTextArea', () => { test('renders correctly with required props', () => { const { container } = render( - - - , + , ); expect(screen.getByLabelText('Test input')).toBeDefined(); @@ -18,14 +15,12 @@ describe('FormTextArea', () => { test('renders correctly with hint', () => { const { container } = render( - - - , + , ); const hint = screen.getByText('Fill me in'); @@ -36,15 +31,13 @@ describe('FormTextArea', () => { test('renders correctly with error', () => { const { container } = render( - - - , + , ); const error = screen.getByText('Field is required'); @@ -55,15 +48,13 @@ describe('FormTextArea', () => { test('aria-describedby is equal to the hint id', () => { render( - - - , + , ); expect(screen.getByText('Fill me in')).toHaveAttribute( @@ -78,15 +69,13 @@ describe('FormTextArea', () => { test('aria-describedby is equal to the error id', () => { render( - - - , + , ); expect(screen.getByText('Field is required')).toHaveAttribute( @@ -101,15 +90,13 @@ describe('FormTextArea', () => { test('aria-describedby contains both hint and error ids', () => { render( - - - , + , ); expect(screen.getByText('Fill me in')).toHaveAttribute( @@ -128,131 +115,4 @@ describe('FormTextArea', () => { expect(ariaDescribedBy).toContain('test-input-error'); expect(ariaDescribedBy).toContain('test-input-hint'); }); - - test('shows a character count message when `maxLength` is above 0', () => { - render( - - - , - ); - - expect( - screen.getByText('You have 10 characters remaining'), - ).toBeInTheDocument(); - }); - - test('aria-describedby contains the character count message id when `maxLength` is above 0', () => { - render( - - - , - ); - - const ariaDescribedBy = screen - .getByLabelText('Test input') - .getAttribute('aria-describedby'); - - expect( - screen.getByText('You have 10 characters remaining'), - ).toHaveAttribute('id', 'test-input-info'); - expect(ariaDescribedBy).toContain('test-input-info'); - }); - - test('does not show a character count message when `maxLength` is below 0', () => { - render( - - - , - ); - - expect( - screen.queryByText(/You have .+ characters remaining/), - ).not.toBeInTheDocument(); - }); - - test('does not show a character count message when `maxLength` is 0', () => { - render( - - - , - ); - - expect( - screen.queryByText(/You have .+ characters remaining/), - ).not.toBeInTheDocument(); - }); - - test('shows correct character count message when difference to `maxLength` is 1', () => { - render( - - - , - ); - - expect( - screen.getByText('You have 1 character remaining'), - ).toBeInTheDocument(); - }); - - test('shows correct character count message when difference to `maxLength` is 0', () => { - render( - - - , - ); - - expect( - screen.getByText('You have 0 characters remaining'), - ).toBeInTheDocument(); - }); - - test('shows correct character count message when difference to `maxLength` is -1', () => { - render( - - - , - ); - - expect( - screen.getByText('You have 1 character too many'), - ).toBeInTheDocument(); - }); }); From 2c3c9867e2a24c07c4e60cac3f032dc6481b5196 Mon Sep 17 00:00:00 2001 From: Tom Jones Date: Fri, 13 Dec 2024 12:54:22 +0000 Subject: [PATCH 3/4] 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 b188f279ea0..ac728bd6ba2 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({ @@ -71,19 +72,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', }, }), ]; @@ -197,7 +198,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`, + ), }), }); } @@ -237,7 +241,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} /> } From dd4f0b83a226355c5ff6ca8abf8e83bbc5cb16ea Mon Sep 17 00:00:00 2001 From: Tom Jones Date: Wed, 18 Dec 2024 11:02:22 +0000 Subject: [PATCH 4/4] EES-5047: Remove shared backend validation messaging. --- .../Controllers/Api/ReleasesController.cs | 5 ++-- .../Requests/DataBlockRequests.cs | 7 ++---- .../Requests/DataGuidanceUpdateRequest.cs | 4 +-- .../Requests/FeaturedTableRequests.cs | 13 +++------- .../Requests/ReleaseFileRequests.cs | 16 ++++-------- .../Services/DataArchiveValidationService.cs | 3 +-- .../Constants/ValidationConstants.cs | 25 ------------------- 7 files changed, 15 insertions(+), 58 deletions(-) delete mode 100644 src/GovUk.Education.ExploreEducationStatistics.Common/Constants/ValidationConstants.cs diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Controllers/Api/ReleasesController.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Controllers/Api/ReleasesController.cs index 7f1003b4f63..87440621bcb 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Controllers/Api/ReleasesController.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Controllers/Api/ReleasesController.cs @@ -14,7 +14,6 @@ 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 { @@ -109,7 +108,7 @@ public async Task>> ReorderDataFiles(Guid releas public async Task> UploadDataSet(Guid releaseVersionId, [FromQuery(Name = "replacingFileId")] Guid? replacingFileId, [FromQuery(Name = "title")] - [MaxLength(SubjectTitleMaxLength, ErrorMessage = SubjectTitleMaxLengthMessage)] + [MaxLength(120)] string title, IFormFile file, IFormFile metaFile) @@ -129,7 +128,7 @@ public async Task> UploadDataSet(Guid releaseVersionI public async Task> UploadDataSetAsZip(Guid releaseVersionId, [FromQuery(Name = "replacingFileId")] Guid? replacingFileId, [FromQuery(Name = "title")] - [MaxLength(SubjectTitleMaxLength, ErrorMessage = SubjectTitleMaxLengthMessage)] + [MaxLength(120)] 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 6d2fe6767ce..d81e49514cd 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataBlockRequests.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataBlockRequests.cs @@ -4,7 +4,6 @@ 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; @@ -28,8 +27,7 @@ public Validator() { RuleFor(request => request.Heading) .NotEmpty() - .MaximumLength(TableTitleMaxLength) - .WithMessage(TableTitleMaxLengthMessage); + .MaximumLength(120); RuleFor(request => request.Name) .NotEmpty(); @@ -60,8 +58,7 @@ public Validator() { RuleFor(request => request.Heading) .NotEmpty() - .MaximumLength(TableTitleMaxLength) - .WithMessage(TableTitleMaxLengthMessage); + .MaximumLength(120); 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 b057848d576..e854959c8ee 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataGuidanceUpdateRequest.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/DataGuidanceUpdateRequest.cs @@ -2,7 +2,6 @@ using FluentValidation; using System; using System.Collections.Generic; -using static GovUk.Education.ExploreEducationStatistics.Common.Constants.ValidationConstants; namespace GovUk.Education.ExploreEducationStatistics.Admin.Requests; @@ -40,8 +39,7 @@ public Validator() RuleFor(request => request.Content) .NotEmpty() - .MaximumLength(FileGuidanceContentMaxLength) - .WithMessage(FileGuidanceContentMaxLengthMessage); + .MaximumLength(250); } } } diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/FeaturedTableRequests.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/FeaturedTableRequests.cs index fe6ead1a189..43933bc1241 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/FeaturedTableRequests.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/FeaturedTableRequests.cs @@ -1,7 +1,6 @@ #nullable enable using FluentValidation; using System; -using static GovUk.Education.ExploreEducationStatistics.Common.Constants.ValidationConstants; namespace GovUk.Education.ExploreEducationStatistics.Admin.Requests; @@ -19,13 +18,11 @@ public Validator() { RuleFor(request => request.Name) .NotEmpty() - .MaximumLength(FeaturedTableNameMaxLength) - .WithMessage(FeaturedTableNameMaxLengthMessage); + .MaximumLength(120); RuleFor(request => request.Description) .NotEmpty() - .MaximumLength(FeaturedTableDescriptionMaxLength) - .WithMessage(FeaturedTableDescriptionMaxLengthMessage); + .MaximumLength(200); } } } @@ -42,13 +39,11 @@ public Validator() { RuleFor(request => request.Name) .NotEmpty() - .MaximumLength(FeaturedTableNameMaxLength) - .WithMessage(FeaturedTableNameMaxLengthMessage); + .MaximumLength(120); RuleFor(request => request.Description) .NotEmpty() - .MaximumLength(FeaturedTableDescriptionMaxLength) - .WithMessage(FeaturedTableDescriptionMaxLengthMessage); + .MaximumLength(200); } } } diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/ReleaseFileRequests.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/ReleaseFileRequests.cs index f9bf20a2a86..4f8c11a184c 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/ReleaseFileRequests.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Requests/ReleaseFileRequests.cs @@ -1,7 +1,6 @@ #nullable enable using FluentValidation; using Microsoft.AspNetCore.Http; -using static GovUk.Education.ExploreEducationStatistics.Common.Constants.ValidationConstants; namespace GovUk.Education.ExploreEducationStatistics.Admin.Requests; @@ -16,8 +15,7 @@ public class Validator : AbstractValidator public Validator() { RuleFor(request => request.Title) - .MaximumLength(SubjectTitleMaxLength) - .WithMessage(SubjectTitleMaxLengthMessage); + .MaximumLength(120); } } } @@ -36,13 +34,11 @@ public Validator() { RuleFor(request => request.Title) .NotEmpty() - .MaximumLength(TitleMaxLength) - .WithMessage(TitleMaxLengthMessage); + .MaximumLength(120); RuleFor(request => request.Summary) .NotEmpty() - .MaximumLength(SummaryMaxLength) - .WithMessage(SummaryMaxLengthMessage); + .MaximumLength(250); RuleFor(request => request.File) .NotEmpty(); @@ -64,13 +60,11 @@ public Validator() { RuleFor(request => request.Title) .NotEmpty() - .MaximumLength(TitleMaxLength) - .WithMessage(TitleMaxLengthMessage); + .MaximumLength(120); RuleFor(request => request.Summary) .NotEmpty() - .MaximumLength(SummaryMaxLength) - .WithMessage(SummaryMaxLengthMessage); + .MaximumLength(250); } } } diff --git a/src/GovUk.Education.ExploreEducationStatistics.Admin/Services/DataArchiveValidationService.cs b/src/GovUk.Education.ExploreEducationStatistics.Admin/Services/DataArchiveValidationService.cs index 2731cf1b6ce..ae3b51bcc35 100644 --- a/src/GovUk.Education.ExploreEducationStatistics.Admin/Services/DataArchiveValidationService.cs +++ b/src/GovUk.Education.ExploreEducationStatistics.Admin/Services/DataArchiveValidationService.cs @@ -15,7 +15,6 @@ 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 { @@ -143,7 +142,7 @@ public async Task>> ValidateBulkDa var filename = row[fileNameIndex]; var datasetName = row[datasetNameIndex].Trim(); - if (datasetName.Length > SubjectTitleMaxLength) + if (datasetName.Length > 120) { errors.Add(ValidationMessages.GenerateErrorDataSetTitleTooLong(datasetName)); } diff --git a/src/GovUk.Education.ExploreEducationStatistics.Common/Constants/ValidationConstants.cs b/src/GovUk.Education.ExploreEducationStatistics.Common/Constants/ValidationConstants.cs deleted file mode 100644 index 69bf57be261..00000000000 --- a/src/GovUk.Education.ExploreEducationStatistics.Common/Constants/ValidationConstants.cs +++ /dev/null @@ -1,25 +0,0 @@ -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"; -}