Skip to content

Commit

Permalink
feat: flash validation errors summary to errorsBag
Browse files Browse the repository at this point in the history
  • Loading branch information
Harminder Virk authored and Harminder Virk committed Mar 19, 2024
1 parent a47763a commit badda3f
Show file tree
Hide file tree
Showing 3 changed files with 100 additions and 0 deletions.
1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@
"@adonisjs/assembler": "^7.2.3",
"@adonisjs/core": "^6.3.1",
"@adonisjs/eslint-config": "^1.3.0",
"@adonisjs/i18n": "^2.0.1",
"@adonisjs/prettier-config": "^1.3.0",
"@adonisjs/redis": "^8.0.1",
"@adonisjs/tsconfig": "^1.3.0",
Expand Down
21 changes: 21 additions & 0 deletions src/session.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
* file that was distributed with this source code.
*/

import type { I18n } from '@adonisjs/i18n'
import lodash from '@poppinss/utils/lodash'
import { cuid } from '@adonisjs/core/helpers'
import type { HttpContext } from '@adonisjs/core/http'
Expand Down Expand Up @@ -334,6 +335,26 @@ export class Session {

this.flashExcept(['_csrf', '_method', 'password', 'password_confirmation'])

/**
* Adding the error summary to the "errorsBag" so that
* we display the validation error globally using
* the "@error" tag.
*/
let summary = 'The form could not be saved. Please check the errors below.'
if ('i18n' in this.#ctx) {
summary = (this.#ctx.i18n as I18n).t(
`errors.${error.code}`,
{
count: error.messages.length,
},
summary
)
}

this.flashErrors({
[String(error.code)]: summary,
})

/**
* Adding to inputErrorsBag for "@inputError" tag
* to read validation errors
Expand Down
78 changes: 78 additions & 0 deletions tests/session.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ import { SimpleErrorReporter } from '@vinejs/vine'
import { CookieClient } from '@adonisjs/core/http'
import { fieldContext } from '@vinejs/vine/factories'
import { AppFactory } from '@adonisjs/core/factories/app'
import { I18nManagerFactory } from '@adonisjs/i18n/factories'
import { ApplicationService, EventsList } from '@adonisjs/core/types'
import { EncryptionFactory } from '@adonisjs/core/factories/encryption'
import EdgeServiceProvider from '@adonisjs/core/providers/edge_provider'
Expand Down Expand Up @@ -973,6 +974,80 @@ test.group('Session | Flash', (group) => {
email: ['Invalid email'],
username: ['Invalid username', 'Username is required'],
},
errorsBag: {
E_VALIDATION_ERROR: 'The form could not be saved. Please check the errors below.',
},
inputErrorsBag: {
email: ['Invalid email'],
username: ['Invalid username', 'Username is required'],
},
},
})
})

test('translate validation error summary', async ({ assert }) => {
const i18nManager = new I18nManagerFactory()
.merge({
config: {
loaders: [
() => {
return {
async load() {
return {
en: {
'errors.E_VALIDATION_ERROR': '{count} errors prohibited form submission',
},
}
},
}
},
],
},
})
.create()

await i18nManager.loadTranslations()

let sessionId: string | undefined

const server = httpServer.create(async (req, res) => {
const request = new RequestFactory().merge({ req, res, encryption }).create()
const response = new ResponseFactory().merge({ req, res, encryption }).create()
const ctx = new HttpContextFactory().merge({ request, response }).create()
ctx.i18n = i18nManager.locale('en')

const session = new Session(sessionConfig, cookieDriver, emitter, ctx)
await session.initiate(false)

const errorReporter = new SimpleErrorReporter()
errorReporter.report('Invalid username', 'alpha', fieldContext.create('username', ''), {})
errorReporter.report(
'Username is required',
'required',
fieldContext.create('username', ''),
{}
)
errorReporter.report('Invalid email', 'email', fieldContext.create('email', ''), {})

session.flashValidationErrors(errorReporter.createError())
sessionId = session.sessionId

await session.commit()
response.finish()
})

const { headers } = await supertest(server).get('/')
const cookies = setCookieParser.parse(headers['set-cookie'], { map: true })

assert.deepEqual(cookieClient.decrypt(sessionId!, cookies[sessionId!].value), {
__flash__: {
errors: {
email: ['Invalid email'],
username: ['Invalid username', 'Username is required'],
},
errorsBag: {
E_VALIDATION_ERROR: '3 errors prohibited form submission',
},
inputErrorsBag: {
email: ['Invalid email'],
username: ['Invalid username', 'Username is required'],
Expand Down Expand Up @@ -1021,6 +1096,9 @@ test.group('Session | Flash', (group) => {
errors: {
name: ['Invalid name'],
},
errorsBag: {
E_VALIDATION_ERROR: 'The form could not be saved. Please check the errors below.',
},
inputErrorsBag: {
name: ['Invalid name'],
},
Expand Down

0 comments on commit badda3f

Please sign in to comment.