summaryrefslogtreecommitdiff
path: root/model/recipe.go
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-09-11Apply go fmt to all source filesxengineering
This applies default Go code style recommendations.
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-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-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-12model: Remove .Touch() methodxengineering
The model package should never modify the data. Thus the functionality to update timestamps is moved to the controller package which is intended to modify data.
2024-03-05model: Move timestamp updates to controllerxengineering
The model package should handle the object relational mapping (ORM). This requires implementing the four CRUD methods: - create - read - update - delete On create and update the model package used to modify the timestamps like `last_changed`. This was detected by the unit tests which tested that the data is not changed in a create / read cycle or is updated correctly in an update / read cycle. This raised the question if it is a good idea that the model package is "smart" and updates timestamps. To keep the model package and the included unit tests simple the new design enforces that the complete data - including metadata - is always exactly the same after using any CRUD methods. The functionality of updating the timestamp is moved to the HTTP handler inside the controller package. This also matches the definition of the controller package as the part of the code which is alone responsible to actually change the data. This commit finally fixes the unit test suite.
2024-03-04model: Implement Stringer interface for Recipexengineering
This allows to print a recipe with a fmt.Printf() call more easily: fmt.Printf("%s\n", recipe) This is also used for better error output in unit tests with t.Fatalf(). The Stringer interface is implemented with the JSON package because an indented version of a recipe is a useful string representation.
2024-03-04model: Create test data with Go instead of SQLxengineering
This allows to formulate the test data with an object-based model which is easier than writing it down in a relational model.
2024-03-03controller: Update recipe based on JSONxengineering
2024-03-03model: Assert one affected row on .Update()xengineering
A missing or wrong .Id field for example otherwise results in a silent error because nothing is actually updated.
2024-03-03model: Use only string types for modelsxengineering
When a HTML form is converted to JSON by JavaScript using `FormData()`, `Object.fromEntries()` and `JSON.stringify` the data type is always `string`. This does not match the Go struct definitions using multiple types including e.g. `int`. There are several options to solve this conflict: 1. use only strings in Go struct definitions 2. write custom functions to parse string-based JSONs to Go structs 3. implement custom functions in JS to use `number` type if possible Option 3 seems to be a very clean solution. Nevertheless it is limited by the fact that JSON anyway has a way more limited type system than Go. So the types used in Go cannot be used and this would reduce this option to a variant of option 2. Option 2 requires significant effort per struct inside the model package. Every object which is transferred via JSON and serialized into Go structs would require a second struct definition with string types and a conversion function. This does not scale. Thus option 1 seems to be the best fit. The reasons for using types like `int` or `bool` are: - less memory consumption than `string` in most cases - implicit data validation (e.g. enforcing positive numbers with `uint`) - better compatibility with certain APIs which rely on e.g. `int` The first argument is not so relevant in this use case. The amount of required memory is still quite small for servers. Implicit data validation is a good thing but not enough. There should anyway be validation method which has to be called on CRUD methods and JSON deserialization.
2024-02-11model: Implement CRUD methods for type Recipexengineering
The new Go type 'Recipe' should contain every information directly related to a recipe. It should be sufficient to pass it to a template to directly render a HTML view or edit page for the recipe or to a template to generate a PDF. The CRUD methods are: - func (r *Recipe) Create() error - func (r *Recipe) Update() error - func (r *Recipe) Read() error - func (r *Recipe) Delete() error Together with the type itself they are the interface the model package provides for recipes.