todo sqlite and inmemory repo
This commit is contained in:
parent
21747e0f72
commit
bb3d9abb7e
54
cmd/main.go
54
cmd/main.go
@ -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)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -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
7
internal/models/todo.go
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
package models
|
||||||
|
|
||||||
|
type Todo struct {
|
||||||
|
Id int64
|
||||||
|
Name string
|
||||||
|
Done bool
|
||||||
|
}
|
58
internal/repositories/todo/inmemorytodorepository.go
Normal file
58
internal/repositories/todo/inmemorytodorepository.go
Normal 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
|
||||||
|
}
|
92
internal/repositories/todo/sqlitetodorepository.go
Normal file
92
internal/repositories/todo/sqlitetodorepository.go
Normal 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
|
||||||
|
}
|
18
internal/repositories/todo/todorepository.go
Normal file
18
internal/repositories/todo/todorepository.go
Normal 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
|
||||||
|
}
|
@ -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
|
||||||
);
|
);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user