Introducing Squeaky

It's been a while since I've created a new package, so I'd like to introduce you to Squeaky! A Laravel validation rule to help catch profanity. You can check out the repo below, which includes instructions on how to set it up and some examples to get you started:
So how did Squeaky come about? Well, Profanify recently enjoyed an article on Laravel News, highlighting the package and what it can do. In the Tweet for that article was this reply:
An interesting use case would be to modify this to be a Laravel validator to check request data
— Tim Leland (@TimLeland) February 17, 2025
So that got me thinking, could I extend Profanify to work as a Laravel validation rule, to do the same thing but for user submitted information? I had 3 choices:
- Build it into the existing package
- Build it into Laravel Core
- Build it as a separate package that uses Profanify
I decided to go with the 3rd one as in my mind, it was the best of the 3 for numerous reasons. I did attempt a PR to get it directly into Laravel, but it was getting too messy and I'm 99.9% sure it would not have been merged anyway.
So I set about building Squeaky. This was my first time working on a package specifically to be used in Laravel applications but luckily, Spatie came to the rescue with their Package Skeleton for Laravel applications, this gave me a great head start. You can view the source code to find out exactly how it's built, there's not much code to it really!
Under the hood, Squeaky uses the config files from Profanify, to know what to fail the validation on. There's a really handy method available in Service Providers called mergeConfigFrom
, which I use to merge the Profanify config, with a Squeaky config. The Squeaky config is then used in the validation rule:
$this->mergeConfigFrom(base_path('vendor').'/jonpurvis/profanify/src/Config/profanities/ar.php', 'profanify-ar');
So how does it work? Well, in it's most basic form, let's imagine you have a form that allows a user to write a biography about themselves, in your existing validation rules (which you should have!) you can simply add the new "Clean" rule to them:
return [
'name' => ['required', 'string', 'max:255', new Clean],
'email' => [
'required',
'string',
'lowercase',
'email',
'max:255',
Rule::unique(User::class)->ignore($this->user()->id),
],
'bio' => ['required', 'string', 'max:255', new Clean],
];
That's all! By default it works with your application locale defined in your .env
file however certain applications can work with several locales, so you have the ability to pass in an array of locales and each one will be checked. Thanks to Ash Allen for that contribution!
So by pairing Profanify and Squeaky, you can ensure both your code base and user submitted input is nice and clean 🧼