Efcore: Reverse Engineer: Better support for namespaces

Created on 6 Dec 2015  路  35Comments  路  Source: dotnet/efcore

Table schemas don't appear to make it into scaffolded entity code in any way; If I have two tables named Users in schemas A and B, the reverse engineering process will simply created Users and Users1 inside the reverse-engineering namespace. This don't seem to be a very satisfactory result.

I think the expected result is probably to map non-default database schemas (i.e. non-dbo in SqlServer) to namespace underneath the reveng namespace. So, assuming we're reverse-engineering to namespace Models, dbo.Users would get scaffolded to Users.cs, while schema2.Users would get scaffolded to Models.Schema2.Users.

area-scaffolding consider-for-current-release needs-design punted-for-2.0 type-enhancement

Most helpful comment

EF Core Power Tools now allows you full control of both namespace and folder location!

All 35 comments

You should get either the [Table] attribute on the POCO class or the equivalent fluent API ToTable() in the context OnModelCreating() method saying what the underlying schema/table name are. Please let me know if you're not seeing this.

I know we've discussed before having the schema in the namespace and decided not to. I'm not sure if this was just to stay consistent with what EF6 did or whether there are other reasons. @rowanmiller do you remember?

We should re-discuss this, it was one of the biggest complaints about reverse engineer in EF6.

Discussed and agreed that this would be good. We could detect if you have multiple schemas and start doing sub-folders/sub-namespaces automatically... or we could just have a flag that you supply.

For the moment, we are just ok with the trivial algorithm we have. If we require the flag when we implement this, then we should write a warning telling folks about the flag.

Note: the "DB schema translates to C# namespace" suggestion was also made as part of #3861 (and many other places I'm sure).

Hope this get developed soon :) It will make it lot easier when generating contexts with different schemas.

@rowanmiller Hi,

In addition to @roji's, another (more general I guess) aproach also making use of a parameter
-ContextNamespace, cause even if they may be stored in subfolders, it would be usefull to override the default namespace in the command for general purposes. Ex:

I want to scaffold all tables into Model\Design :
This makes much sense, since I want to keep Model root folder to adaptations and extensions to the scaffolded Model (wich EF also suggests by design while defining classes as partial).

Scaffold-DbContext (...) -OutputDir Model\Design will generate on namespace <project>.Models.Design. ...Too Bad :(

the goal is not only to have a parameter we can use to place the files into a folder (-OutputDir), but also another parameter to define the context namespace. In this case:

Scaffold-DbContext (...) -OutputDir Model\Design -ContextNamespace <project>.Models...

This would output in Models\Design files with namespace '.Models'.

@roji's scenario would be accomplished with something like this:

Scaffold-DbContext (...) -OutputDir Model\Schema -Schema <schema> -ContextNamespace <project>.Model.<schema> in the core, iterating thru the schemas ....

@olavorn your suggestion seems like a separate issue - allowing users to control the namespace separately from the OutputDir isn't the same as asking for schemas to be automatically translated into C# namespaces (even if it your feature proposal would provide a clunky manual workaround). I'd open a separate issue on this.

@roji I agree with you, but before that, seems to me a good idea to exaust this a little bit here; As for the clunky workaround, the last example I put was more like an implementation suggestion for your feature, but it could alse be wrapped all up in a single flag like :

Scaffold-DbContext (...) -OutputDir Models -ContextNamespace <project>.Models -SchemaHierarchy [single|subfolders]

Like that, impact on the current feature may be minimun..

I totally agree with @olavorn's idea. Now I only have 2 choices:

  • Change my partial classes' namespace to the one of my Scaffolded models (I want the other way)
  • mix all my partial classes together (Scaffolded and non-Scaffolded)

I like none of these choices, I would much prefer be able to specify the namespace of my Scaffolded models so I could store my partial classes in separate folders

i would also like to be able to define my namespace when scaffolding the db, has there been any traction with this?

@aday12345 This issue is in our backlog milestone. That means we don't have immediate plans to work on it. I am however clearing up the milestone to re-discuss the priority. If anyone wanted to make a contribution here, we would be happy to consider it too.

Note: when designing/implementing this, also consider the issues raised in #8852

just wondering if there has been any progress on defining the name space when scaffolding an existing db

@aday12345 We discussed and even though we are doing some work in the background for reverse engineering, unfortunately we don't think this specific issue fits in the schedule for 2.1.

ok, thanks.

Is the any usage documentation for scaffolding the list all the options and switch?

@aday12345 - Use Get-Help Scaffold-DbContext in Package Manager console or dotnet ef dbcontext scaffold --help if you are using dotnet CLI

thanks

@bricelam Is there any development for a flag, or any other option, for having the schemas separates in the different folders?

Seems a lot of people want this (myself included) any chance this will actually make it into EFCore soon?

We NEED this ! :-)

I see two problems with namespaces / schemas in databases, with contradicting requirements:

  • Use those to structure a big database context with lots of tables, e. G. by different aspects of the product (e. G. User Management, Invoices, Stock, ...)

  • Use schema name for tenant implementations, and passing the schema name to the DB Context before working with the data. All schemas replicate the same identical layout (represented by the DB Context), but separate data (one per tenant / region / division / ...).

I see use cases for both, and AFAICS, both are not supported well yet as of EF Core 2.1.

What I did for my code ef core scaffolder is that I used meta tags,

which allows to specify the namespace like so

YourProjectName.[SCHEMA].Dal

will generate entities and do replace Schema with the schema name of your table.

If you want to take a look, its here:
https://github.com/PoweredSoft/DbUtils

and implemented here::
https://github.com/PoweredSoft/DbUtils/blob/532853432774c94a42e23adbb44b86219a02e108/PoweredSoft.DbUtils.EF.Generator.SqlServer/Extensions.cs#L14

used by this:
https://github.com/PoweredSoft/DbUtils/blob/532853432774c94a42e23adbb44b86219a02e108/PoweredSoft.DbUtils.EF.Generator/DatabaseGeneratorBase.cs#L72

I think that would be a good solution for the dotnet ef core scaffolder, allows flexibility to the developer to choose where the schema should be placed in the namespace.

The --context-dir parameter should respect the namespace conventions. Using a different context-dir justs creates a dbcontext with the same namespace as the model entites.

@symbiogenesis Would you submit a new issue?

We have a very large MSSql database with different schemas and redundant table names. Some tables have dependencies to tables on different schema. That dependencies appears in Poco Model as a navigation member.

The current behavior to name the redundant tables with an increasing number suffix is not good because you did not know which table is used.
We decide to use database-first for the first time because not to break all internal workflow at once.

Refactor manually all the tables is not really an acceptable solution because that work must be done daily, or at least weakly.

To run the scaffold tool for each schema separate is not working because the generated navigation properties between the schemas are missing then.

So we have a need that the scaffold tool generates subfolders/namespaces for each non-default schema.

@symbiogenesis You are right. We have generated the context outside of the model folder but the namespace does not match the directory. For me it looks like an issue, because that is not an expected behavior.

Our team experiments currently with EFCore.

@Der-Kraken Have you tried EF Core Power Tools? it has a new table renaming option via a json file in the project.

@Der-Kraken Have you tried EF Core Power Tools? it has a new table renaming option via a json file in the project.

@ErikEJ: how does the table renaming option in "EF Core Power Tools" work? Is it the efpt.config.json?

@ErikEJ Thanks, this does work. Although it would be really nice for it to not just prepend each class with the schema name, but to sort classes into schema-named directories. Is that currently possible with the rename feature?

@mikethibault I suggest you create an issue in the EF Core Power Tools repo

Note from triage: ping @bricelam for the simple change that can be made here.

The simple fix is to uniquify class names so we stop writing over the top of other files. (that's a bug) The needs-design fix involves using the schema as part of the namespace. (that's more of a feature)

EF Core Power Tools now allows you full control of both namespace and folder location!

The simple fix is to uniquify class names so we stop writing over the top of other files. (that's a bug)

Looks like we already do this.

Was this page helpful?
0 / 5 - 0 ratings

Related issues

abdu292 picture abdu292  路  99Comments

anpete picture anpete  路  100Comments

matteocontrini picture matteocontrini  路  88Comments

JocaPC picture JocaPC  路  77Comments

satyajit-behera picture satyajit-behera  路  275Comments