Skip to content

Convention Packs

Convention packs provide a powerful way to apply consistent configuration across all your MongoDB class mappings. Cratis Applications includes a comprehensive system for creating, providing, and filtering convention packs.

Convention packs are collections of conventions that MongoDB applies automatically to class maps during registration. They allow you to:

  • Apply consistent rules: Ensure all classes follow the same patterns
  • Reduce boilerplate: Avoid repeating configuration in every class map
  • Conditional application: Apply different rules to different types
  • System-wide changes: Modify behavior across your entire application

Cratis Applications automatically registers several convention packs:

Applies your configured naming policy to all property names:

// Registered automatically with name: "Naming policy convention"
RegisterConventionAsPack(
conventionPackFilters,
NamingPolicyNameConvention.ConventionName,
new NamingPolicyNameConvention()
);

Ignores unknown properties during deserialization:

// Registered automatically with name: "Ignore extra elements convention"
RegisterConventionAsPack(
conventionPackFilters,
ConventionPacks.IgnoreExtraElements,
new IgnoreExtraElementsConvention(true)
);

To provide your own convention packs, implement ICanProvideMongoDBConventionPacks:

public interface ICanProvideMongoDBConventionPacks
{
IEnumerable<MongoDBConventionPackDefinition> Provide();
}
public class CustomConventionPackProvider : ICanProvideMongoDBConventionPacks
{
public IEnumerable<MongoDBConventionPackDefinition> Provide()
{
// Read-only conventions
yield return new MongoDBConventionPackDefinition(
"ReadOnly Properties",
new ConventionPack
{
new ReadOnlyPropertiesConvention()
}
);
// Enum string serialization
yield return new MongoDBConventionPackDefinition(
"Enum Conventions",
new ConventionPack
{
new EnumRepresentationConvention(BsonType.String)
}
);
// Custom discriminator handling
yield return new MongoDBConventionPackDefinition(
"Custom Discriminator",
new ConventionPack
{
new CustomDiscriminatorConvention(
CustomObjectDiscriminatorConvention.Instance,
GetTypesWithExistingDiscriminators()
)
}
);
}
private static IEnumerable<Type> GetTypesWithExistingDiscriminators()
{
// Return types that already have discriminator configuration
yield return typeof(BaseDocument);
yield return typeof(AuditableEntity);
}
}
public class DomainConventionPackProvider : ICanProvideMongoDBConventionPacks
{
public IEnumerable<MongoDBConventionPackDefinition> Provide()
{
// ID field conventions
yield return new MongoDBConventionPackDefinition(
"ID Field Conventions",
new ConventionPack
{
new NamedIdMemberConvention("Id", "id", "_id"),
new StringObjectIdIdGeneratorConvention()
}
);
// Ignore null values
yield return new MongoDBConventionPackDefinition(
"Ignore Null Values",
new ConventionPack
{
new IgnoreIfNullConvention(true)
}
);
// Custom date handling
yield return new MongoDBConventionPackDefinition(
"Date Conventions",
new ConventionPack
{
new DateTimeSerializationOptionsConvention(
DateTimeKind.Utc,
BsonType.DateTime
)
}
);
}
}

You can control which types convention packs apply to using filters.

Implement this interface to create custom filters:

public interface ICanFilterMongoDBConventionPacksForType
{
bool ShouldInclude(string conventionPackName, IConventionPack conventionPack, Type type);
}
public class DomainModelFilter : ICanFilterMongoDBConventionPacksForType
{
public bool ShouldInclude(string conventionPackName, IConventionPack conventionPack, Type type)
{
// Only apply naming conventions to domain models
if (conventionPackName == NamingPolicyNameConvention.ConventionName)
{
return type.Namespace?.Contains("Domain.Models") == true;
}
return true;
}
}
public class NoConventionsForDTOs : ICanFilterMongoDBConventionPacksForType
{
public bool ShouldInclude(string conventionPackName, IConventionPack conventionPack, Type type)
{
// Don't apply any conventions to DTOs
if (type.Name.EndsWith("DTO") || type.Name.EndsWith("Dto"))
{
return false;
}
return true;
}
}
public class LegacySystemFilter : ICanFilterMongoDBConventionPacksForType
{
public bool ShouldInclude(string conventionPackName, IConventionPack conventionPack, Type type)
{
// Don't apply naming conventions to legacy types
if (conventionPackName == NamingPolicyNameConvention.ConventionName &&
type.Namespace?.Contains("Legacy") == true)
{
return false;
}
return true;
}
}

For fine-grained control, use the IgnoreConventions attribute on specific types:

[IgnoreConventions]
public class RawDocument
{
public string _id { get; set; } // Keep exact field names
public string user_name { get; set; } // No naming policy applied
public object extra_data { get; set; } // No serialization conventions
}
[IgnoreConventions(NamingPolicyNameConvention.ConventionName)]
public class ExactFieldNames
{
public string UserName { get; set; } // Stored as "UserName"
public string EmailAddr { get; set; } // Stored as "EmailAddr"
}
[IgnoreConventions(ConventionPacks.IgnoreExtraElements)]
public class StrictDocument
{
public string Name { get; set; }
// Will throw exception if extra fields are present during deserialization
}
[IgnoreConventions(NamingPolicyNameConvention.ConventionName)]
[IgnoreConventions("Custom Enum Convention")]
public class SpecialDocument
{
public string PropertyName { get; set; } // No naming policy
public MyEnum Status { get; set; } // No enum convention
}

The framework defines constants for well-known convention pack names:

public static class ConventionPacks
{
public const string IgnoreExtraElements = "Ignore extra elements convention";
}
public class NamingPolicyNameConvention
{
public const string ConventionName = "Naming policy convention";
}
public class AuditConventionPackProvider : ICanProvideMongoDBConventionPacks
{
public IEnumerable<MongoDBConventionPackDefinition> Provide()
{
yield return new MongoDBConventionPackDefinition(
"Audit Fields",
new ConventionPack
{
new AuditFieldConvention()
}
);
}
}
public class AuditFieldConvention : ConventionBase, IMemberMapConvention
{
public void Apply(BsonMemberMap memberMap)
{
var memberName = memberMap.MemberName;
// Auto-configure audit fields
if (memberName == "CreatedAt" || memberName == "UpdatedAt")
{
memberMap.SetSerializer(new DateTimeOffsetSupportingBsonDateTimeSerializer());
if (memberName == "CreatedAt")
{
memberMap.SetIgnoreIfDefault(true);
}
}
// Configure user audit fields
if (memberName == "CreatedBy" || memberName == "UpdatedBy")
{
memberMap.SetIgnoreIfNull(true);
}
}
}
public class ValidationConventionPackProvider : ICanProvideMongoDBConventionPacks
{
public IEnumerable<MongoDBConventionPackDefinition> Provide()
{
yield return new MongoDBConventionPackDefinition(
"Required Fields",
new ConventionPack
{
new RequiredFieldConvention()
}
);
}
}
public class RequiredFieldConvention : ConventionBase, IMemberMapConvention
{
public void Apply(BsonMemberMap memberMap)
{
var memberInfo = memberMap.MemberInfo;
// Check for Required attribute
if (memberInfo.GetCustomAttribute<RequiredAttribute>() != null)
{
memberMap.SetIgnoreIfDefault(false);
memberMap.SetIgnoreIfNull(false);
}
}
}

Convention pack providers are automatically discovered during setup:

// This happens during UseCratisMongoDB()
var types = Types.Instance;
var providers = types.FindMultiple<ICanProvideMongoDBConventionPacks>();
var filters = types.FindMultiple<ICanFilterMongoDBConventionPacksForType>();
// Providers and filters are registered automatically

Convention packs are registered in the order they’re provided. If multiple conventions affect the same aspect, later conventions may override earlier ones.

For each convention pack, all filters are consulted:

static bool ShouldInclude(
IEnumerable<ICanFilterMongoDBConventionPacksForType> filters,
string conventionPackName,
IConventionPack conventionPack,
Type type)
{
// All filters must return true for the convention pack to be applied
return filters.All(filter =>
filter.ShouldInclude(conventionPackName, conventionPack, type));
}

Convention pack filters are called for every type, so keep them efficient:

// Good: Simple, fast checks
public bool ShouldInclude(string conventionPackName, IConventionPack conventionPack, Type type)
{
return !type.Name.EndsWith("DTO");
}
// Avoid: Expensive operations
public bool ShouldInclude(string conventionPackName, IConventionPack conventionPack, Type type)
{
return !type.GetCustomAttributes().Any(attr => attr is DTOAttribute);
}

Consider caching filter results for frequently-checked types:

public class CachedFilter : ICanFilterMongoDBConventionPacksForType
{
private static readonly ConcurrentDictionary<(string, Type), bool> _cache = new();
public bool ShouldInclude(string conventionPackName, IConventionPack conventionPack, Type type)
{
return _cache.GetOrAdd((conventionPackName, type), key =>
ComputeShouldInclude(key.Item1, conventionPack, key.Item2));
}
private bool ComputeShouldInclude(string conventionPackName, IConventionPack conventionPack, Type type)
{
// Expensive computation here
return ExpensiveCheck(type);
}
}

You can test your convention packs to ensure they work correctly:

[Test]
public void should_apply_naming_convention_to_domain_models()
{
// Arrange
var classMap = new BsonClassMap<DomainModel>();
classMap.AutoMap();
// Act
classMap.ApplyConventions();
// Assert
var memberMap = classMap.GetMemberMap(m => m.PropertyName);
memberMap.ElementName.ShouldEqual("propertyName"); // camelCase applied
}
[Test]
public void should_ignore_conventions_when_attribute_present()
{
// Arrange
var classMap = new BsonClassMap<IgnoredConventionsModel>();
classMap.AutoMap();
// Act
classMap.ApplyConventions();
// Assert
var memberMap = classMap.GetMemberMap(m => m.PropertyName);
memberMap.ElementName.ShouldEqual("PropertyName"); // No naming convention applied
}

Each convention should have a single responsibility:

// Good: Single purpose
public class DateTimeUtcConvention : IMemberMapConvention
{
public void Apply(BsonMemberMap memberMap)
{
if (memberMap.MemberType == typeof(DateTime))
{
memberMap.SetSerializer(new DateTimeSerializer(DateTimeKind.Utc));
}
}
}
// Avoid: Multiple concerns
public class MegaConvention : IMemberMapConvention
{
public void Apply(BsonMemberMap memberMap)
{
// Handles dates, strings, numbers, etc. - too complex
}
}

Convention pack names should clearly indicate their purpose:

// Good: Clear naming
"Audit Field Conventions"
"Required Field Validation"
"Legacy System Compatibility"
// Avoid: Vague naming
"Custom Convention"
"Special Rules"
"Fixes"

Make filter logic clear and well-documented:

public class ApiModelFilter : ICanFilterMongoDBConventionPacksForType
{
/// <summary>
/// Applies naming conventions only to API models (types ending with "ApiModel")
/// and excludes internal types from convention processing.
/// </summary>
public bool ShouldInclude(string conventionPackName, IConventionPack conventionPack, Type type)
{
if (conventionPackName == NamingPolicyNameConvention.ConventionName)
{
return type.Name.EndsWith("ApiModel") && !type.IsNotPublic;
}
return true;
}
}