Skip to content

An open source library to help you to create sqlite queries for mobile solutions avoiding a lot of dirty string concatenations

License

Notifications You must be signed in to change notification settings

mludlum/sqlitequerybuilder

 
 

Repository files navigation

SQLiteQueryBuilder

It’s an amazing, free and open source .net library to get rid of repetitive query writing.

Donations

Did you like it? Did it useful for you? Do you want to see it get even better? You can give us a cup of coffee to help with maintenance and improving! Donate

WHAT IT ISN'T

This is NOT an ORM framework nor database connection manager. All it does is compose sql query strings

What is it?

It’s a simple tool that use reflection to dynamically provide strings with SQLite statements using the C# object notation, what making DAL maintenance pretty easy and avoid a lot of dangerous Ctrl + C, Ctrl + V based changes on strings annoying concatenations.

The library was designed to be basic enough to do the job but allowing you to create elaborated solutions according to your project's architecture.

Targets

Every .Net solution wherever you have the need to build SQLite query strings it would fit. Please let me know if you're getting troubles with a specific platform.  

Features

On the following board, you can see the library’s features plan. We’re glad to receive your suggestions about what more it can do to make our development cool and amazing.

Feature Available Library version Notes
Select query Available 0.0.1.0+ Simple, Distinct, Top, Count and custom
Join Available 0.0.1.0+ Inner and Outer
Single Conditions Available 0.0.1.0+
Block conditions Available 0.0.1.0+
Operators support Available 0.0.1.0+ OR, AND, LIKE, IN, NOT IN, >=, <=, >, < and <>
Custom column select Available 0.0.1.0+
Union Available 0.0.1.5+
Inferred joins* Available 0.0.1.6+ Just pass table types as generic parameters
Exists and Not Exists operators Available 0.0.1.6+
Lambda Expressions support on conditions Developing Less parameters to query construction
Group By Soon
Methods for return expression trees Soon
Class x Table and Property x Column mapping Soon
Custom conditions Unavailable
Custom joins Unavailable
Order By Unavailable

* Just for single keyed tables with attributes PrimaryKey and ForeignKey defined

Why can it be useful for you?

Clean, Reusable and Maintenable

string param = "not";
string[] filter = new [] { "easy", "quick"};

string phrase = "SELECT 'It is " + param + " just a SQL trauma' as [MainReason] ";
phrase += "FROM Too_Many_Years_Facing_Cumbersome_Crazyness_Like_This as table ";
phrase += " WHERE table.Maintenance " + param + " IN + ("  + string.Join(", ", filter) + ") ";
  • Enough with headaches wasting hours seeking for a single character messing the things up after just to add a column. Your time deserves more
  • There's no reasonable reason to keep thousands of classes to make the same queries where almost only columns and tables names are changed
  • You get much more performance than using the common built-in lambda based feature (they loads the whole table data to memory to perform filters)

Coding without handle string concatenations keep the code dry and clean, allowing you to create reusable custom features to your data access strategies and makes the code fluent and pretty easy to understand and maintain.

How to use

Let's see some code!

Supposing a simple class (database table equivalent):

public class Person
{
    public long Id { get; set; }
    public string Name { get; set; }
    public DateTime BirthDate { get; set; }
    public int Age { get { return Convert.ToInt32((DateTime.Now.Date - BirthDate.Date).TotalDays / 365); } }

    /*
       The `Age` property is read only, so it will be ignored by QueryBuilder on SELECT clause.
       It's the same for that one with `Ignore` or relationships attribute, like these:

       [Ignore]
       public int Age { get; set; }
       [ManyToOne]
       public Person Father { get; set; }
    */
}

Simple, distinct, count and top queries for persons born before October 14, 1995

Easy and quick! Just like this:

// Instantiate it setting the table type
QueryBuilder qb = new QueryBuilder<Person>();

// Create conditions
var filter = new Condition<Person, DateTime>(nameof(Person.BirthDate), BoolComparisonType.LessThan, new DateTime(1995, 10, 14));

// Add conditions to querybuilder
qb.AddWhereCondition(filter);

// Just run your desired method
string qCount = qb.BuildCount();
string qDistinct = qb.BuildSelectDistinct();
string qTop = qb.BuildSelectTop(7);

The results are, respectively:

-- Count
SELECT COUNT(*)
FROM Person Person 
WHERE  Person.BirthDate < '1995-10-14 00:00:00' 

-- Distinct
SELECT DISTINCT Person.Id, Person.Name, Person.BirthDate
FROM Person Person 
WHERE  Person.BirthDate < '1995-10-14 00:00:00' 

-- Top
SELECT TOP 7 Person.Id, Person.Name, Person.BirthDate
FROM Person Person 
WHERE  Person.BirthDate < '1995-10-14 00:00:00' 

Joins and Condition Blocks (complex queries)

To exemplify joins, lets include a new property on Person and create the Dog class:

public class Person
{
    (...) // As we set before
	
    [OneToMany] // As it's a relationship, it'll be ignored on query
    public List<Dog> Pets { get; set; }
}

public class Dog
{
    public string Name { get; set; }
    // A person id reference
    public long IdOwner { get; set; }
    public string FavoriteFood { get; set; }
    public double Weight { get; set; }

    [ManyToOne] // As it's a relationship, it'll be ignored on query
    public Person Owner { get; set; } 

    public void Speak()
    {
	System.Diagnostics.Debug.WriteLine("Rooff!");
    }
}

Now, lets take persons with the previous filter conditions and additionally have a dog with one of the names of the list [ "Spike", "Thor" ] OR the dog's FavoriteFood have the word "fruit" AND weighs less than 7.5kg.

Just add this statements to the previous query:

// Create the join conditions
var joinCondition = new Condition<Person, Dog>(nameof(Person.Id), BoolComparisonType.Equals, nameof(Dog.IdOwner));

// Explicitly add the join with the join condition
qb.AddJoin<Person, Dog>(JoinType.Inner /* Join type */, null /* alias for the right type table */, joinCondition);

var singleCondition = new Condition<Dog, double>(nameof(Dog.Weight) /* Left table property name */, BoolComparisonType.LessThanOrEqualTo /* Operator */, 7.5 /* raw value or right table property */);

var innerCondOne = new Condition<Dog, string>(nameof(Dog.Name), BoolComparisonType.In, new object[] { "Spike", "Thor" }, LogicalOperatorType.Or /* logical operator for condition's concatenation */);
var innerCondTwo = new Condition<Dog, string>(nameof(Dog.FavoriteFood), BoolComparisonType.Like, "%fruit%");

var block = new ExpressionBlock(LogicalOperatorType.And /* logical operator for condition's concatenation */, innerCondOne, innerCondTwo ),

qb.AddWhereCondition(block, singleCondition);

Result:

SELECT TOP 7 Person.Id, Person.Name, Person.BirthDate 
FROM Person Person 
    JOIN Dog Dog ON  Person.Id = Dog.IdOwner  
WHERE  Person.BirthDate < '1995-10-14 00:00:00'  AND 
(  Dog.Name IN ( 'Spike', 'Thor' )  OR 
 Dog.FavoriteFood LIKE '%fruit%'  )
 AND 
 Dog.Weight <= 7.5 

Using Join inference

To use Join inference, you must decorate the PrimaryKey and ForeignKey properties on its respective classes.

For example, in our Person and Dog classes, we should chande the properties declaration to this:

public class Person
{
    [Primarykey]
    public long Id { get; set; }
    
    (...)
}

public class Dog
{
    [Primarykey]
    public long Id { get; set; }
    
    [ForeignKey(typeof(Person))]
    public long IdOwner { get; set; }
    (...)
}

Now you are able to do:

QueryBuilder qb = new QueryBuilder<Person>("owner"); // Giving the 'owner' alias to 'Person'
qb.AddJoin<Person,Dog>(); // To use LeftJoin, just add a `JoinType.Outer` parameter to method.

qb.BuildSelectDistinct();

Result:

SELECT DISTINCT owner.Id, owner.Name, owner.BirthDate 
FROM Person owner 
    INNER JOIN Dog dog ON  owner.Id = dog.IdOwner  

The Exists and Not Exists operator

To get the Exists and Not Exists operators, you must build an ExistenceCondition using a inner query as parameter. For example, if we want to get Persons that don't have a Dog:

QueryBuilder qb = new SQLiteQueryBuilder.QueryBuilder<Person>("P");

QueryBuilder innerQuery = new SQLiteQueryBuilder.QueryBuilder<Dog>("D");
innerQuery.AddWhereCondition(new Condition<Person, Dog>(nameof(Person.Id), "P", BoolComparisonType.Equals, nameof(Dog.IdOwner), "D" ));

qb.AddWhereCondition(new ExistenceCondition(innerQuery, ExistenceType.NotExists));

qb.BuildSelectDistinct();

Result:

SELECT DISTINCT P.Id, P.Name, P.BirthDate 
FROM Person P 
WHERE NOT EXISTS ( 
    SELECT  1 
    FROM Dog D 
    WHERE P.Id = D.IdOwner 
 ) 

Contribution, bug report and questions

Contributions are welcome! Feel free to report if you find a bug or if you want a new feature or give any kind of suggestions.

Have a nice coding!

About

An open source library to help you to create sqlite queries for mobile solutions avoiding a lot of dirty string concatenations

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • C# 100.0%