Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/mongoose-unique-validator/mongoose-unique-validator
mongoose-unique-validator is a plugin which adds pre-save validation for unique fields within a Mongoose schema.
https://github.com/mongoose-unique-validator/mongoose-unique-validator
Last synced: about 22 hours ago
JSON representation
mongoose-unique-validator is a plugin which adds pre-save validation for unique fields within a Mongoose schema.
- Host: GitHub
- URL: https://github.com/mongoose-unique-validator/mongoose-unique-validator
- Owner: mongoose-unique-validator
- Created: 2013-07-09T00:02:01.000Z (over 11 years ago)
- Default Branch: master
- Last Pushed: 2024-09-30T17:00:55.000Z (about 1 month ago)
- Last Synced: 2024-10-30T03:42:24.400Z (15 days ago)
- Language: JavaScript
- Homepage:
- Size: 221 KB
- Stars: 548
- Watchers: 8
- Forks: 74
- Open Issues: 14
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- Contributing: CONTRIBUTING.md
Awesome Lists containing this project
README
mongoose-unique-validator
=========================mongoose-unique-validator is a plugin which adds pre-save validation for unique fields within a Mongoose schema.
This makes error handling much easier, since you will get a Mongoose validation error when you attempt to violate a
[unique constraint](http://mongoosejs.com/docs/api.html#schematype_SchemaType-unique), rather than an E11000 error
from MongoDB.Usage
-----Yarn: `yarn add mongoose-unique-validator`
NPM: `npm install --save mongoose-unique-validator`
Then, apply the plugin to your schema:
```js
const mongoose = require('mongoose');
const uniqueValidator = require('mongoose-unique-validator');const mySchema = mongoose.Schema(/* put your schema definition here */);
mySchema.plugin(uniqueValidator);
```Example
-------Let’s say you have a user schema. You can easily add validation for the unique constraints in this schema by applying
the `uniqueValidator` plugin to your user schema:```js
const mongoose = require('mongoose');
const uniqueValidator = require('mongoose-unique-validator');// Define your schema as normal.
const userSchema = mongoose.Schema({
username: { type: String, required: true, unique: true },
email: { type: String, index: true, unique: true, required: true },
password: { type: String, required: true }
});// Apply the uniqueValidator plugin to userSchema.
userSchema.plugin(uniqueValidator);
```Now when you try to save a user, the unique validator will check for duplicate database entries and report them just
like any other validation error:```js
const user = new User({ username: 'JohnSmith', email: '[email protected]', password: 'j0hnNYb0i' });
user.save(function (err) {
console.log(err);
});
``````js
{
message: 'Validation failed',
name: 'ValidationError',
errors: {
username: {
message: 'Error, expected `username` to be unique. Value: `JohnSmith`',
name: 'ValidatorError',
kind: 'unique',
path: 'username',
value: 'JohnSmith'
}
}
}
```Find + Updates
--------------When using `findOneAndUpdate` and related methods, mongoose doesn't automatically run validation. To trigger this,
you need to pass a configuration object. For technical reasons, this plugin requires that you also set the context
option to `query`.`{ runValidators: true, context: 'query' }`
A full example:
```js
User.findOneAndUpdate(
{ email: '[email protected]' },
{ email: '[email protected]' },
{ runValidators: true, context: 'query' },
function(err) {
// ...
}
)
```Custom Error Types
------------------You can pass through a custom error type as part of the optional `options` argument:
```js
userSchema.plugin(uniqueValidator, { type: 'mongoose-unique-validator' });
```After running the above example the output will be:
```js
{
message: 'Validation failed',
name: 'ValidationError',
errors: {
username: {
message: 'Error, expected `username` to be unique. Value: `JohnSmith`',
name: 'ValidatorError',
kind: 'mongoose-unique-validator',
path: 'username',
value: 'JohnSmith'
}
}
}
```You can also specify a default custom error type by overriding the plugin `defaults.type` variable:
```js
uniqueValidator.defaults.type = 'mongoose-unique-validator'
```Custom Error Messages
---------------------You can pass through a custom error message as part of the optional `options` argument:
```js
userSchema.plugin(uniqueValidator, { message: 'Error, expected {PATH} to be unique.' });
```You have access to all of the standard Mongoose error message templating:
* `{PATH}`
* `{VALUE}`
* `{TYPE}`You can also specify a default custom error message by overriding the plugin `defaults.message` variable:
```js
uniqueValidator.defaults.message = 'Error, expected {PATH} to be unique.'
```Case Insensitive
---------------------For case-insensitive matches, include the `uniqueCaseInsensitive` option in your schema. Queries will treat `[email protected]` and `[email protected]` as duplicates.
```js
const userSchema = mongoose.Schema({
username: { type: String, required: true, unique: true },
email: { type: String, index: true, unique: true, required: true, uniqueCaseInsensitive: true },
password: { type: String, required: true }
});
```Additional Conditions
---------------------For additional unique-constraint conditions (ex: only enforce unique constraint on non soft-deleted records), the MongoDB option `partialFilterExpression` can be used.
Note: the option `index` must be passed as an object containing `unique: true`, or else `partialFilterExpression` will be ignored.
```js
const userSchema = mongoose.Schema({
username: { type: String, required: true, unique: true },
email: {
type: String,
required: true,
index: {
unique: true,
partialFilterExpression: { deleted: false }
}
},
password: { type: String, required: true }
});
```Caveats
-------Because we rely on async operations to verify whether a document exists in the database, it's possible for two queries to execute at the same time, both get 0 back, and then both insert into MongoDB.
Outside of automatically locking the collection or forcing a single connection, there's no real solution.
For most of our users this won't be a problem, but is an edge case to be aware of.