2024-10-16 12:01:16 +02:00
|
|
|
const std = @import("std");
|
|
|
|
const zollections = @import("zollections");
|
2024-10-22 14:09:02 +02:00
|
|
|
const database = @import("database.zig");
|
2024-10-16 12:01:16 +02:00
|
|
|
const _sql = @import("sql.zig");
|
2024-10-16 14:07:40 +02:00
|
|
|
const _conditions = @import("conditions.zig");
|
2024-11-22 15:40:10 +01:00
|
|
|
const _relations = @import("relations.zig");
|
2024-10-16 12:01:16 +02:00
|
|
|
const query = @import("query.zig");
|
|
|
|
const insert = @import("insert.zig");
|
|
|
|
const update = @import("update.zig");
|
|
|
|
|
|
|
|
// Type of the "model from SQL data" function.
|
|
|
|
pub fn ModelFromSql(comptime Model: type, comptime TableShape: type) type {
|
|
|
|
return *const fn (raw: TableShape) anyerror!Model;
|
|
|
|
}
|
|
|
|
// Type of the "model to SQL data" function.
|
|
|
|
pub fn ModelToSql(comptime Model: type, comptime TableShape: type) type {
|
|
|
|
return *const fn (model: Model) anyerror!TableShape;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Repository configuration structure.
|
|
|
|
pub fn RepositoryConfiguration(comptime Model: type, comptime TableShape: type) type {
|
|
|
|
return struct {
|
|
|
|
/// Table name for this repository.
|
|
|
|
table: []const u8,
|
|
|
|
|
|
|
|
/// Insert shape used by default for inserts in the repository.
|
|
|
|
insertShape: type,
|
|
|
|
|
|
|
|
/// Key(s) of the model.
|
|
|
|
key: []const []const u8,
|
|
|
|
|
|
|
|
/// Convert a model to an SQL table row.
|
|
|
|
fromSql: ModelFromSql(Model, TableShape),
|
|
|
|
/// Convert an SQL table row to a model.
|
|
|
|
toSql: ModelToSql(Model, TableShape),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2024-10-16 14:07:40 +02:00
|
|
|
/// Build the type of a model key, based on the given configuration.
|
|
|
|
pub fn ModelKeyType(comptime Model: type, comptime TableShape: type, comptime config: RepositoryConfiguration(Model, TableShape)) type {
|
|
|
|
if (config.key.len == 0) {
|
|
|
|
// Get the type of the simple key.
|
|
|
|
return std.meta.fields(TableShape)[std.meta.fieldIndex(TableShape, config.key[0]).?].type;
|
|
|
|
} else {
|
|
|
|
// Build the type of the composite key.
|
|
|
|
|
|
|
|
// Build key fields.
|
|
|
|
var fields: [config.key.len]std.builtin.Type.StructField = undefined;
|
|
|
|
inline for (config.key, &fields) |keyName, *field| {
|
|
|
|
// Build NULL-terminated key name as field name.
|
|
|
|
var fieldName: [keyName.len:0]u8 = undefined;
|
|
|
|
@memcpy(fieldName[0..keyName.len], keyName);
|
|
|
|
|
2024-10-17 18:34:10 +02:00
|
|
|
// Get current field type.
|
|
|
|
const fieldType = std.meta.fields(TableShape)[std.meta.fieldIndex(TableShape, keyName).?].type;
|
|
|
|
|
2024-10-16 14:07:40 +02:00
|
|
|
field.* = .{
|
|
|
|
.name = &fieldName,
|
2024-10-17 18:34:10 +02:00
|
|
|
.type = fieldType,
|
2024-10-16 14:07:40 +02:00
|
|
|
.default_value = null,
|
|
|
|
.is_comptime = false,
|
2024-10-17 18:34:10 +02:00
|
|
|
.alignment = @alignOf(fieldType),
|
2024-10-16 14:07:40 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return @Type(.{
|
|
|
|
.Struct = std.builtin.Type.Struct{
|
|
|
|
.layout = std.builtin.Type.ContainerLayout.auto,
|
|
|
|
.fields = &fields,
|
|
|
|
.decls = &[_]std.builtin.Type.Declaration{},
|
|
|
|
.is_tuple = false,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-11-22 15:40:10 +01:00
|
|
|
/// Model relations definition type.
|
|
|
|
pub fn RelationsDefinitionType(comptime rawDefinition: anytype) type {
|
|
|
|
const rawDefinitionType = @typeInfo(@TypeOf(rawDefinition));
|
|
|
|
|
|
|
|
// Build model relations fields.
|
|
|
|
var fields: [rawDefinitionType.Struct.fields.len]std.builtin.Type.StructField = undefined;
|
|
|
|
inline for (rawDefinitionType.Struct.fields, &fields) |originalField, *field| {
|
|
|
|
field.* = .{
|
|
|
|
.name = originalField.name,
|
|
|
|
.type = _relations.ModelRelation,
|
|
|
|
.default_value = null,
|
|
|
|
.is_comptime = false,
|
|
|
|
.alignment = @alignOf(_relations.ModelRelation),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return built type.
|
|
|
|
return @Type(.{
|
|
|
|
.Struct = std.builtin.Type.Struct{
|
|
|
|
.layout = std.builtin.Type.ContainerLayout.auto,
|
|
|
|
.fields = &fields,
|
|
|
|
.decls = &[_]std.builtin.Type.Declaration{},
|
|
|
|
.is_tuple = false,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-10-16 12:01:16 +02:00
|
|
|
/// Repository of structures of a certain type.
|
2024-11-22 15:40:10 +01:00
|
|
|
pub fn Repository(comptime Model: type, comptime TableShape: type, comptime repositoryConfig: RepositoryConfiguration(Model, TableShape)) type {
|
2024-10-16 12:01:16 +02:00
|
|
|
return struct {
|
|
|
|
const Self = @This();
|
|
|
|
|
2024-11-22 15:40:10 +01:00
|
|
|
pub const ModelType = Model;
|
|
|
|
pub const TableType = TableShape;
|
|
|
|
pub const config = repositoryConfig;
|
|
|
|
|
2024-10-16 12:01:16 +02:00
|
|
|
pub const Query: type = query.RepositoryQuery(Model, TableShape, config);
|
|
|
|
pub const Insert: type = insert.RepositoryInsert(Model, TableShape, config, config.insertShape);
|
|
|
|
|
2024-10-16 14:07:40 +02:00
|
|
|
/// Type of one model key.
|
|
|
|
pub const KeyType = ModelKeyType(Model, TableShape, config);
|
|
|
|
|
2024-11-22 15:40:10 +01:00
|
|
|
pub const relations = struct {
|
|
|
|
/// Make a "one to one" relation.
|
|
|
|
pub fn one(comptime toRepo: anytype, comptime oneConfig: _relations.OneConfiguration) type {
|
|
|
|
return _relations.one(Self, toRepo, oneConfig);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Make a "one to many" or "many to many" relation.
|
|
|
|
pub fn many(comptime toRepo: anytype, comptime manyConfig: _relations.ManyConfiguration) type {
|
|
|
|
return _relations.many(Self, toRepo, manyConfig);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Define a relations object for a repository.
|
|
|
|
pub fn define(rawDefinition: anytype) RelationsDefinitionType(rawDefinition) {
|
|
|
|
const rawDefinitionType = @TypeOf(rawDefinition);
|
|
|
|
|
|
|
|
// Initialize final relations definition.
|
|
|
|
var definition: RelationsDefinitionType(rawDefinition) = undefined;
|
|
|
|
|
|
|
|
// Check that the definition structure only include known fields.
|
|
|
|
inline for (std.meta.fieldNames(rawDefinitionType)) |fieldName| {
|
|
|
|
if (!@hasField(Model, fieldName)) {
|
|
|
|
@compileError("No corresponding field for relation " ++ fieldName);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Alter definition structure to add the field name.
|
|
|
|
@field(definition, fieldName) = .{
|
|
|
|
.relation = @field(rawDefinition, fieldName),
|
|
|
|
.field = fieldName,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return altered definition structure.
|
|
|
|
return definition;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-10-16 12:01:16 +02:00
|
|
|
pub fn InsertCustom(comptime InsertShape: type) type {
|
|
|
|
return insert.RepositoryInsert(Model, TableShape, config, InsertShape);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn Update(comptime UpdateShape: type) type {
|
|
|
|
return update.RepositoryUpdate(Model, TableShape, config, UpdateShape);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Try to find the requested model.
|
2024-10-16 14:07:40 +02:00
|
|
|
/// For simple keys: modelKey type must match the type of its corresponding field.
|
|
|
|
/// modelKey can be an array / slice of keys.
|
|
|
|
/// For composite keys: modelKey must be a struct with all the keys, matching the type of their corresponding field.
|
|
|
|
/// modelKey can be an array / slice of these structs.
|
2024-10-22 14:09:02 +02:00
|
|
|
pub fn find(allocator: std.mem.Allocator, connector: database.Connector, modelKey: anytype) !RepositoryResult(Model) {
|
2024-10-16 12:01:16 +02:00
|
|
|
// Initialize a new query.
|
2024-10-22 14:09:02 +02:00
|
|
|
var modelQuery = Self.Query.init(allocator, connector, .{});
|
2024-10-16 12:01:16 +02:00
|
|
|
defer modelQuery.deinit();
|
|
|
|
|
2024-11-22 15:40:10 +01:00
|
|
|
try modelQuery.whereKey(modelKey);
|
2024-10-16 12:01:16 +02:00
|
|
|
|
|
|
|
// Execute query and return its result.
|
|
|
|
return try modelQuery.get(allocator);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Perform creation of the given new model in the repository.
|
|
|
|
/// The model will be altered with the inserted values.
|
2024-10-22 14:09:02 +02:00
|
|
|
pub fn create(allocator: std.mem.Allocator, connector: database.Connector, newModel: *Model) !RepositoryResult(Model) {
|
2024-10-16 12:01:16 +02:00
|
|
|
// Initialize a new insert query for the given model.
|
2024-10-22 14:09:02 +02:00
|
|
|
var insertQuery = Self.Insert.init(allocator, connector);
|
2024-10-16 12:01:16 +02:00
|
|
|
defer insertQuery.deinit();
|
|
|
|
try insertQuery.values(newModel);
|
|
|
|
insertQuery.returningAll();
|
|
|
|
|
|
|
|
// Execute insert query and get its result.
|
|
|
|
const inserted = try insertQuery.insert(allocator);
|
|
|
|
|
|
|
|
if (inserted.models.len > 0) {
|
|
|
|
// Update model with its inserted values.
|
|
|
|
newModel.* = inserted.models[0].*;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return inserted result.
|
|
|
|
return inserted;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Perform save of the given existing model in the repository.
|
2024-10-22 14:09:02 +02:00
|
|
|
pub fn save(allocator: std.mem.Allocator, connector: database.Connector, existingModel: *Model) !RepositoryResult(Model) {
|
2024-10-16 12:01:16 +02:00
|
|
|
// Convert the model to its SQL form.
|
|
|
|
const modelSql = try config.toSql(existingModel.*);
|
|
|
|
|
|
|
|
// Initialize a new update query for the given model.
|
2024-10-22 14:09:02 +02:00
|
|
|
var updateQuery = Self.Update(TableShape).init(allocator, connector);
|
2024-10-16 12:01:16 +02:00
|
|
|
defer updateQuery.deinit();
|
|
|
|
try updateQuery.set(modelSql);
|
|
|
|
updateQuery.returningAll();
|
|
|
|
|
|
|
|
// Initialize conditions array.
|
2024-11-22 15:40:10 +01:00
|
|
|
var conditions: [config.key.len]_sql.RawQuery = undefined;
|
2024-10-16 12:01:16 +02:00
|
|
|
inline for (config.key, &conditions) |keyName, *condition| {
|
|
|
|
// Add a where condition for each key.
|
|
|
|
condition.* = try updateQuery.newCondition().value(@TypeOf(@field(modelSql, keyName)), keyName, "=", @field(modelSql, keyName));
|
|
|
|
}
|
|
|
|
// Add WHERE to the update query with built conditions.
|
|
|
|
updateQuery.where(try updateQuery.newCondition().@"and"(&conditions));
|
|
|
|
|
|
|
|
// Execute update query and get its result.
|
|
|
|
const updated = try updateQuery.update(allocator);
|
|
|
|
|
|
|
|
if (updated.models.len > 0) {
|
|
|
|
// Update model with its updated values.
|
|
|
|
existingModel.* = updated.models[0].*;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return updated result.
|
|
|
|
return updated;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A repository query result.
|
|
|
|
pub fn RepositoryResult(comptime Model: type) type {
|
|
|
|
return struct {
|
|
|
|
const Self = @This();
|
|
|
|
|
|
|
|
allocator: std.mem.Allocator,
|
|
|
|
mapperArena: std.heap.ArenaAllocator,
|
|
|
|
|
|
|
|
/// The retrieved models.
|
|
|
|
models: []*Model,
|
|
|
|
/// The retrieved models collection (memory owner).
|
|
|
|
collection: zollections.Collection(Model),
|
|
|
|
|
|
|
|
/// Get the first model in the list, if there is one.
|
|
|
|
pub fn first(self: Self) ?*Model {
|
|
|
|
if (self.models.len > 0) {
|
|
|
|
return self.models[0];
|
|
|
|
} else {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Initialize a new repository query result.
|
|
|
|
pub fn init(allocator: std.mem.Allocator, models: zollections.Collection(Model), mapperArena: std.heap.ArenaAllocator) Self {
|
|
|
|
return .{
|
|
|
|
.allocator = allocator,
|
|
|
|
.mapperArena = mapperArena,
|
|
|
|
.models = models.items,
|
|
|
|
.collection = models,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Deinitialize the repository query result.
|
|
|
|
pub fn deinit(self: *Self) void {
|
|
|
|
self.collection.deinit();
|
|
|
|
self.mapperArena.deinit();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|