Skip to content
This repository has been archived by the owner on Nov 2, 2022. It is now read-only.
/ QueryMutator Public archive

Queryable and Enumerable extensions for automapping objects and mapping multiple expressions into one.

License

Notifications You must be signed in to change notification settings

yugabe/QueryMutator

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

21 Commits
 
 
 
 
 
 
 
 

Repository files navigation

Note

QueryMutator had a great run. I'm very proud of how it turned out and the small impact it had on the few people it did have an impact on.

But (obviously) the repo is not maintained anymore, for a few reasons. In hindsight, using automapping was a great way to reduce the quantity of code on a few of our projects, and it was more performant in many cases than the custom user code would've been. But it made a few bugs difficult to debug, as most things that happened were kind of a black magic. It was great when it worked, and a bit painful when it didn't (it could've been a typo in user code, like the naming of a property, but still). Then of course, EF Core improved a lot, and made many of the benefits of QueryMutator moot. Then, EF Core improved even more, and QueryMutator stopped working in many cases. So I don't advise people to use QueryMutator in new projects.

The code itself is a time capsule and very dear to me. Expression mutation is still a very interesting topic to cover, and I'll come back to it sometime in the future. Until then, keep on mutatin'!


QueryMutator

Queryable and Enumerable extensions for automapping objects and mapping multiple expressions into one.

What's new in 1.3.1?

  • Finally, QueryMutator is out of beta! Most scenarios should work as expected as long as you use the mapping behavior as intended. Even though QueryMutator is out of beta, be sure to thoroughly test your application. If you find a bug or would like a feature, don't hesitate to open an issue here on GitHub, or review the code and submit a pull request. Be sure to check it out and keep an eye out for QueryMutator v2!
  • Fixed bugs occuring in certain versions of .NET Standard and EF Core where AsQueryable was not implemented.

What's new in 1.3.0-beta?

  • There is now an option to choose how collections will be mapped based on a certain phenomena (a Linq-to-Entities Expression should not check whether a collection is null, but an object-to-object mapping should).
  • QueryMutator now targets .NET Standard 1.0, so it is compatible with .NET Framework 4.5 and up, .NET Core 1.0 and up, other .NET Standard libraries 1.0 and up, as well as any other .NET Standard-capable runtime. Yay!

What's new in 1.2.0-beta?

  • Easily make a shallow copy of an object.
var dog = new Dog { Name = "Bloki" };
var cloneDog = dog.Clone();

This is only an easy shorthand for the appropriate MapTo method.

What's new in 1.1.0-beta?

  • Now you can map a single object to another!
var dog = new Dog { Name = "Bloki" };
var cloneDog = dog.MapTo<Dog, Dog>();
var cat = dog.MapTo<Dog, Cat>();

Keep in mind that this uses a safe extension, so calling MapTo on an object won't cause a NullReferenceException if the object is null.

What's it for?

It's an object mapper for any .NET runtime. If you ever tried to map objects from an IQueryable to a custom type of your own, you probably saw and wrote lots of code like this:

List<UserDto> users;
using (var ctx = new MyContext())
{
    users = ctx.Users.Select(u => new UserDto
    {
        Id = u.Id,
        FirstName = u.FirstName,
        // and so on
    }).ToList();
}

Using QueryMutator makes the above query a lot easier. Basically, you can register type-pairs which you would like to convert between, or you can let it lazy load and generate the mapping at runtime.

List<UserDto> users;
using (var ctx = new MyContext())
{
    users = ctx.Users.MapToList<User, UserDto>();
}

QueryMutator matches properties by name, and tries to match the object types against each other. It works by recursively transversing the type graphs. Do not try to mutate recursive data structures, as it might result in unwanted behavior! First and foremost QueryMutator should be used to map and mutate between an entity type and a DTO to reduce lots of boilerplate code to a single method call.

Why not just use AutoMapper instead?

QueryMutator has a neat little side effect: as the query is built at runtime (even if you don't register the expressions you want to use for mapping), you can merge the automatically generated or custom registered mapping with another expression to effectively extend your object or remap a property at runtime.

List<UserWithFriendsCountDto> users;
using (var ctx = new MyContext())
{
    users = ctx.Users.MapToList<User, UserWithFriendsCountDto>(u => new UserWithFriendsCountDto 
    {
        FriendsCount = u.Friends.Count()
		// Nothing else needs to be mapped!
    });
}

The above code will automap the properties of User to the DTO type, and add another property to the mapped object. This will all happen as the query is executed, so no unneeded round trips to the database!

Great! What else?

QueryMutator lives in the System.Linq namespace. It might seem a bit overkill, but actually it can make your life easier, as wherever you might find a Select extension method, you'll find QueryMutator's methods:

  • IQueryable MapTo extension: automatically map the source to the target or provide a merge expression to merge the output with another object. At runtime, only one instantiation will occur.
  • IQueryable MapToList extension: same as above, but you don't have to call ToList() on it to pull it to memory.
  • IEnumerable MapTo and MapToList extensions: same as with IQueryable, but compiles the available expression at runtime so that you can use QueryMutator in every situation.
  • QueryMutator.GetMapping: retrieve or lazy load a mapping between two types.
  • QueryMutator.RegisterMapping: register an automatically mapped expression or provide a custom one to use later. You can even provide that types as parameters instead of generic type arguments, as to make it easier to automate generating mappings in your own code using Reflection.
  • QueryMutator.CurrentConfiguration: the configuration object allows you to set options on how QueryMutator generates expressions, when should it throw an exception, etc. The extension methods use the CurrentConfiguration static property, but you can switch it out anytime at runtime.

I'm sold. Where can I get it?

You can get the source here at GitHub, feel free to send a pull request if you'd like to contribute. The project is open source and free to use.
You can download the NuGet package from NuGet.

About

Queryable and Enumerable extensions for automapping objects and mapping multiple expressions into one.

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Contributors 4

  •  
  •  
  •  
  •  

Languages