Understanding `null` vs. `undefined` in JavaScript and Best Practices for Mongoose Schemas

Created by eneaslari 18/10/2024

javascript mongoose nodejs

In this blog post, we’ll dive into the differences between null and undefined in JavaScript, how to declare nullable fields in Mongoose schemas, and best practices for creating and managing schemas. These topics often cause confusion among developers, especially when working with MongoDB and Mongoose in a Node.js environment. By the end of this post, you'll have a better understanding of how to handle optional fields, validation, and common pitfalls.


null vs. undefined in JavaScript

1. undefined:

  • Meaning: A variable that has been declared but has not been assigned a value yet.

  • Example:

    let x;
    console.log(x); // Output: undefined
    

    Here, x is declared but not assigned any value, so its value is undefined.

  • Use Case: undefined is typically used by JavaScript itself when a variable is not initialized or a function parameter is missing.

2. null:

  • Meaning: Represents an intentional absence of value. It’s like saying, “this variable should be empty.”

  • Example:

    let y = null;
    console.log(y); // Output: null
    

    Here, y is explicitly set to null, which means it is intentionally empty.

  • Use Case: null is used when you want to explicitly indicate that a variable has no value.

Key Differences Recap:

  • undefined is assigned by JavaScript when a variable is declared but not initialized.
  • null is assigned by developers to indicate intentional absence of a value.

Making Fields Nullable in Mongoose Schemas

When working with Mongoose, you might need to make some fields optional or nullable. Here’s how you can do this:

1. Allowing null for Optional Fields

By default, fields in Mongoose are optional unless specified otherwise. You can set a default value like null to indicate the absence of a value explicitly:

const userSchema = new mongoose.Schema({
  name: {
    type: String,
    required: true,
  },
  age: {
    type: Number,
    default: null, // Age can be null if not provided
  },
  address: {
    type: String,
    default: null, // Address can be null explicitly
  },
});

2. Using required and Allowing null

You can also control the requirement of a field with a custom condition:

const userSchema = new mongoose.Schema({
  bio: {
    type: String,
    required: function() {
      return this.bio !== null; // Required but can be explicitly null
    },
  },
});

3. Using Mixed Type for Flexibility

If you want a field to accept multiple types, including null, you can use Schema.Types.Mixed:

const userSchema = new mongoose.Schema({
  data: {
    type: mongoose.Schema.Types.Mixed,
    default: null, // Can be null or any type
  },
});

What Happens When You Don’t Set a Value for a Field?

The behavior when a field is not provided depends on how the schema is defined:

1. Field with a Default Value:

If a field has a default value, Mongoose will use that value when creating a document without specifying the field.

const userSchema = new mongoose.Schema({
  name: { type: String, required: true },
  age: { type: Number, default: 18 },
});

const user = new User({ name: 'Alice' });
console.log(user.age); // Output: 18

2. Field Without a Default Value:

If a field is optional and has no default value, it will be undefined when not provided.

const userSchema = new mongoose.Schema({
  address: { type: String },
});

const user = new User({ name: 'Alice' });
console.log(user.address); // Output: undefined

3. Required Fields:

If a field is marked as required, you must provide it when creating a document. If not, Mongoose will throw a validation error.


Best Practices for Defining Optional Fields in Mongoose

Here are some tips to ensure your Mongoose schemas are clean, maintainable, and predictable:

1. Use Default Values When Appropriate:

Provide default values for fields when it makes sense for your application logic. This helps maintain consistency:

const userSchema = new mongoose.Schema({
  bio: { type: String, default: '' }, // Default to an empty string
});

2. Keep Optional Fields Undefined:

If a field doesn’t need a specific default value, simply define its type without required:

const userSchema = new mongoose.Schema({
  phoneNumber: { type: String }, // Optional, remains undefined if not provided
});

3. Use null for Intentional Absence:

Use null when you want to clearly indicate that a field is intentionally empty:

const userSchema = new mongoose.Schema({
  profilePicture: { type: String, default: null },
});

4. Avoid Overusing Mixed Types:

Use Schema.Types.Mixed sparingly as it bypasses Mongoose’s schema validation. Define the type if possible for better structure:

const userSchema = new mongoose.Schema({
  metadata: { type: Map, of: String, default: {} },
});

Improving Validation with Async Validators

When checking for the existence of a related object, it’s better to use async functions for validation in Mongoose. Here’s an example:

export const CheckObjectsExists = async (model, id) => {
    const exists = await model.exists({ _id: id });
    if (!exists) {
        throw new Error(`FK Constraint 'CheckObjectsExists' for '${id.toString()}' failed`);
    }
    return true;
};

This function checks if a document with a given ID exists. It uses the exists method for efficiency and returns true if the document exists, otherwise throws an error. This can be used in a Mongoose schema validator like so:

const CategorySchema = new mongoose.Schema({
  parentcategory: {
    type: mongoose.Schema.ObjectId,
    ref: "Category",
    validate: {
      validator: async function (v) {
        if (!v) return true; // Allow `null` values
        return await CheckObjectsExists(mongoose.model("Category"), v);
      },
      message: "Parent category does not exist",
    },
  },
});

Should You Add Default Values to Optional String Fields?

In many cases, it’s better to leave optional string fields without a default value, allowing them to be undefined if not provided:

  • Pros of leaving as undefined: Easier to check if a field was provided, simpler data handling.
  • Pros of using an empty string as default: Consistent data type, avoids checking for undefined.

Example:

const SubscriberSchema = new mongoose.Schema({
  name: { type: String, trim: true },
  email: { type: String, trim: true, required: 'email is required' },
  message: { type: String, trim: true },
  verificationID: { type: String, trim: true },
});

This approach keeps the optional fields as undefined if not provided, which is often preferred for a clean database structure.


Conclusion

By understanding the differences between null and undefined, and applying best practices for Mongoose schemas, you can ensure that your data structures are robust and maintainable. Whether you’re defining optional fields, managing default values, or validating references, these tips should help you create cleaner and more efficient Mongoose models.

Feel free to share your thoughts or ask further questions in the comments below! Happy coding!

More to read