Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/zoff99/csorma
C Simple ORM (Android?) - written in pure C
https://github.com/zoff99/csorma
c orm sqlite threadsafe
Last synced: about 2 months ago
JSON representation
C Simple ORM (Android?) - written in pure C
- Host: GitHub
- URL: https://github.com/zoff99/csorma
- Owner: zoff99
- License: gpl-3.0
- Created: 2024-05-01T06:26:12.000Z (9 months ago)
- Default Branch: master
- Last Pushed: 2024-05-20T08:33:11.000Z (8 months ago)
- Last Synced: 2024-05-20T20:13:33.737Z (8 months ago)
- Topics: c, orm, sqlite, threadsafe
- Language: C
- Homepage:
- Size: 5.13 MB
- Stars: 0
- Watchers: 1
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# C Simple ORM (Android?)
### Oh nooo, Why?
I was looking around for something very simple and easy to use in pure C and there was nothing around like that.
It is based on the wonderful Android-Orma by FUJI Goro
and on my (not so wonderful) sorma2
https://github.com/maskarade/Android-Orma
https://github.com/gfx
https://github.com/zoff99/sorma2### Features
* pure C
* Thread safe
* safe Strings (UTF-8 or even broken UTF-8 or just random bytes)
* easy to use (for most common SQL operations)
* no dependencies (other than SQLite3 amalagamtion soure file)
* works with ASAN
* works with TSAN
* works with UBSAN
* sqlcipher encryption option (you need to have openssl and libssl and libcrypto installed)### What is does NOT do
* NOT optimized for speed
* NOT optimized for small memory footprint
* NO complex DB operations like JOIN or UNION etc.
* NO multi column primary keys
* NO non ASCII characters in table and column names
* table and column names MUST NOT start or end with a `_` or a `number`
* table and column names MUST only contain `[a-z][_]` (NO uppercase)
* if table or column name starts with (or contains) `public` or `static` there could be issues**try to avoid SQL reserved words in table and column names, there may also be some other naming conflicts.
### Supported architechtures
* Linux
* Raspi
* Windows
* macOS
* macOS silicon arm64
* s390
* mips
* riscV### Usage
#### Create your first C project with csorma
create a directory for your project:
```bash
mkdir -p ./mysuperstuff/
```create one file for each database table that you need.
create file for db table `Person` as `./mysuperstuff/_csorma_Person.java`
(don't worry it is not really Java, we just use the syntax here)
```Java
@Table
public class Person
{
@PrimaryKey(autoincrement = true)
public long id;
@Column
public String name;
@Column
public String address;
@Column
public int social_number;
}
```now create the C sources with the Java CSORMA Generator. you need at least java 17.
```bash
javac csorma_generator.java && java csorma_generator ./mysuperstuff/
```your project is now ready to start.
enter the project directory:
```bash
cd ./mysuperstuff/gen/
```now build your C project stub and run it:
```bash
make csorma_stub
./csorma_stub
```if you want to build your project with sqlcipher (you need to have openssl and libssl and libcrypto installed):
```bash
make clean # clean from the previous build
ENCRYPT_CS=1 make csorma_stub
./csorma_stub
```the output should look (something) like this:
```
STUB: CSORMA version: 0.99.0
STUB: CSORMA SQLite version: 3.45.3
STUB: creating table: Person
STUB: res1: 0STUB: all OK
```you now have your working C project stub.
#### Add some more SQL stuff to your new project
open your C project stub in your favorite C Code Editor or IDE:
```
vim csorma_stub.c
```now let's add commands
(after the `"OrmaDatabase_run_multi_sql()"` line)
to insert a row:
```C
{ // HINT: using blocks here to have `p` be a local var
Person *p = orma_new_Person(o->db);
p->name = csb("Larry Wilson");
p->address = csb("1 Larry Drive, Sunset Town");
p->social_number = 381739;
int64_t inserted_id = orma_insertIntoPerson(p);
orma_free_Person(p);
printf("STUB: inserted id: %lld\n", (long long)inserted_id);
} // HINT: using blocks here to have `p` be a local var
```now lets count how many Persons have the social number `381739`:
```C
{
Person *p = orma_selectFromPerson(o->db);
printf("STUB: count: %d\n", (int)p->social_numberEq(p, 381739)->count(p));
}
```now we do the same but use the `less than` operator `Lt()`
```C
{
Person *p = orma_selectFromPerson(o->db);
printf("STUB: count: %d\n", (int)p->social_numberLt(p, 400000)->count(p));
}
```and insert another Person:
```C
{
Person *p = orma_new_Person(o->db);
p->name = csb("Martha Liebowitz");
p->address = csb("2035 Morning Road, Big City");
p->social_number = 139807;
int64_t inserted_id = orma_insertIntoPerson(p);
orma_free_Person(p);
printf("STUB: inserted id: %lld\n", (long long)inserted_id);
}
```lets iterate through the result of a select statement:
```C
{
Person *p = orma_selectFromPerson(o->db);
PersonList *pl = p->toList(p);
printf("STUB: pl->items=%lld\n", (long long)pl->items);
Person **pd = pl->l;
for(int i=0;iitems;i++)
{
printf("STUB: id=%ld\n", (*pd)->id);
printf("STUB: name=\"%s\"\n", (*pd)->name->s);
printf("STUB: address=\"%s\"\n", (*pd)->address->s);
printf("STUB: social_number=\"%d\"\n", (*pd)->social_number);
pd++;
}
orma_free_PersonList(pl);
}
```here we update all addresses:
```C
{
Person *p = orma_updatePerson(o->db);
int64_t affected_rows3 = p->addressSet(p, csb("1337 Funky Lane, Lala Land"))->execute(p);
printf("STUB: affected rows: %d\n", (int)affected_rows3);
}
```if we do the iteration from above again we will see the changed data
it will look something like that:
```
STUB: affected rows: 2
STUB: pl->items=2
STUB: id=1
STUB: name="Larry Wilson"
STUB: address="1337 Funky Lane, Lala Land"
STUB: social_number="381739"
STUB: id=2
STUB: name="Martha Liebowitz"
STUB: address="1337 Funky Lane, Lala Land"
STUB: social_number="139807"
```delete specifc rows:
```C
{
Person *p = orma_deleteFromPerson(o->db);
int64_t affected_rows2 = p->social_numberEq(p, 139807)->
nameEq(p, csb("Martha Liebowitz"))->execute(p);
printf("STUB: affected rows: %d\n", (int)affected_rows2);
}
```in the end run a freehand SQL to drop the table:
```C
{
char *sql3 = "DROP TABLE Person;";
CSORMA_GENERIC_RESULT res3 = OrmaDatabase_run_multi_sql(o, (const uint8_t *)sql3);
printf("STUB: res3: %d\n", res3);
}
```#### Stub C code, some functions explained
helper function csb and csc:
csb() will build a `csorma_str*` from a `const char*` NULL terminated C string
and csc() will append (or create) a `csorma_str*` from a `buffer and length` (no need for NULL termination)
```C
#define csb(buf)
#define csc(buf,len)
```include the header file:
```C
#include "csorma_runtime.h"
```initialize the database:
```C
const char *db_dir = "./";
const char *db_filename = "stub.db";
OrmaDatabase *o = OrmaDatabase_init(
(uint8_t*)db_dir, strlen(db_dir),
(uint8_t*)db_filename, strlen(db_filename)
);
```you can also initialize an in-memory database:
```C
const char *db_dir = ":memory:";
const char *db_filename = "";
OrmaDatabase *o = OrmaDatabase_init(
(uint8_t*)db_dir, strlen(db_dir),
(uint8_t*)db_filename, strlen(db_filename)
);
```run a freehand SQL:
```C
char *sql1 = "CREATE TABLE IF NOT EXISTS Message ("
"message_id INTEGER NOT NULL,"
"read BOOLEAN,"
"direction INTEGER ,"
"text TEXT,"
"id INTEGER,"
"PRIMARY KEY(\"id\" AUTOINCREMENT)"
");"
"insert into message(message_id,text) values('123','test message');";
CSORMA_GENERIC_RESULT res1 = OrmaDatabase_run_multi_sql(o, sql1);
```shutdown the database:
```C
OrmaDatabase_shutdown(o);
```### sqlcipher source code
sqlcipher source code is generated from latest [sqlcipher git repository](https://github.com/sqlcipher/sqlcipher) using this [tool](https://github.com/zoff99/gen_sqlcipher_amalgamation)
you can download the generated sqlcipher code from https://github.com/zoff99/gen_sqlcipher_amalgamation/releases/tag/nightly
Any use of this project's code by GitHub Copilot, past or present, is done
without our permission. We do not consent to GitHub's use of this project's
code in Copilot.
No part of this work may be used or reproduced in any manner for the purpose of training artificial intelligence technologies or systems.