There are several ways to handle schema changes in RavenDB. When I am talking about schema changes, I am talking about changing the format of documents in production. RavenDB doesn’t have a “schema”, of course, but if your previous version of the application had a Name property for customer, and your new version have FirstName and LastName, you need to have some way of handling that.
Please note that in this case I am explicitly talking about a rolling migration, not something that you need to do immediately.
We will start with the following code bases:
Version 1.0 | Version 2.0 |
public class Customer { public string Name {get;set;} public string Email {get;set;} public int NumberOfOrders {get;set;} } |
public class Customer { public string FirstName {get;set;} public string LastName {get;set;} public string CustomerEmail {get;set;} public bool PreferredCustomer {get;set;} } |
As I said, there are several approaches, depending on exactly what you are trying to do. Let us enumerate them in order.
Removing a property – NumberOfOrders
As you can see, NumberOfOrders was removed from v1 to v2. In this case, there is absolutely no action required of us. The next time that this customer will be loaded, the NumberOfOrders property will not be bound to anything, RavenDB will note that the document have changed (missing a property) and save it without the now invalid property. It is self cleaning .
Adding a property – PreferredCustomer
In this situation, what we have is a new property, and we need to provide a value for it. If there isn’t any value for the property in the stored json, it won’t be set, which means that the default value (or the one set in the constructor) will be the one actually set. Again, RavenDB will note that the document have changed, (have an extra property) and save it with the new property. It is self healing .
Modifying properties – Email –> CustomerEmail, Name –> FirstName, LastName
This is where things gets annoying. We can’t rely on the default behavior for resolving this. Luckily, we have the extension points to help us.
public class CustomerVersion1ToVersion2Converter : IDocumentConversionListener { public void EntityToDocument(object entity, RavenJObject document, RavenJObject metadata) { Customer c = entity as Customer; if (c == null) return; metadata["Customer-Schema-Version"] = 2; // preserve the old Name proeprty, for now. document["Name"] = c.FirstName + " " + c.LastName; document["Email"] = c.CustomerEmail; } public void DocumentToEntity(object entity, RavenJObject document, RavenJObject metadata) { Customer c = entity as Customer; if (c == null) return; if (metadata.Value<int>("Customer-Schema-Version") >= 2) return; c.FirstName = document.Value<string>("Name").Split().First(); c.LastName = document.Value<string>("Name").Split().Last(); c.CustomerEmail = document.Value<string>("Email"); } }
Using this approach, we can easily convert between the two version, including keeping the old schema in place in case we still need to be compatible with the old schema.
Pretty neat, isn’t it?