todo sqlite and inmemory repo

This commit is contained in:
Michael Thomson 2025-02-06 16:33:49 -05:00
parent 21747e0f72
commit bb3d9abb7e
No known key found for this signature in database
GPG Key ID: 8EFECCD347C72F7D
7 changed files with 233 additions and 12 deletions

View File

@ -1,9 +1,61 @@
package main package main
import ( import (
"database/sql"
"fmt"
"log"
"gitea.michaelthomson.dev/mthomson/habits/internal/migrate" "gitea.michaelthomson.dev/mthomson/habits/internal/migrate"
"gitea.michaelthomson.dev/mthomson/habits/internal/models"
repositories "gitea.michaelthomson.dev/mthomson/habits/internal/repositories/todo"
) )
func main() { func main() {
migrate.Migrate(); db, err := sql.Open("sqlite3", "./habits.db")
if err != nil {
log.Fatal("Failed to open db pool")
}
defer db.Close()
// run migrations
migrate.Migrate(db);
repo := repositories.NewSqliteTodoRepository(db)
// create todo
newTodo := &models.Todo{Name: "clean dishes", Done: false}
err = repo.Create(newTodo)
if err != nil {
log.Fatal(err)
}
fmt.Printf("id of created todo: %d\n", newTodo.Id)
// update todo
newTodo.Done = true
err = repo.Update(newTodo)
if err != nil {
log.Fatal(err)
}
fmt.Printf("todo is now done: %t\n", newTodo.Done)
// delete todo
err = repo.Delete(newTodo.Id)
if err != nil {
log.Fatal(err)
}
// get todo
_, err = repo.GetById(newTodo.Id)
if err != nil {
log.Fatal(err)
}
} }

View File

@ -14,21 +14,14 @@ type Migration struct {
Name string Name string
} }
func Migrate() { func Migrate(db *sql.DB) {
db, err := sql.Open("sqlite3", "./habits.db")
if err != nil {
log.Fatal("Failed to open db pool")
}
defer db.Close()
migrationTableSql := ` migrationTableSql := `
CREATE TABLE IF NOT EXISTS migrations( CREATE TABLE IF NOT EXISTS migrations(
version INTEGER PRIMARY KEY, version INTEGER PRIMARY KEY,
name VARCHAR(50) name VARCHAR(50)
);` );`
_, err = db.Exec(migrationTableSql) _, err := db.Exec(migrationTableSql)
if err != nil { if err != nil {
log.Fatal(err) log.Fatal(err)
} }

7
internal/models/todo.go Normal file
View File

@ -0,0 +1,7 @@
package models
type Todo struct {
Id int64
Name string
Done bool
}

View File

@ -0,0 +1,58 @@
package repositories
import "gitea.michaelthomson.dev/mthomson/habits/internal/models"
type InMemoryTodoRepository struct {
db map[int64]*models.Todo
id int64
}
func NewInMemoryTodoRepository() InMemoryTodoRepository {
return InMemoryTodoRepository{
db: make(map[int64]*models.Todo),
id: 1,
}
}
func (r *InMemoryTodoRepository) GetById(id int64) (*models.Todo, error) {
todo, found := r.db[id]
if !found {
return todo, ErrNotFound
}
return todo, nil
}
func (r *InMemoryTodoRepository) Create(todo *models.Todo) error {
todo.Id = r.id
r.db[r.id] = todo
r.id++
return nil
}
func (r *InMemoryTodoRepository) Update(todo *models.Todo) error {
_, found := r.db[todo.Id]
if !found {
return ErrNotFound
}
r.db[todo.Id] = todo
return nil
}
func (r *InMemoryTodoRepository) Delete(id int64) error {
_, found := r.db[id]
if !found {
return ErrNotFound
}
delete(r.db, id)
return nil
}

View File

@ -0,0 +1,92 @@
package repositories
import (
"database/sql"
"gitea.michaelthomson.dev/mthomson/habits/internal/models"
)
type SqliteTodoRepository struct {
db *sql.DB
}
func NewSqliteTodoRepository(db *sql.DB) SqliteTodoRepository {
return SqliteTodoRepository{
db: db,
}
}
func (r *SqliteTodoRepository) GetById(id int64) (*models.Todo, error) {
todo := &models.Todo{}
row := r.db.QueryRow("SELECT * FROM todo WHERE id = ?;", id)
err := row.Scan(todo.Id, todo.Name, todo.Done)
if err != nil {
if err == sql.ErrNoRows {
return todo, ErrNotFound
}
return todo, err
}
return todo, nil
}
func (r *SqliteTodoRepository) Create(todo *models.Todo) error {
result, err := r.db.Exec("INSERT INTO todo (name, done) VALUES (?, ?)", todo.Name, todo.Done)
if err != nil {
return err
}
id, err := result.LastInsertId()
if err != nil {
return err
}
todo.Id = id
return nil
}
func (r *SqliteTodoRepository) Update(todo *models.Todo) error {
result, err := r.db.Exec("UPDATE todo SET name = ?, done = ? WHERE id = ?", todo.Name, todo.Done, todo.Id)
if err != nil {
return err
}
rowsAffected, err := result.RowsAffected()
if err != nil {
return err
}
if rowsAffected == 0 {
return ErrNotFound
}
return nil
}
func (r *SqliteTodoRepository) Delete(id int64) error {
result, err := r.db.Exec("DELETE FROM todo WHERE id = ?", id)
if err != nil {
return err
}
rowsAffected, err := result.RowsAffected()
if err != nil {
return err
}
if rowsAffected == 0 {
return ErrNotFound
}
return nil
}

View File

@ -0,0 +1,18 @@
package repositories
import (
"errors"
"gitea.michaelthomson.dev/mthomson/habits/internal/models"
)
var (
ErrNotFound error = errors.New("Todo cannot be found")
)
type TodoRepository interface {
Create(todo *models.Todo) error
GetById(id int64) (*models.Todo, error)
Update(todo *models.Todo) error
Delete(id int64) error
}

View File

@ -1,4 +1,5 @@
CREATE TABLE todo( CREATE TABLE todo(
id INT PRIMARY KEY, id INTEGER PRIMARY KEY,
name VARCHAR(50) name VARCHAR(50),
done INTEGER
); );