mon-entreprise/server/source/utils.ts

58 lines
1.5 KiB
TypeScript

import { BodyType } from './functions/send-crisp-message.js'
type CamelCase<S extends string> =
S extends `${infer P1}_${infer P2}${infer P3}`
? `${Lowercase<P1>}${Uppercase<P2>}${CamelCase<P3>}`
: Lowercase<S>
const snakeToCamelCase = <T extends string>(str: T) =>
str
.replace(/_/g, ' ')
.replace(/(?<!\p{L})\p{L}|\s+/gu, (m) => (+m === 0 ? '' : m.toUpperCase()))
.replace(/^./, (m) => m?.toLowerCase()) as CamelCase<T>
export type KeysToCamelCase<T> = {
[K in keyof T as CamelCase<string & K>]: T[K]
}
export const snakeToCamelCaseKeys = <T extends object>(object: T) =>
Object.fromEntries(
Object.entries(object).map(
([key, value]) => [snakeToCamelCase(key), value],
{}
)
) as KeysToCamelCase<T>
export const shuffleArray = <T>(array: T[]) => {
const shuffle = [...array]
for (let i = shuffle.length - 1; i > 0; i--) {
const j = Math.floor(Math.random() * (i + 1))
const [shuffleI, shuffleJ] = [shuffle[i], shuffle[j]]
shuffle[i] = shuffleJ
shuffle[j] = shuffleI
}
return shuffle
}
const isStringAndNotEmpty = (value: string) =>
value !== undefined && value !== '' && typeof value === 'string'
const SHORT_MAX_LENGTH = 254
export const validateCrispBody = (body: BodyType): BodyType => {
const { subject, message, email } = body || {}
if (
isStringAndNotEmpty(subject) &&
subject.length <= SHORT_MAX_LENGTH &&
isStringAndNotEmpty(message) &&
typeof email === 'string' &&
email.length <= SHORT_MAX_LENGTH
) {
return body
}
throw Error('Body validation failed.')
}