Funcular.Data.Orm
3.1.0-alpha1
See the version list below for details.
dotnet add package Funcular.Data.Orm --version 3.1.0-alpha1
NuGet\Install-Package Funcular.Data.Orm -Version 3.1.0-alpha1
<PackageReference Include="Funcular.Data.Orm" Version="3.1.0-alpha1" />
<PackageVersion Include="Funcular.Data.Orm" Version="3.1.0-alpha1" />
<PackageReference Include="Funcular.Data.Orm" />
paket add Funcular.Data.Orm --version 3.1.0-alpha1
#r "nuget: Funcular.Data.Orm, 3.1.0-alpha1"
#:package Funcular.Data.Orm@3.1.0-alpha1
#addin nuget:?package=Funcular.Data.Orm&version=3.1.0-alpha1&prerelease
#tool nuget:?package=Funcular.Data.Orm&version=3.1.0-alpha1&prerelease
Recent Changes
- v3.1.0: 🐘 PostgreSQL Support! FunkyORM now supports PostgreSQL via the new
Funcular.Data.Orm.PostgreSqlNuGet package. Full LINQ-to-SQL, remote keys/properties, transactions, and reserved word handling — everything you know from the MSSQL provider, now on Postgres. See Database Provider Differences for details.- v3.0.1: Major Refactoring & New Features. Introduced
ISqlDialectfor multi-database support. Added[RemoteKey]and[RemoteProperty]attributes for easy relationship mapping. Added support forGuidandStringprimary keys, genericInsert<T, TKey>overloads, and non-identity key handling.
Funcular / Funky ORM: a speedy, lambda-powered .NET micro-ORM for MSSQL & PostgreSQL

For AI Agents: Please refer to COPILOT_INSTRUCTIONS.md for strict coding guidelines and "Happy Path" patterns. This file is included in both NuGet packages. A PostgreSQL-specific supplement is at COPILOT_INSTRUCTIONS.md.
Tip for Consumers: To help AI agents (Copilot, Cursor, etc.) generate correct FunkyORM code in your project, copy
COPILOT_INSTRUCTIONS.mdfrom the NuGet package to your project root or.github/folder.
Overview
Welcome to Funcular ORM (aka FunkyORM), the micro-ORM designed for developers who want the speed of a micro-ORM with the simplicity and type safety of LINQ.
If you're tired of wrestling with raw SQL strings (Dapper) or debugging generated queries from a heavy framework (Entity Framework), FunkyORM is your sweet spot.
Why FunkyORM?
- Instant Lambda Queries: Write C# lambda expressions, get optimized SQL.
- Performance: Outperforms EF Core in single-row writes and matches it in reads. (See our Usage Guide for benchmarks).
- Zero Configuration: No
DbContext, no mapping files. Just POCOs and a connection string. - Safe: All queries are parameterized to prevent SQL injection.
- Mass Delete Prevention: Includes safeguards against accidental "delete all" operations (e.g., blocking
1=1), though this does not guarantee prevention of all crafty circumventions. - Convention over Configuration: Sensible defaults for primary key naming conventions (like
id,tablename_id, orTableNameId) mean less boilerplate and more productivity. - Remote Keys & Properties: Flatten your object graph by mapping properties directly to columns in related tables (e.g.,
Person.EmployerCountryName) without writing joins. The ORM handles the graph traversal for you. - Explicit Collection Population: Leverage
RemoteKeyproperties to easily populate related collections without the overhead of massive object graphs or N+1 queries. - Cached Reflection: Funcular ORM caches reflection results to minimize overhead and maximize performance.
- Nullable-Friendly: Nullable properties work seamlessly in LINQ queries—no need for
.Valueor.HasValue. The ORM handles the unwrapping for you.
Getting Started
1. Installation
Add the provider package for your database:
SQL Server:
dotnet add package Funcular.Data.Orm
PostgreSQL:
dotnet add package Funcular.Data.Orm.PostgreSql
2. Initialization
Create a provider instance. You can register it as a singleton in your DI container, or create it as needed.
SQL Server:
using Funcular.Data.Orm.SqlServer;
var connectionString = "Server=.;Database=MyDb;Integrated Security=True;TrustServerCertificate=True;";
var provider = new SqlServerOrmDataProvider(connectionString);
PostgreSQL:
using Funcular.Data.Orm.PostgreSql;
var connectionString = "Host=localhost;Port=5432;Database=mydb;Username=myuser;Password=mypassword";
var provider = new PostgreSqlOrmDataProvider(connectionString);
Note: Both providers implement the same base class and support the same LINQ query API, CRUD operations, remote keys/properties, and transactions. Your entity classes and query code are fully portable between providers.
3. Define Your Data Models
FunkyORM is designed to keep your code clean. You usually don't need attributes.
By default, we map the intersection of your class properties and the database table columns.
- If your class has a
FullNameproperty but the table doesn't, we ignore it. No[NotMapped]needed. - If the table has a
CreatedDatecolumn but your class doesn't, we ignore it. No errors.
We also infer table names and primary keys automatically.
// No attributes needed!
// Maps to table 'Person' or 'PERSON' (case-insensitive)
public class Person
{
// Automatically detected as Primary Key
public int Id { get; set; }
// Maps to column 'FirstName', 'first_name', etc.
public string FirstName { get; set; }
public string LastName { get; set; }
public int Age { get; set; }
// Ignored automatically if no matching column exists
public string FullName => $"{FirstName} {LastName}";
}
If you need to deviate from conventions (e.g., mapping Person class to tbl_Users), you can still use standard System.ComponentModel.DataAnnotations:
using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;
[Table("tbl_Users")]
public class Person
{
[Key]
[Column("user_id")]
public int Id { get; set; }
// ...
}
4. Start Querying
// Insert a new record and get the ID
var newPerson = new Person { FirstName = "Jane", LastName = "Doe", Age = 25 };
var newId = provider.Insert<Person, int>(newPerson);
// Get by ID
var person = provider.Get<Person>(newId);
// Complex Querying with LINQ
var adults = provider.Query<Person>()
.Where(p => p.Age >= 18)
.Where(p => p.LastName.StartsWith("D"))
.OrderByDescending(p => p.Age)
.Take(10)
.ToList();
5. Superpowers: Remote Keys & Properties
This is where FunkyORM shines. You can map properties in your entity to columns in related tables without writing joins or loading the entire object graph.
[RemoteProperty]: Fetch a value from a related table (e.g.,Employer.Name) directly into yourPersonobject.[RemoteKey]: Fetch the ID of a related entity (e.g.,Employer.CountryId) directly.
public class Person
{
// ... standard properties ...
// Link to the Organization table
[RemoteLink(targetType: typeof(Organization))]
public int? EmployerId { get; set; }
// SUPERPOWER 1: Get the Employer's Name without loading the Organization object
[RemoteProperty(remoteEntityType: typeof(Organization), keyPath: new[] { nameof(EmployerId), nameof(Organization.Name) })]
public string EmployerName { get; set; }
// SUPERPOWER 2: Get the Employer's Country ID (2 hops away!)
// Person -> Organization -> Address -> Country
[RemoteKey(remoteEntityType: typeof(Country), keyPath: new[] {
nameof(EmployerId),
nameof(Organization.HeadquartersAddressId),
nameof(Address.CountryId),
nameof(Country.Id) })]
public int? EmployerCountryId { get; set; }
}
// Now you can query and filter by these remote properties as if they were local!
var usEmployees = provider.Query<Person>()
.Where(p => p.EmployerCountryId == 1) // Filters by joined table!
.ToList();
The "Superpower" Advantage
To achieve this without FunkyORM, you'd typically have to do this:
Entity Framework Core:
// Requires loading the entire graph or creating a custom DTO
var person = context.People
.Include(p => p.Employer)
.ThenInclude(e => e.Address)
.ThenInclude(a => a.Country) // Heavy!
.FirstOrDefault();
// Access: person.Employer.Address.Country.Name
Dapper:
// Requires writing raw SQL joins and manual mapping
var sql = @"SELECT p.*, c.Name as CountryName
FROM Person p
JOIN Organization o ON p.EmployerId = o.Id ..."; // ... and so on
FunkyORM:
// Just add the attribute. We handle the joins.
// Path: Person -> Organization (via EmployerId) -> Address -> Country
[RemoteProperty(remoteEntityType: typeof(Country), keyPath: new[] {
nameof(EmployerId),
nameof(Organization.HeadquartersAddressId),
nameof(Address.CountryId),
nameof(Country.Name) })]
public string EmployerCountryName { get; set; }
Database Provider Differences
FunkyORM generates database-specific SQL through its ISqlDialect abstraction. Your entity classes and LINQ queries are portable, but the generated SQL differs to match each platform's conventions.
| Feature | SQL Server (Funcular.Data.Orm) |
PostgreSQL (Funcular.Data.Orm.PostgreSql) |
|---|---|---|
| NuGet Package | Funcular.Data.Orm |
Funcular.Data.Orm.PostgreSql |
| Provider Class | SqlServerOrmDataProvider |
PostgreSqlOrmDataProvider |
| Identifier Quoting | [brackets] |
"double-quotes" |
| Insert Return | OUTPUT INSERTED.id |
RETURNING id |
| Paging | OFFSET…FETCH NEXT |
LIMIT…OFFSET |
| String Concat | + |
‖ |
| Date Parts | YEAR(), MONTH(), DAY() |
EXTRACT(YEAR FROM …) |
| Boolean Type | BIT (0/1) |
Native BOOLEAN |
| Target Frameworks | net8.0, netstandard2.0, net48 |
net8.0, netstandard2.0 |
| ADO.NET Driver | Microsoft.Data.SqlClient |
Npgsql |
PostgreSQL-Specific Notes
- Naming convention: PostgreSQL is case-sensitive for quoted identifiers. Unquoted identifiers are folded to lowercase. FunkyORM quotes reserved words automatically (e.g.,
"User","Order"), and leaves non-reserved names unquoted (matching PostgreSQL's lowercase convention). - Npgsql versions: The PostgreSQL provider uses Npgsql 9.x for
net8.0and Npgsql 8.x fornetstandard2.0(last version with netstandard support). - Timestamps: The provider sets
AppContext.SetSwitch("Npgsql.EnableLegacyTimestampBehavior", true)to ensureDateTimevalues are handled consistently without requiringtimestamptzconversions.
Documentation
For detailed usage examples, performance benchmarks, and a comparison with other ORMs, please see our Usage Guide.
Comparison: FunkyORM vs. The World
| Feature | Entity Framework | Dapper | FunkyORM |
|---|---|---|---|
| Setup | Heavy (DbContext, Config) | Light | Lightest |
| Query Style | LINQ | SQL Strings | LINQ |
| Performance | Good (if tuned) | Excellent | Excellent |
| Mapping | Strict (needs config) | Manual/Strict | Forgiving/Auto |
| SQL Injection | Protected | Manual Parameterization | Protected |
| Vibe | Enterprise Java | Hardcore Metal | Cool Jazz |
| Product | Versions Compatible and additional computed target framework versions. |
|---|---|
| .NET | net5.0 was computed. net5.0-windows was computed. net6.0 was computed. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. 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 is compatible. 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. net10.0 was computed. net10.0-android was computed. net10.0-browser was computed. net10.0-ios was computed. net10.0-maccatalyst was computed. net10.0-macos was computed. net10.0-tvos was computed. net10.0-windows was computed. |
| .NET Core | netcoreapp2.0 was computed. netcoreapp2.1 was computed. netcoreapp2.2 was computed. netcoreapp3.0 was computed. netcoreapp3.1 was computed. |
| .NET Standard | netstandard2.0 is compatible. netstandard2.1 was computed. |
| .NET Framework | net461 was computed. net462 was computed. net463 was computed. net47 was computed. net471 was computed. net472 was computed. net48 is compatible. net481 was computed. |
| MonoAndroid | monoandroid was computed. |
| MonoMac | monomac was computed. |
| MonoTouch | monotouch was computed. |
| Tizen | tizen40 was computed. tizen60 was computed. |
| Xamarin.iOS | xamarinios was computed. |
| Xamarin.Mac | xamarinmac was computed. |
| Xamarin.TVOS | xamarintvos was computed. |
| Xamarin.WatchOS | xamarinwatchos was computed. |
-
.NETFramework 4.8
- Microsoft.Data.SqlClient (>= 6.0.0)
- System.ComponentModel.Annotations (>= 5.0.0)
-
.NETStandard 2.0
- Microsoft.Data.SqlClient (>= 6.0.0)
- System.ComponentModel.Annotations (>= 5.0.0)
- System.Data.Common (>= 4.3.0)
-
net8.0
- Microsoft.Data.SqlClient (>= 6.0.0)
- System.ComponentModel.Annotations (>= 5.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.