Here are some steps to follow to create a new .NET 5 + EntityFramework Core 5 backend, using Breeze to handle the data management.
Later, we'll work on a client that talks to our server.
We will assume that you've already created the database and the server directory, following the steps in the STEPS document.
For the server, we'll start with an empty directory, and implement a Breeze API that our client can use to query and update data in the database. Along the way we will:
- Create a .NET 5 solution
- Create C# entity classes from the database using EF
- Create an API for interacting with the entity model
- Create metadata from the entity model
Here we create the Visual Studio solution and the backend projects. For this demo, we will have two projects: the model project, which implements the data model, and the server project, which implements the API.
-
Create the server project.
- In Visual Studio 2019, select File / New / Project...
- Choose project type Web
- Choose ASP.NET Core Empty
- Set the Project Name to NorthwindServer
- Set the Location to the server directory that you created earlier
- Set the Solution Name to NorthwindNet5
- Make sure "Place solution and project is the same directory" is NOT checked
- Click Next, which takes you to the next dialog
- Choose .NET 5.0 target
- Uncheck "Configure for https" (for simplicity)
- Click Create.
-
Create the model project
- In the Solution Explorer, right-click on the solution and choose Add / New Project... (or, from the top menu, select File / Add / New Project... )
- Choose project type Library
- Choose Class Library
- Click Next
- Set the project Name to NorthwindModel
- Set the Location to the NorwindNet5 directory that was created for the solution
- Click Next
- Choose target framework .NET 5.0
- Click Create
-
Add a reference to the model project
- Right-click on the NorthwindServer project and select Add / Project Reference...
- Choose Projects / Solution and check NorthwindModel
- Click OK
- Set NorthwindServer as the startup project if it is not already.
-
Add Nuget packages to the server project. These support the data API that we will create using Breeze.
- Select Tools / Nuget Package Manager / Package Manager Console
- Set Default project to NorthwindServer
Install-Package Breeze.AspNetCore.NetCore
Install-Package Breeze.Persistence.EFCore
Install-Package Microsoft.EntityFrameworkCore.Tools
Install-Package Microsoft.AspNetCore.Mvc.NewtonsoftJson
-
Add Nuget packages to the model project. These support using Entity Framework for data access and data model creation.
- Select Tools / Nuget Package Manager / Package Manager Console
- Set Default project to NorthwindModel
Install-Package Microsoft.EntityFrameworkCore.SqlServer
We will create the data model classes from the database schema. For this, we need a connection string for connecting to the database. The commands below assume a local SQL Express instance; if you are using a separate server, you will need to change the Data Source and the security information in the connection string.
If you haven't created the database yet, go back to the STEPS document and see the Create the database section.
- Select Tools / Nuget Package Manager / Package Manager Console
- Set Default project to NorthwindModel
Scaffold-DbContext "Data Source=.;Initial Catalog=Northwind;Integrated Security=True;MultipleActiveResultSets=True" Microsoft.EntityFrameworkCore.SqlServer -OutputDir Models -Force -NoOnConfiguring
- (The command above should all be on one line)
Now you should have a Models folder in the NorthwindModel project, which contains classes for each of the Northwind database tables, and a NorthwindContext.cs
that contains the EF DbContext class for accessing the database.
If you change the database schema, you can re-run the Scaffold-DbContext
command to re-build the model classes.
The PersistenceManager is a Breeze class that wraps the DbContext to provide Breeze data management. We will create it in the NorthwindServer project
- Right-click on the NorthwindServer project and select Add / Class...
- Name the class
NorthwindPersistenceManager.cs
and click Add. This will create the class file. - In the class file, remove the default
using
statements and replace them with
using Breeze.Persistence.EFCore;
using NorthwindModel.Models;
- Make the class extend EFPersistenceManager
public class NorthwindPersistenceManager : EFPersistenceManager<NorthwindContext>
- Add a constructor to create it from our DbContext
public NorthwindPersistenceManager(NorthwindContext dbContext) : base(dbContext) {}
We will add a controller class that will be the interface between the client and server.
- Right-click on the NorthwindServer project and select Add / New Item...
- Choose type API Controller - Empty
- Set the Name to
BreezeController.cs
- Click Add. This will create the controller class with some default methods.
- Delete all the existing methods in the BreezeController class
- Update the class attributes so it can perform Breeze queries:
using Breeze.AspNetCore;
using Breeze.Persistence;
using Microsoft.AspNetCore.Mvc;
using Newtonsoft.Json.Linq;
using NorthwindModel.Models;
using System.Linq;
...
[Route("api/[controller]/[action]")]
[BreezeQueryFilter]
public class BreezeController : ControllerBase
Note that
- The
Route
attribute specifies the[action]
as part of the path. - We've added the
[BreezeQueryFilter]
attribute
Add a new persistenceManager
field to the BreezeController
class, and add a constructor that takes a NorthwindContext
and sets the persistenceManager
field. This will be called by dependency injection.
private NorthwindPersistenceManager persistenceManager;
public BreezeController(NorthwindContext dbContext)
{
persistenceManager = new NorthwindPersistenceManager(dbContext);
}
Add a HttpGet method returning IQueryable<>
for each of the Customer
, Order
, Product
, and Supplier
types in the data model. We won't do one for OrderItem
because we will only query those with an Order
[HttpGet]
public IQueryable<Customer> Customers()
{
return persistenceManager.Context.Customers;
}
[HttpGet]
public IQueryable<Order> Orders()
{
return persistenceManager.Context.Orders;
}
[HttpGet]
public IQueryable<Product> Products()
{
return persistenceManager.Context.Products;
}
[HttpGet]
public IQueryable<Supplier> Suppliers()
{
return persistenceManager.Context.Suppliers;
}
This HttpPost method will be called by the client to create/update/delete entities.
[HttpPost]
public ActionResult<SaveResult> SaveChanges([FromBody] JObject saveBundle)
{
return persistenceManager.SaveChanges(saveBundle);
}
Now make sure the solution compiles. Don't run it yet, there are a few more things to add.
Add the connection string to the appsettings.json
file
"ConnectionStrings": {
"Northwind": "Data Source=.;Initial Catalog=Northwind;Integrated Security=True;MultipleActiveResultSets=True"
},
Change the default port numbers to 4000 in all the applicationUrl
settings inside Properties/launchSettings.json
.
"applicationUrl": "http://localhost:4000",
Edit the Startup.cs file in the NorthwindServer project.
First we add the needed using
statements:
using Breeze.AspNetCore;
using Breeze.Core;
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using Newtonsoft.Json.Serialization;
using NorthwindModel.Models;
We will need access to the appsettings.json
configuration, so we will need to add a constructor to accept an IConfiguration instance:
private IConfiguration configuration;
public Startup(IConfiguration configuration)
{
this.configuration = configuration;
}
In the ConfigureServices
method, we need to
- Enable MVC, so our
BreezeController
class can be used to handle requests - Set JSON serialization options so the client-side Breeze can send and receive entities
- Add an exception filter, so errors are communicated to the Breeze client
- Add the DbContext to dependency injection, so our BreezeController can receive it
public void ConfigureServices(IServiceCollection services)
{
var mvcBuilder = services.AddMvc();
services.AddControllers().AddNewtonsoftJson(opt => {
// Set Breeze defaults for entity serialization
var ss = JsonSerializationFns.UpdateWithDefaults(opt.SerializerSettings);
if (ss.ContractResolver is DefaultContractResolver resolver)
{
resolver.NamingStrategy = null; // remove json camelCasing; names are converted on the client.
}
ss.Formatting = Newtonsoft.Json.Formatting.Indented; // format JSON for debugging
});
// Add Breeze exception filter to send errors back to the client
mvcBuilder.AddMvcOptions(o => { o.Filters.Add(new GlobalExceptionFilter()); });
// Add DbContext using connection string
var connectionString = configuration.GetConnectionString("Northwind");
services.AddDbContext<NorthwindContext>(options => options.UseSqlServer(connectionString));
}
In the Configure
method, add CORS, and add MapControllers
to the app.UseEndpoints()
statement. Note
that the order of the statements is important in this method.
public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
if (env.IsDevelopment())
{
app.UseDeveloperExceptionPage();
}
app.UseRouting();
// Allow any host - development only!
app.UseCors(builder => builder
.AllowAnyHeader()
.AllowAnyMethod()
.SetIsOriginAllowed((host) => true)
.AllowCredentials()
);
app.UseEndpoints(endpoints =>
{
endpoints.MapControllers();
endpoints.MapGet("/", async context =>
{
await context.Response.WriteAsync("Hello World!");
});
});
}
To make client-side development easier, we are going to generate Breeze metadata from our entity model. This metadata represents the structure of our entities, and will be shared between the server and the client. Later we will use the metadata to generate TypeScript classes representing our entity types.
When we run our .NET 5 application from the command line, it starts in the Main
method of Program.cs
.
So we will change the Main
method to produce metadata output, instead of starting the server, when we give it a "metadata" command-line parameter:
public static void Main(string[] args)
{
if (args.Length > 0 && args[0].Contains("metadata"))
{
// Generate metadata and exit
var dbContext = new NorthwindModel.Models.NorthwindContext();
var persistenceManager = new NorthwindPersistenceManager(dbContext);
var metadata = persistenceManager.Metadata();
Console.Out.WriteLine(metadata);
}
else
{
// Start web server
CreateHostBuilder(args).Build().Run();
}
}
Compile the solution, then open a command prompt in the NorthwindNet5
directory. Then run:
dotnet NorthwindServer\bin\Debug\net5.0\NorthwindServer.dll metadata > ..\metadata.json
NOTE: Switch separators to /
for OSX or Linux.
That will create a metadata.json
file in the parent server
directory. The file
contains a big blob of JSON that represents the type information that EF knows about our data model.:
{"structuralTypes":[{"shortName":"Customer","namespace":"NorthwindModel.Models","autoGeneratedKeyType":"Identity","defaultResourceName":"Customer",...
We'll use the metadata.json
file later when we create the web client.
Compile and run the solution. It should open a browser and attempt to open a page on http://localhost:4000 ,
which should return "Hello World" due to endpoints.MapGet
in Startup.cs
.
Now let's test a Breeze query. Change the URL to http://localhost:4000/api/breeze/customers
Now you should get a JSON result containing all the rows from the Customers table in the Northwind database:
[
{
"$id": "1",
"$type": "NorthwindModel.Models.Customer, NorthwindModel",
"Id": 1,
"FirstName": "Maria",
"LastName": "Anders",
"City": "Berlin",
"Country": "Germany",
"Phone": "030-0074321",
"Orders": []
},
{
"$id": "2",
"$type": "NorthwindModel.Models.Customer, NorthwindModel",
"Id": 2,
"FirstName": "Ana",
"LastName": "Trujillo",
"City": "México D.F.",
"Country": "Mexico",
"Phone": "(5) 555-4729",
"Orders": []
}...
]
If you don't get the result above, review the previous steps and try to see where you may have gone wrong.
Note that each Customer entity has an $id
and $type
property.
-
The
$id
is for resolving circular references - each entity appear only once in the JSON tree, and if the entity is referenced again in the tree, it will be replaced by a$ref
that references the$id
. -
The
$type
identifies the entity type. The Breeze client will use this to determine what type of entity to create on the client as a result of the query.
If you don't see the $id
and $type
, go back to the Startup.cs
file and make sure the settings are correct.
If everything looks good on the server, we are ready to work on the client application.
Go to the README in the client directory to start your client app.