summaryrefslogtreecommitdiff
path: root/model
AgeCommit message (Collapse)Author
2024-11-03model: Implement favorite recipesxengineering
This adds the ability to store a flag `is_favorite` inside the database. The corresponding SQL migration is part of this commit. Furthermore the Go code in the model package is adapted.
2024-10-24view: Order ingredient summary by ingredient namexengineering
This makes it easier to write a shopping list for the required ingredients.
2024-10-24Rework error handling of model.DB.Transaction()xengineering
The functions in the sub-packages model, view and controller should not log too much. Instead they should return errors which can be extended by fmt.Errorf("<some mesage>: %w", err).
2024-10-23Fix unit testsxengineering
API changes were not adopted in the unit tests.
2024-10-21model: Rename to db.goxengineering
The old name database.go did not match the type name DB.
2024-10-21model: Refactor public API of DBxengineering
This commit makes not externally needed methods private and adds error return values since something like log.Fatal() should be called outside this package since it is control-flow-related.
2024-10-21model: Save int-based schema version in DBxengineering
Ceres v0.4.0 used the `git describe` output as database schema and enforced exactly matching versions between the database and the executable. This turned out to be not flexible enough. It is way easier to version the database separately with a simple integer and require the same database schema version integer between the application and the database. This commit implements this new approach.
2024-10-15model: Handle schema versions internally by integersxengineering
This references the database schemas known from the past with integer values internally. 0 is used for an empty database and 1 for what was used in ceres version 0.4.0. The old approach to write the `git describe` output into the database worked well for released versions but was problematic during development. A numerical and separate database schema versioning is easier to handle.
2024-10-13model: Simplify DB method signaturesxengineering
2024-10-13model: Replace global db variable by custom typexengineering
Reducing global variables makes it easier to understand functions independently of the rest of the code. Adding the new model.DB type as a custom variant of the sql.DB type makes it possible to write methods for the database which makes the code way more readable.
2024-10-13Introduce xengineering.eu/ceres/model/migrationsxengineering
This new package is only for database migrations. All data for Ceres should be stored inside the SQLite3 database. Thus migrations can always be executed with functions with the following signature. func(tx *sql.Tx) error Those migration functions should be stored inside the new package. Bigger SQL code can be stored in *.sql files for better syntax highlighting. This code is embedded into the final Go executable by using the embed package.
2024-10-12model: Fix direct access to databasexengineering
Instead a database transaction has to be used. Each database interaction should be wrapped into a transaction to make sure any possible change (even side-effects) can be rolled back in case of errors.
2024-09-11Apply go fmt to all source filesxengineering
This applies default Go code style recommendations.
2024-05-17model: Add strict Ingredient.Validate()xengineering
2024-05-17model: Add strict Step.Validate()xengineering
2024-05-17model: Fix another unhandled errorxengineering
This could also lead to bugs.
2024-05-17model: Fix ignored errorxengineering
An ignored return value here caused a serious bug as soon as validation for ingredients was tried. The validation could raise an error e.g. on a negative amount for the ingredient. This error was ignored at the changed line which resulted into deleted ingredients for the whole recipe.
2024-05-17model: Rework recipe validationxengineering
This reduces code duplication and enforces time stamps.
2024-05-15model: Make Recipe.Validate() more strictxengineering
Before the next release this method should be as strict as possible to avoid cases where actually invalid enters databases.
2024-05-12view: Add ingredient overview to recipe read pagexengineering
2024-05-12view: Show ingredients on read pagexengineering
2024-05-12model: Add per-step ingredientsxengineering
2024-05-09Inject examples only with new --example flagxengineering
The default use case should be to not inject example recipes.
2024-05-09model: Fix unit testsxengineering
2024-05-09model: Rename version to execVersionxengineering
This makes the code easier to understand because there is an executable version and a database version handled inside that file.
2024-05-09model: Require same version for executable and DBxengineering
Currently only an empty database and an existing database with the same version are supported. Support for migrations based on semantic versioning will be added in future versions of Ceres.
2024-05-09model: Migrate only in empty databasesxengineering
2024-05-09model: Initial database version injectionxengineering
If the database was empty on startup a metadata table with a key and value row is created. In addition the Ceres executable version is inserted as value under the key 'version'. This allows to detect on not-empty databases which Ceres version was used before which is the starting point to implement migrations.
2024-05-09model: Detect if database is emptyxengineering
An empty database requires to add the metadata table with the version entry to make migrations possible. Thus this detection will be required.
2024-05-09model: Wrap migration completely in transactionxengineering
This makes it more clear that the full migration will be rolled back on errors.
2024-05-09model: Enforce recipe titlesxengineering
If a recipe has no title it is hard to reference in the front end. Especially the /recipes page makes problems in that case since it is impossible to click on that recipes and thus also to remove it.
2024-05-09model: Use defer for tx.Rollback()xengineering
A committed transaction cannot be rolled back. Using defer to roll back guarantees that the transaction is always rolled back if not the commit in the last line of model.Transaction was excuted. [1]: https://go.dev/doc/database/execute-transactions
2024-05-09Restructure database-related functionsxengineering
2024-05-08Introduce model.Transaction()xengineering
It is a very common pattern that some function needs to access the database and wants to wrap all the actions into one transaction. The advantage of a transaction is that it is ACID: - atomic - consistent - isolated - durable In Go it is required to request a new transaction, execute functionality on it and handle rollback or commit of this transaction based on the success of the operation. All this and the error handling can be written down in the model.Transaction() function exactly once. The full signature of it is: func Transaction(f func(*sql.Tx) error) error It requires a function or closure passed as argument which takes the transaction (*sql.Tx) and returns an error which might be nil. This is very generic. It is applied to: - injecting test data - database migrations - data read requests - data write requests
2024-05-08Fix unit testsxengineering
2024-05-07model: Init database with database pathxengineering
2024-05-06model: Do not write version.txt inside storagexengineering
The intention of this file was that a Ceres executable could compare its version with the version of the storage folder. If the versions match the storage folder could be directly used. If the storage version is lower the executable can apply migrations to the storage folder until the versions match. The problem is that executing migrations inside the database and updating the version.txt cannot be atomic. In contrast the version string could be saved inside the database itself in a metadata table. In that case the migration together with the update of the version string can be executed inside one database transaction which guarantees atomicity. The problem could still be that migrations should be applied also to the files and folders inside the storage folder. This problem can only be avoided by not using files to store data and instead use the BLOB datatype if necessary. Even in case of a future filesystem use it is still better to have the guarantee that the database with file paths and metadata and the there included version string are in sync.
2024-05-06model: Introduce NewStorage() functionxengineering
It is a common pattern inside the Go standard library to provide a constructor with this naming scheme to custom types of the package. Doing this here results in a style closer to the standard library which improves readability.
2024-05-04Move storage path logging to main() functionxengineering
The model package where this used to be implemented should not care too much about logging. Furthermore it is easier to compare the log output with the main() function if the log statements are there.
2024-05-04model: Add storage.Exists() and storage.Create()xengineering
These new methods provide essential functionality related to the storage folder.
2024-05-04model: Introduce type Storagexengineering
This new type definition will make it easier to handle the storage directory of Ceres and related functionality which can be implemented with methods.
2024-05-01Do not remove storage folderxengineering
This is not useful in production. Furthermore in the debug use case the default storage path is now ./storage which can easily be removed by `rm -rf storage`. This also allows to not remove the storage folder for further analysis of the storage folder.
2024-05-01Use default storage path instead of temp dirxengineering
2024-05-01model: Add version.txt file to storage folderxengineering
This prepares the ability to check for compatibility between a Ceres executable build and an existing storage folder.
2024-04-07model: Add helper function for safe CRUDxengineering
This removes the redundant setup of a database/sql.Tx in each HTTP handler.
2024-04-07model: Add model.Object interfacexengineering
This interface will allow to implement generic functions based on the Object interface which covers the four CRUD methods create, read, update and delete. This should be possible for every object handled by the server.
2024-04-06model: CRUD methods only for targeted objectsxengineering
A create, read, update or delete (CRUD) method should only care about the object which provides the receiver and the relations to its child objects. For example the method func (r *Recipe) Create(tx *sql.Tx) error {} should only create the relational data inside the database for the recipe, not for the steps nested into this Recipe struct. This should be covered by the func (s *Step) Create(tx *sql.Tx) error {} method which is then called by `func (r *Recipe) Create()`. This has the advantage that every CRUD method has a constraint scope and is more unified since the Step CRUD methods now have a Step struct as receiver instead of a Recipe receiver.
2024-04-06model: Always pass *sql.Tx to CRUD methodsxengineering
When nesting objects like steps into other objects like recipes it is required to pass a *sql.Tx value to the CRUD methods of the inner object to be able to roll back the whole transaction. The top level object used to be responsible for the creation of this *sql.Tx inside its CRUD methods. This is now moved to the caller of the CRUD methods (here the HTTP handler function). The advantage is that all CRUD methods now accept a *sql.Tx as only argument which makes those methods more consistent.
2024-03-24model: Add recipe stepsxengineering
This provides the infrastructure to create views and HTTP handlers to provide recipe steps.
2024-03-24model: Crash on failed test recipe injectionxengineering
This error used to be silent. Since it is just about test recipes and thus a debugging environment it is best to directly give up and log the error.