fediglam/src/sql/lib.zig

109 lines
3.2 KiB
Zig
Raw Normal View History

2022-07-15 00:58:08 +00:00
const std = @import("std");
const c = @cImport({
@cInclude("sqlite3.h");
});
pub const Sqlite = struct {
db: *c.sqlite3,
pub fn open(path: [:0]const u8) !Sqlite {
var db: ?*c.sqlite3 = undefined;
const err = c.sqlite3_open_v2(path, &db, c.SQLITE_OPEN_READWRITE | c.SQLITE_OPEN_CREATE, null);
if (err != c.SQLITE_OK) return error.UnknownError;
return Sqlite{
.db = db.?,
};
}
pub fn close(self: *Sqlite) void {
2022-07-15 07:27:27 +00:00
_ = c.sqlite3_close_v2(self.db);
2022-07-15 00:58:08 +00:00
}
pub fn prepare(self: *Sqlite, sql: []const u8) !PreparedStmt {
2022-07-15 07:27:27 +00:00
var stmt: ?*c.sqlite3_stmt = undefined;
const err = c.sqlite3_prepare_v2(self.db, sql.ptr, @intCast(c_int, sql.len), &stmt, null);
2022-07-16 18:41:09 +00:00
if (err != c.SQLITE_OK) {
std.debug.print("sql error {}\n", .{err});
std.debug.print("{s}\n", .{c.sqlite3_errmsg(self.db)});
return error.UnknownError;
}
2022-07-15 00:58:08 +00:00
2022-07-15 07:27:27 +00:00
return PreparedStmt{ .stmt = stmt.?, .db = self.db };
2022-07-15 00:58:08 +00:00
}
};
pub const Row = struct {
stmt: *c.sqlite3_stmt,
2022-07-15 07:27:27 +00:00
db: *c.sqlite3,
2022-07-15 00:58:08 +00:00
2022-07-15 07:27:27 +00:00
pub fn getI64(self: Row, idx: u15) !i64 {
2022-07-15 00:58:08 +00:00
return @intCast(i64, c.sqlite3_column_int64(self.stmt, idx));
}
2022-07-15 07:27:27 +00:00
pub fn getText(self: Row, idx: u15, buf: []u8) ![]u8 {
2022-07-15 00:58:08 +00:00
const ptr = c.sqlite3_column_text(self.stmt, idx);
2022-07-15 07:27:27 +00:00
const size = @intCast(usize, c.sqlite3_column_bytes(self.stmt, idx));
2022-07-15 00:58:08 +00:00
if (size > buf.len) return error.StreamTooLong;
for (ptr[0..size]) |ch, i| buf[i] = ch;
return buf[0..size];
}
2022-07-15 07:27:27 +00:00
pub fn getTextAlloc(self: Row, idx: u15, alloc: std.mem.Allocator) ![]u8 {
2022-07-15 00:58:08 +00:00
const size = c.sqlite3_column_bytes(self.stmt, idx);
2022-07-15 07:27:27 +00:00
var buf = try alloc.alloc(u8, @intCast(usize, size));
2022-07-15 00:58:08 +00:00
errdefer alloc.free(buf);
return self.getText(idx, buf);
}
};
pub const PreparedStmt = struct {
stmt: *c.sqlite3_stmt,
2022-07-15 07:27:27 +00:00
db: *c.sqlite3,
2022-07-15 00:58:08 +00:00
pub fn bindNull(self: *PreparedStmt, idx: u15) !void {
return switch (c.sqlite3_bind_null(self.stmt, idx)) {
2022-07-15 07:27:27 +00:00
.SQLITE_OK => {},
2022-07-15 00:58:08 +00:00
else => error.UnknownError,
};
}
pub fn bindText(self: *PreparedStmt, idx: u15, str: []const u8) !void {
2022-07-15 07:27:27 +00:00
return switch (c.sqlite3_bind_text(self.stmt, idx, str.ptr, @intCast(c_int, str.len), c.SQLITE_TRANSIENT)) {
c.SQLITE_OK => {},
2022-07-15 00:58:08 +00:00
else => error.UnknownError,
};
}
pub fn bindI64(self: *PreparedStmt, idx: u15, val: i64) !void {
return switch (c.sqlite3_bind_int64(self.stmt, idx, val)) {
2022-07-15 07:27:27 +00:00
.SQLITE_OK => {},
2022-07-15 00:58:08 +00:00
else => error.UnknownError,
};
}
pub fn step(self: *PreparedStmt) !?Row {
return switch (c.sqlite3_step(self.stmt)) {
2022-07-15 07:27:27 +00:00
c.SQLITE_ROW => Row{ .stmt = self.stmt, .db = self.db },
c.SQLITE_DONE => null,
2022-07-15 00:58:08 +00:00
2022-07-16 05:29:08 +00:00
else => |err| blk: {
std.debug.print("sql error {}\n", .{err});
std.debug.print("{s}\n", .{c.sqlite3_errmsg(self.db)});
break :blk error.UnknownError;
},
2022-07-15 00:58:08 +00:00
};
}
pub fn finalize(self: *PreparedStmt) void {
_ = c.sqlite3_finalize(self.stmt);
}
pub fn reset(self: *PreparedStmt) void {
_ = c.sqlite3_reset(self.stmt);
}
};