summaryrefslogtreecommitdiff
path: root/model
diff options
context:
space:
mode:
authorxengineering <me@xengineering.eu>2024-03-03 07:56:39 +0100
committerxengineering <me@xengineering.eu>2024-03-03 13:50:58 +0100
commit110f9db31a2a2b72031292558f7b1d5bab7649c3 (patch)
treef34d8bb30322e40a696dd6f601a65520f78e6686 /model
parent38ee758c3d553faae683169eb74f2e6de0ab4207 (diff)
downloadceres-110f9db31a2a2b72031292558f7b1d5bab7649c3.tar
ceres-110f9db31a2a2b72031292558f7b1d5bab7649c3.tar.zst
ceres-110f9db31a2a2b72031292558f7b1d5bab7649c3.zip
model: Use only string types for models
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.
Diffstat (limited to 'model')
-rw-r--r--model/recipe.go16
1 files changed, 9 insertions, 7 deletions
diff --git a/model/recipe.go b/model/recipe.go
index 1510379..e5cf0a6 100644
--- a/model/recipe.go
+++ b/model/recipe.go
@@ -3,23 +3,24 @@ package model
import (
"database/sql"
"errors"
+ "fmt"
"time"
)
type Recipe struct {
- Id int64 `json:"id"`
+ Id string `json:"id"`
Title string `json:"title"`
- Portions int `json:"portions"`
+ Portions string `json:"portions"`
Url string `json:"url"`
Notes string `json:"notes"`
- Created int64 `json:"created"`
- LastChanged int64 `json:"last_changed"`
+ Created string `json:"created"`
+ LastChanged string `json:"last_changed"`
}
func (r *Recipe) Create() error {
now := time.Now().Unix()
- r.Created = now
- r.LastChanged = now
+ r.Created = fmt.Sprint(now)
+ r.LastChanged = r.Created
query := `INSERT INTO recipes
(title, portions, url, notes, created, last_changed)
@@ -32,10 +33,11 @@ VALUES
return err
}
- r.Id, err = result.LastInsertId()
+ id, err := result.LastInsertId()
if err != nil {
return err
}
+ r.Id = fmt.Sprint(id)
return nil
}