Efcore.Notation.Extensions
7.0.13.1
dotnet add package Efcore.Notation.Extensions --version 7.0.13.1
NuGet\Install-Package Efcore.Notation.Extensions -Version 7.0.13.1
<PackageReference Include="Efcore.Notation.Extensions" Version="7.0.13.1" />
paket add Efcore.Notation.Extensions --version 7.0.13.1
#r "nuget: Efcore.Notation.Extensions, 7.0.13.1"
// Install Efcore.Notation.Extensions as a Cake Addin #addin nuget:?package=Efcore.Notation.Extensions&version=7.0.13.1 // Install Efcore.Notation.Extensions as a Cake Tool #tool nuget:?package=Efcore.Notation.Extensions&version=7.0.13.1
Notation extender for EntityFramework Core.
This library adds support for Notation Attribute in EntityFramework Core. It is also possible to create custom data types and add custom notations. You don't need to learn anything else to use this library. All you need is to know a few Notations in addition to the standard features of EntityFramework Core and how to write custom Notations!
Installation.
Visit nuget package gallery: https://www.nuget.org/packages/Efcore.Notation.Extensions
A way to implement custom notation attribute.
Custom Notation attributes can be specified on both Type
and Property
.
The previously mentioned Type
is a data Type
, not an Entity
.
This means that any manipulation of how a Property
is stored,
how it is structured in the database, etc. can be manipulated.
Below code is example notation for creating Index
to database entity.
/// <summary>
/// Index attribute.
/// Marks a property as index.
/// </summary>
[AttributeUsage(AttributeTargets.Property, AllowMultiple = true, Inherited = true)]
public class IndexAttribute : NotationAttribute
{
/// <summary>
/// Name of index.
/// </summary>
public string Name { get; set; }
/// <summary>
/// Order.
/// </summary>
public int Order { get; set; } = int.MaxValue / 2;
/// <summary>
/// Collection.
/// </summary>
private class Collection : HashSet<(string Name, int Order, string Property)>
{
}
/// <inheritdoc/>
protected override void OnConfigure<TEntity, TProperty>(NotationContext<TEntity, TProperty> Context)
{
Context.Items.TryGetValue(typeof(Collection), out var Temp);
if (Temp is not Collection Collection)
{
Context.Items[typeof(Collection)] = Collection = new Collection();
Context.LazyWorks.Enqueue(() =>
{
foreach(var Each in Collection.GroupBy(X => X.Name))
{
var Properties = Each
.OrderBy(X => X.Order).Select(X => X.Property)
.ToArray();
Context.Entity.HasIndex(Properties, Each.Key).IsUnique(false);
}
});
}
var Name = this.Name;
if (string.IsNullOrWhiteSpace(Name))
{
var Property = Context.PropertyInfo.Name.ToUpper();
var Entity = Context.PropertyInfo.DeclaringType.Name.ToUpper();
Name = $"IX_{Entity}_{Property}";
}
Collection.Add((Name, Order, Property: Context.PropertyInfo.Name));
}
}
Usage of IndexAttribute
is:
[Table("MyEntities")]
public class MyEntity
{
/// <summary>
/// GUID.
/// </summary>
[MultiKey(Order = 0)]
public Guid Guid { get; set; }
/// <summary>
/// Number.
/// </summary>
[MultiKey(Order = 1), Index(Name = "MyIndex", Order = 1)]
public int Number { get; set; }
/// <summary>
/// Remote Address.
/// </summary>
[Index(Name = "MyIndex", Order = 0)]
public IPAddress RemoteAddress { get; set; }
/// <summary>
/// Text.
/// </summary>
[Column(TypeName = "LONGTEXT")]
public string Text { get; set; }
/// <summary>
/// Password.
/// </summary>
public Password Password { get; set; }
}
As you can see above, it's very simple.
A way to implement custom type conversion attribute.
This is an example of a conversion attribute that is always converted to JSON and saved when this Attribute is specified.
/// <summary>
/// Store the property as JSON string.
/// </summary>
[AttributeUsage(AttributeTargets.Class, AllowMultiple = false, Inherited = true)]
public class StoreAsJsonAttribute : ConversionAttribute
{
/// <inheritdoc/>
protected override string Stringify<TProperty>(TProperty Property)
{
if (Property is null)
return string.Empty;
try { return JsonConvert.SerializeObject(Property, Formatting.None); }
catch
{
}
return string.Empty;
}
/// <inheritdoc/>
protected override bool TryParse<TProperty>(string Input, out TProperty Property)
{
if (string.IsNullOrWhiteSpace(Input))
{
Property = default;
return false;
}
try
{
Property = JsonConvert.DeserializeObject<TProperty>(Input);
return true;
}
catch { }
Property = default;
return false;
}
/// <inheritdoc/>
protected override void OnConfigureProperty<TEntity, TProperty>(NotationContext<TEntity, TProperty> Context)
{
base.OnConfigureProperty(Context);
Context.Property.HasColumnType("LONGTEXT");
}
}
You can even save custom Type
s right away using this method!
Take a look at how the Password
structure is implemented in this library as example!
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net7.0 is compatible. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. |
-
net7.0
- Microsoft.EntityFrameworkCore (>= 7.0.13)
- Microsoft.EntityFrameworkCore.Relational (>= 7.0.13)
- Newtonsoft.Json (>= 13.0.3)
- Newtonsoft.Json.Bson (>= 1.0.2)
- Secp256k1.Native (>= 0.1.23)
- Secp256k1.Net (>= 0.1.55)
- SimpleBase (>= 4.0.0)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.