This shows example of building, using Mutations, bacthing in query and enabling transactions
slight modifications had to be made the initial Query class to enable the ability to call multiple queries in the same post message
Batching in GraphQL is handled differently then with batching in other batch calls like OData could do queries and edits at the same time if necessary, GraphQL wants the items to be all the same within one call.
You can link multiples queries together, but you cannot perform a query and then a mutation in the same call. This is because the way that mutations are resolved in series versus Queries which are resolved in parallel
To do multiple queries each operation must have a name and the URL has to be modified to achieve it to work so if we name two queries A and B the url would look like the following
https://localhost:7019/graphql?batchOperations=[A,B]
and the GraphQL query would look like
query A {
books {
addressId
title
price
}
}
query B {
addresses {
street
city
}
}
Batching helps alleviate the 'chattyness' of GraphQL and allows the queries to be gouped to simplify the work. The @export
command allows the queries to be linked.
If we wanted to have the results of one query filter another query we would use @export like so (assuming the same url as before in batching)
query A {
books {
addressId @export(as: "ids")
title
price
}
}
query B {
addressesById(id: $ids) {
street
city
}
}
The AddressIds are exported as a variable ids (lower case is important since it matches the routes) Note that is now variables are declared in GraphQL, by using the dollar sign More about variables when we reach mutations.
Mutations are the way that we edit, add, and delete data. Generally a mutation required the defined input, the return type. Hot Chocolate uses a specific naming convention in the input and return type variable naming. To add a book entry you need to define the input as xxxxPayLoad. For this example I am using the new C# feature of record to make it easier to define the input objects.
public record InputBookPayLoad(string Isbn, string Title, string Author, decimal Price, Guid AddressId, Guid PressId);
public async Task<Book> AddBook(InputBookPayLoad input)
{
var context = dbContextFactory.CreateDbContext();
var book = new Book
{
Id = Guid.NewGuid(),
Isbn = input.Isbn,
Title= input.Title,
Author = input.Author,
Price = input.Price,
AddressId = input.AddressId,
PressId = input.PressId
};
context.Books.Add(book);
await context.SaveChangesAsync();
return book;
}
In the example, we have a mutation called AddBook, with a input variable type called InputBookPayLoad. To call the mutation we call it like so:
mutation($newBook:InputBookPayLoadInput!) {
addBook(input: $newBook) {
id
}
}
--GraphQL variable
{
"newBook" : {
"isbn" : "3-8453-3507-4",
"title" : "A good sample Insert",
"author": "Elie Wiesel",
"addressId": "E8EC7B2D-0804-8FCB-2113-4238E16A36CA",
"pressId": "1E3E4158-3687-1175-F730-4FF545DF8EA9"
}
}
In this mutation we define the variable newBook and state it is type InputBookPayLoadInput. The ! means it is not a nullable type. Then using that predefined data we can use it in the addBook method and returnm the id when it is done inserting the data. as shown the newbook has to declare the variables as shown in the C# code and pass that as input. when the call is made, it will add a new book with that data and return the new Id. Updates can be done the same way where the update data is declared, the update method is then called and invoked in a similiar fashion and would also return the updated entity. With mutations we added the .AddDefaultTransactionScopeHandler() in the Program.cs so we are putting those items in a transaction scope. Mutations can be executed like queries and can be batched as well. When mutations are batched they are not in parallel, but in serial so they can depend on the results of a previous insert or update, like inserting a child table and exporting that id into a variable and feeing that into the parent insert that needed that child id.
To enable security on Mutations, the [Authorize] attribute must be applied to the mutation itself. This will then trigger the IAuthorizeHander that we have implemented associated to the Query. The check will occur at the Mutation level and not the column level. All incoming data would be checked ay that point.
The way that GraphQL requires columns to be explicitly called generates some issues when it comes to add and edit capability. If the user doesn't have access to a field in a table. There are a couple of ways to handle the issue. One possible solution would be to modify the tables so that they are aligned to their permissions. View can be built for items for queries if the application has permissions to multiple parts of the data. This also assumes that all permissions are the same. If the permissions differ then we may have to implement something like a patch that can inspect the json data coming in and ensure that the fields in the json data match the columns allowed for the operation.