-
-
Notifications
You must be signed in to change notification settings - Fork 304
Reverse Engineering
Reverse engineering requires Visual Studio 2022 (amd64) and the .NET 6 x64 (for EF Core 6 unsupported) or .NET 8 x64 (for EF Core 8 and 9) runtime.
If you do not use Visual Studio, a cross platform dotnet tool for reverse engineering is available, more information here
EF Core allows you to use the Scaffold-DbContext
or the dotnet ef dbcontext scaffold
commands to reverse engineer a database and create entity and DbContext classes for your database tables.
This approach has several disadvantages, however:
- You must add design time components to your project, that you can easily end up deploying to production, thus increasing your app footprint (size) for no reason
- It is not easy to specify which tables to scaffold, and there is no easy way to generate only entity or DbContext classes.
In addition, the reverse engineering offered in EF Core Power Tools has a many additional options, so you can tailor the generated code.
Select a project in Solution Explorer.
Right click the project and select Add, New Item, Data - (or Ctrl+Shift+A) and pick EF Core Database First Wizard
.
Or right click the project and select EF Core Power Tools
- Reverse Engineer
.
Connect to SQL Server 2012 or later, Azure SQL DB (also MFA login), SQLite, PostgreSQL, MySQL, Firebird, Oracle or Snowflake, a SQL Database project or a .dacpac file:
You can remove unwanted database connections from this dropdown by clicking the ➖ button.
To use SQL Server/Azure SQL DB, choose the SQL Server provider in the Add dialog.
To use another DDEX provider than the default SQL Server provider, do as follows: Choose Add => Add Database Connection => Change button, and then pick either PostgreSQL Database
or SQLite
or Firebird
or Oracle ODP.NET
To use PostgreSQL or SQLite install the EF Core Power Pack which includes these DDEX providers: VisualStudio.Data.Sqlite and Npgsql PostgreSQL Integration
To use Oracle, install the Oracle Visual Studio Tools, and make sure to use the ODP.NET Managed driver when connecting to the database.
To use FireBird, install the BlackbirdSql DDEX and SqlEditor for Firebird.
To use MySQL or Snowflake*, use a custom connection as described below.
To use a .dacpac, the tool will list any .dacpac files and Database projects found in your solution. You can also choose an additional .dacpac file from disk.
You can also target another EF Core version from this dialog.
You can use the drop-down button to add custom data sources, for MySQL and Snowflake, that do not have a proper or current DDEX provider. The connections created with this dialog are stored securely in Credential Manager
You can then choose which tables, views (and even SQL Server stored procedures and functions) to include in your model. If you are adding or removing items, make sure to select all the items that you want to be part of your DbContext in this dialog.
You can also exclude individual columns via the checkboxes in this dialog CLI: excludedColumns
, and rename tables/views and their columns by pressing the F2 key. For an example of mapping enum values to a column by excluding the base column, see this.
You can also exclude indexes (SQL Server only), for example to work around this EF Core bug - add a list of ExcludedIndexes strings to your table object in efpt.config.json
. CLI: excludedIndexes
Sample:
{
"Name": "[dbo].[my_table]",
"ObjectType": 0,
"ExcludedIndexes": [
"IX_name_of_the_index"
]
}
Notice that any changes you make in this dialog will be persisted in .config.json files in your project.
In the options dialog, you can choose
DbContext class and namespaces
- the name of the DbContext class
CLI: names/dbcontext-name
code sample - the namespace to be used for the generated code
CLI: names/dbcontext-namespace
code sample - optional subfolder path for entity classes in the current project (by default Models)
CLI: file-layout/output-path
code sample
See this example to get an idea of the customization options.
To generate entity classes in a project folder, and the DbContext in the root folder, see the tip here
To use completely different namespaces for entities and DbContext, see this sample
What to generate
- whether to generate both DbContext and entity classes, or just one or the other
CLI: code-generation/type
Naming of entity classes/DbSet
- pluralize or singularize generated names (entity class names singular and DbSet names plural)
CLI: code-generation/use-inflector
code sample - use table and column names from the database
CLI: code-generation/use-database-names
code sample
Other options
- to use DataAnnotation attributes rather than the fluent API (as much as possible)
CLI: code-generation/use-data-annotations
code sample - to customize code using templates
CLI: code-generation/use-t4
(see below) - to not include the current connection string in the generated code
CLI: code-generation/enable-on-configuring
code sample - optionally add the relevant EF Core packages to the project code sample
- Use many to many entity - preserve a many to many entity instead of skipping it - see #1148
CLI: code-generation/use-many-to-many-entity
- Use nullable reference types
CLI: code-generation/use-nullable-reference-types
- Remove SQL default from bool columns
CLI: code-generation/remove-defaultsql-from-bool-properties
(to avoid them being bool? - see #160 - Remove all navigations from entity classes (experimental)
CLI: code-generation/use-no-navigations-preview
- Remove Default DbContext constructor (only generated if "Inlcude connection string" is selected in EF Core 7+) - see #995
- Always include all objects, useful for the Refresh context menu item
- Custom sub-namespace for entity classes (by default path will be used) (preview)
CLI: names/model-namespace
code sample - Optional subfolder path for DbContext class in the current project, or even a full path to anywhere on your disk
CLI: file-layout/output-dbcontext-path
code sample - Custom sub-namespace for DbContext class (by default path will be used)
CLI: names/dbcontext-namespace
code sample - Split DbContext into IEntityConfiguration classes in a Configurations folder below the DbContext (preview)
CLI: file-layout/split-dbcontext-preview
OBSOLETE: the supported way of doing this will be via the new T4 template option, see below - Use schema folders for generated entity classes (experimental)
CLI: file-layout/use-schema-folders-preview
- Use schema namespaces for generated entity classes (experimental)
CLI: file-layout/use-schema-namespaces-preview
- Map spatial types with SQL Server, PostgreSQL and MySQL
CLI: type-mappings/use-spatial
More info - Map hierarchyid with SQL Server
CLI: type-mappings/use-HierarchyId
- Map DateOnly (date) and TimeOnly (time) with SQL Server
CLI: type-mappings/use-DateOnly-TimeOnly
- Map Noda Time types with PostgreSQL, SQL Server and SQLite
CLI: type-mappings/use-NodaTime
- Use EF6 Pluralizer - by default the tool uses the Humanizer package for pluralization
CLI: code-generation/use-legacy-inflector
Click OK, and C# code with the desired options will be generated in the current project.
After completed code generation, a file named efpt.config.json
will be added to the project - this will allow the tool to load the same settings next time you run reverse engineering. You can right click the file, and choose to either re-generate the code, or edit the file.
To always include all objects (including newly added) during Refresh, you can use the advanced option Always include all objects
.
For CLI, there is an option to prevent addition of objects after initial discovery: CLI: code-generation/refresh-object-lists
If you are adding or removing tables, leave any additional tables selected on subsequent runs of the tool, or all unused files will be removed. If you explicitly want to keep unselected files, remove the first line with the "autogenerated" comment from each (or rename the file). This is an example of a tool to take an existing classic Entity Framework 6 EDMX file and generate config files from it.
If you wish to do some post processing after code generation, you can place a .cmd file named efpt.postrun.cmd
in the same folder as the config file, and it will be executed each time you run code generation. It is suggested to use ANSI encoding for the file.
All the generated code files are generated a partial classes, which will allow you to extend both the entity classes and the generated DbContext class, with overrides of existing virtual methods on the DbContext and additional, non-mapped properties on you entity classes. In addition, you can customize the generated model further, by implementing the partial OnModelCreatingPartial method. An example of how this can be used is available here. You can also add additional DataAnnotations to the entity classes in a "buddy" metadata class, as described here
The pluralizer will make names of DbSets plural, and names of Entity classes singular. The default pluralizer uses Humanizer for pluralization. It is possible to exclude words (for example Status and Data) from the pluralizer by modifying the efpt.config.json
like this:
"UncountableWords": [
"Status",
"Data"
],
The tool can map SQL Server stored procedures, scalar and table valued functions, by selecting them from the list of objects to scaffold. Support for these database object types in .dacpac's is currently in preview.
The stored procedure result mapping is limited to what SET FMTONLY can handle, so not all stored procedures are supported. There is a global option to use sp_describe_first_result_set instead, if that works better for you.
You can also enforce to use sp_describe_first_result_set instead of SET FMTONLY for individual objects (stored procedures or functions), by manually updating the efpt.config.json file:
{
"Name": "[dbo].[MultiSet]",
"ObjectType": 1,
"UseLegacyResultSetDiscovery": true
},
As a workaround, you can add the missing properties in a partial class, that extends the generated empty result set class that is created if a single stored procedure result cannot be discovered.
If you know that your stored procedure does not return a result set, but an empty result set class is still generated, you can add this line to the top of your stored procedure to prevent the empty result set class from being generated:
IF NOT EXISTS(SELECT SESSIONPROPERTY('fmtonly'))
BEGIN
SET FMTONLY OFF;
RETURN;
END
If you use #temp tables in your stored procedure, and that prevents result set discovery, you can use the tip here to "help" expose the result set shape.
If result set discovery still fails, please let me know.
It is also possible to map the result of a stored procedure to a class (DbSet) (or any class with EF Core 8 or later) in your DbContext as follows:
{
"Name": "[dbo].[Top 10 Customers]",
"ObjectType": 1,
"MappedType": "Customer"
},
Support for mapping multiple result sets per procedure is currently in preview and you must opt-in via the global Visual Studio option, see below. This feature depends on Dapper.
Sample usage of the generated code:
using (var db = new NorthwindContext())
{
var orders = await db.Procedures.CustOrderHistAsync("ALFKI");
foreach (var order in orders)
{
Console.WriteLine($"{order.ProductName}: {order.Total}");
}
var outOverallCount = new OutputParameter<int?>();
var customers = await db.Procedures.SP_GET_TOP_IDSAsync(10, outOverallCount);
Console.WriteLine($"Db contains {outOverallCount.Value} Customers.");
foreach (var customer in customers)
{
Console.WriteLine(customer.CustomerId);
}
// Use scalar functions inline in queries:
var result = db.Orders.Where(o => o.Reference == NorthWindContext.GetReference(o.OrderName)).ToList();
var tvfTest = db.ProductsWithMinimumInStock(5).OrderBy(p => p.ProductName).ToList();
}
To generate synchronous procedures signatures, you can set this option:
"UseAsyncStoredProcedureCalls": false
(not available in the CLI)
By default, to avoid runtime warnings, all decimal properties in the result set classes are decorated with a TypeName attribute, this can be disabled:
"UseDecimalDataAnnotationForSprocResult": false
CLI: code-generation/use-decimal-data-annotation-for-sproc-results
To use better names than directly from the database objects, you can set this option:
"UseDatabaseNamesForRoutines": false
CLI: code-generation/use-database-names-for-routines
The tool can derive a model from a .dacpac file, this works very well for tables, but for views and stored procedure results, there may be some missing properties, due to for example computed columns. To work around this, publish the .dacpac to a live database and generate the code from the live database, or manually add the missing properties in a partial class.
The tool includes other features, that makes it easy for you to use a .dacpac file in your workflow:
-
If you have an empty C# project in your solution, the tool can launch the reverse engineeering wizard from the context menu of a Database Project (.sqlproj) -
Create EF Core DbContext...
-
To create a database project from your EF Core model, you can do the following:
- Right click the project with your DbContext, and select the
View DbContext DDL SQL
menu item - This will add a new .sql file to your DbContext project
- Add a new Database Project (.sqlproj)
- Right click, and select
Import
andScript (*.sql)
and point to the .sql script generated above.
- Right click the project with your DbContext, and select the
If you choose "Customize code using templates" and choose C# - T4
the tool will add a CodeTemplates folder to your project, that contains an EFCore folder with DbContext.t4 and EntityType.t4 templates. The tool uses them to generate customizable context and entity classes.
This only works for EF Core 8 and later projects.
You can use this extension for a better T4 editing experience.
You can specify a custom location for your CodeTemplates
folder in efpt.config.json
like this (a relative path to the current project or a full path):
"UseT4": true,
"T4TemplatePath": "..\\..\\",
If you have multiple template folders in your project with the same directory structure as mentioned above, they will all be processed, for example:
/CodeTemplates/EFCore
/Views/CodeTemplates/EFCore
/Services/CodeTemplates/EFCore
For an example of adding Enum mappings, using comments in other places and replacing property names, see the sample templates here
There are links to many other customization samples in this issue
To split your DbContext into EntityTypeConfiguration classes, use the C# - T4 (DbContext split)
option. This will create Entity configuration files in a Configurations folder under your DbContext class.
You can also choose to generate "POCO" classes for use with for example micro ORMs, by picking the C# - T4 (POCO)
option.
If you have old templates, your will get this warning:
If you choose "Customize code using templates" and pick the Handlebars options, the tool will add a CodeTemplates folder to your project, that contains CSharpDbContext and CSharpEntityType folders with Handlebars templates, and tool will use them to generate customizable context and entity classes.
You can supply your own code templates, by placing a CodeTemplates.zip file at the root of your project.
For example, there is a Properties.hbs template in CSharpEntityType/Partials that you can customize to use List instead of ICollection:
public List<{{nav-property-type}}> {{nav-property-name}} { get; set; }
Add support for Lazy Loading using Handlebars
You can add support for Lazy Loading using Handlebars templates (which is a new feature in EF Core 2.1, read more here - as you can see, this feature requires that navigation properties to be "virtual" (can be overridden).
In order to add support for Lazy Loading to the generated code, modify Properties.hbs template in CSharpEntityType/Partials like this:
{{spaces 8}}public virtual ICollection<{{nav-property-type}}> {{nav-property-name}} { get; set; }
{{spaces 8}}public virtual {{nav-property-type}} {{nav-property-name}} { get; set; }
Likewise, you can also customize other aspects of the class, such as imports or the constructor, or you can inherit from a particular base class. This feature is based on the EntityFrameworkCore.Scaffolding.Handlebars project from Tony Sneed
To customize names of tables, stored procedures, functions and columns, you can place a file named efpt.renaming.json at the root of your project.
(This file will be created for you if you use the UI based renaming feature)
You can also have a renaming file per DbContext in a project, if your rename the file following the convention for multiple DbContexts in the same project described below. Or you can have multiple renaming files in folders with a corresponding efpt.config.json file.
efpt.renaming.json
sample file:
[
{
"ColumnPatternReplaceWith": "",
"ColumnRegexPattern": "^(tbl)",
"SchemaName": "dbo",
"TablePatternReplaceWith": "",
"TableRegexPattern": "^(tbl)",
"Tables": [
{
"Columns": [],
"Name": "tblSIMCard",
"NewName": "SIMCard"
}
],
"UseSchemaName": false
}
]
To preserve casing when using RegEx based renaming, add the following to efpt.config.json
"PreserveCasingWithRegex": true,
For an example of dealing with duplicate table names in different schemas, see this
For an example demonstrating how to override casing and being compatible with some Entity Framework 6 conventions, see this
For an example of a programmatic approach to create the efpt.renaming.json file, see this
For an example of a programmatic approach to create the efpt.renaming.json and table list from an EDMX (EF 6 Classic) file, see this
For an example of a SQL based approach to create the efpt.renaming.json file see this
You can also do custom replacement of names using regular expressions, see this and this
You can force EF Core 8 to use the legacy, prefix based navigation naming with:
"UsePrefixNavigationNaming": true,
CLI: code-generation/use-prefix-navigation-naming
To generate code for multiple DbContexts in the same project, make sure to use a separate project folder for each DbContext and set of Entity classes, so the files do not overlap and get overwritten.
If you have multiple DbContexts in the same project, the tool supports multiple efpt.config.json
files in the project root folder, as long as they follow the "efpt.*.config.json" naming pattern. So you could have two .config.json files named for example efpt.config.json
and efpt.Chinook.config.json
.
Alternatively, you can have multiple files named efpt.config.json
in individual folders in the project.
You can also have a renaming file (efpt.renaming.json
) per DbContext in a project, if your rename the file following the same convention (or lives in the same folder as the corresponding efpt.config.json
file.
Once you have multiple efpt.*.config.json
files in your project, you will be prompted to select the file you want to use when you start the reverse engineer process. You can also right click the file and launch the reverse engineer tool directly from the context menu.
In addition to the options mentioned above, there are a couple of "global" options available, that affects the behaviour of the reverse engineering process. Reach these via the Options
menu item or via Tools
, Options
.
- Open Generated DbContext: Set this to false if you do not want the genrated DbContext to open after reverse engineering.
- Run Cleanup of Obsolete files: Set this to false if you do not want to have the tool to remove files that are no longer in use.
CLI: code-generation/soft-delete-obsolete-files
Note that only files with this exact content on first line are removed:
// <auto-generated> This file has been auto generated by EF Core Power Tools. </auto-generated>
- Discover multiple result sets from SQL stored procedures (preview)
CLI: code-generation/discover-multiple-stored-procedure-resultsets-preview
- Use alternate result set discovery - use sp_describe_first_result_set to retrieve stored procedure result sets
CLI: code-generation/use-alternate-stored-procedure-resultset-discovery
- Merge .dacpac files that depend on each other into a single file
CLI: code-generation/merge-dacpacs