feat: added endpoint for ingredients and ingredeients per recipe
This commit is contained in:
parent
3c98c82f8a
commit
083b2fcaa3
210
server/main.go
210
server/main.go
|
@ -1,46 +1,31 @@
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"database/sql"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
"net/http"
|
"net/http"
|
||||||
"encoding/json"
|
|
||||||
"os"
|
"os"
|
||||||
"github.com/golang-jwt/jwt"
|
|
||||||
"time"
|
|
||||||
"database/sql"
|
|
||||||
_ "github.com/lib/pq"
|
_ "github.com/lib/pq"
|
||||||
)
|
)
|
||||||
|
|
||||||
func GenerateJwt(username string) (string, error) {
|
|
||||||
token := jwt.New(jwt.SigningMethodEdDSA)
|
|
||||||
|
|
||||||
claims := token.Claims.(jwt.MapClaims)
|
|
||||||
claims["exp"] = time.Now().Add(10 * time.Minute)
|
|
||||||
claims["authorized"] = true
|
|
||||||
claims["user"] = username
|
|
||||||
|
|
||||||
tokenString, err := token.SignedString(os.Getenv("GREPFOOD_JWT_SECRET"))
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
|
|
||||||
return tokenString, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func WriteJson(w http.ResponseWriter, v any) {
|
func WriteJson(w http.ResponseWriter, v any) {
|
||||||
w.Header().Set("Content-Type", "application/json")
|
w.Header().Set("Content-Type", "application/json")
|
||||||
|
|
||||||
jsonData, err := json.Marshal(v)
|
jsonData, err := json.Marshal(v)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
http.Error(w, err.Error(), http.StatusInternalServerError)
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
w.Write(jsonData)
|
w.Write(jsonData)
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleVersion(w http.ResponseWriter, r *http.Request) {
|
func handleVersion(w http.ResponseWriter, r *http.Request) {
|
||||||
version := struct {Version string `json:version`} {
|
version := struct {
|
||||||
|
Version string `json:version`
|
||||||
|
}{
|
||||||
Version: "0.1.0-demo",
|
Version: "0.1.0-demo",
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -49,7 +34,7 @@ func handleVersion(w http.ResponseWriter, r *http.Request) {
|
||||||
|
|
||||||
func InitDatabase(db *sql.DB) {
|
func InitDatabase(db *sql.DB) {
|
||||||
_, err := db.Exec(
|
_, err := db.Exec(
|
||||||
`CREATE TABLE IF NOT EXISTS ingredient (
|
`CREATE TABLE IF NOT EXISTS ingredient (
|
||||||
id SERIAL,
|
id SERIAL,
|
||||||
name VARCHAR(64) UNIQUE NOT NULL,
|
name VARCHAR(64) UNIQUE NOT NULL,
|
||||||
icon VARCHAR(64),
|
icon VARCHAR(64),
|
||||||
|
@ -63,7 +48,7 @@ func InitDatabase(db *sql.DB) {
|
||||||
log.Print("created table for ingredients")
|
log.Print("created table for ingredients")
|
||||||
|
|
||||||
_, err = db.Exec(
|
_, err = db.Exec(
|
||||||
`CREATE TABLE IF NOT EXISTS recipe (
|
`CREATE TABLE IF NOT EXISTS recipe (
|
||||||
id SERIAL,
|
id SERIAL,
|
||||||
name VARCHAR(64) UNIQUE NOT NULL,
|
name VARCHAR(64) UNIQUE NOT NULL,
|
||||||
icon VARCHAR(64),
|
icon VARCHAR(64),
|
||||||
|
@ -76,7 +61,7 @@ func InitDatabase(db *sql.DB) {
|
||||||
log.Print("created table for receipts")
|
log.Print("created table for receipts")
|
||||||
|
|
||||||
_, err = db.Exec(
|
_, err = db.Exec(
|
||||||
`CREATE TABLE IF NOT EXISTS recipe_ingredient (
|
`CREATE TABLE IF NOT EXISTS recipe_ingredient (
|
||||||
recipe_id INT NOT NULL,
|
recipe_id INT NOT NULL,
|
||||||
ingredient_d INT NOT NULL,
|
ingredient_d INT NOT NULL,
|
||||||
amount INT,
|
amount INT,
|
||||||
|
@ -91,9 +76,9 @@ func InitDatabase(db *sql.DB) {
|
||||||
}
|
}
|
||||||
|
|
||||||
type Ingredient struct {
|
type Ingredient struct {
|
||||||
Name string `json:name`
|
Name string `json:name`
|
||||||
Icon string `json:icon`
|
Icon string `json:icon`
|
||||||
Price int `json:price`
|
Price int `json:price`
|
||||||
}
|
}
|
||||||
|
|
||||||
type Recipe struct {
|
type Recipe struct {
|
||||||
|
@ -102,9 +87,145 @@ type Recipe struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
type RecipeIngredient struct {
|
type RecipeIngredient struct {
|
||||||
Recipe int `json:recipe`
|
Recipe int `json:recipe`
|
||||||
Ingredient int `json:ingredient`
|
Ingredient int `json:ingredient`
|
||||||
Amount int `json:amount`
|
Amount int `json:amount`
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetRecipes(db *sql.DB) ([]Recipe, error) {
|
||||||
|
rows, err := db.Query("SELECT name, icon FROM recipe")
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
recipes := []Recipe{}
|
||||||
|
|
||||||
|
for rows.Next() {
|
||||||
|
recipe := Recipe{}
|
||||||
|
rows.Scan(&recipe.Name, &recipe.Icon)
|
||||||
|
recipes = append(recipes, recipe)
|
||||||
|
}
|
||||||
|
|
||||||
|
return recipes, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetIngredients(db *sql.DB) ([]Ingredient, error) {
|
||||||
|
rows, err := db.Query("SELECT name, icon, price FROM ingredient")
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
ingredients := []Ingredient{}
|
||||||
|
|
||||||
|
for rows.Next() {
|
||||||
|
ingredient := Ingredient{}
|
||||||
|
rows.Scan(&ingredient.Name, &ingredient.Icon, &ingredient.Price)
|
||||||
|
ingredients = append(ingredients, ingredient)
|
||||||
|
}
|
||||||
|
|
||||||
|
return ingredients, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetRecipeIngredients(recipe_name string, db *sql.DB) ([]Ingredient, error) {
|
||||||
|
rows, err := db.Query(fmt.Sprintf("SELECT id FROM recipe WHERE name='%s'", recipe_name))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
id := 0
|
||||||
|
if rows.Next() {
|
||||||
|
rows.Scan(&id)
|
||||||
|
} else {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
rows, err = db.Query(fmt.Sprintf("SELECT ingredient.name, ingredient.icon, ingredient.price FROM ingredient INNER JOIN recipe_ingredient ON recipe_ingredient.ingredient_d=ingredient.id WHERE recipe_ingredient.recipe_id=%d", id))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
ingredients := []Ingredient{}
|
||||||
|
|
||||||
|
for rows.Next() {
|
||||||
|
ingredient := Ingredient{}
|
||||||
|
rows.Scan(&ingredient.Name, &ingredient.Icon, &ingredient.Price)
|
||||||
|
ingredients = append(ingredients, ingredient)
|
||||||
|
}
|
||||||
|
|
||||||
|
return ingredients, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func HandleApiRecipes(w http.ResponseWriter, r *http.Request, db *sql.DB) {
|
||||||
|
if r.Method != http.MethodGet && r.Method != "" {
|
||||||
|
http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)
|
||||||
|
log.Print("Mismatched method: ", r.Method)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
recipes, err := GetRecipes(db)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, "Internal server error", http.StatusInternalServerError)
|
||||||
|
log.Print("Error fetching recipes: ", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
WriteJson(w, recipes)
|
||||||
|
|
||||||
|
log.Print("Fetched ", len(recipes), " recipes from database")
|
||||||
|
}
|
||||||
|
|
||||||
|
func HandleApiIngredients(w http.ResponseWriter, r *http.Request, db *sql.DB) {
|
||||||
|
if r.Method != http.MethodGet && r.Method != "" {
|
||||||
|
http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)
|
||||||
|
log.Print("Mismatched method: ", r.Method)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
ingredients, err := GetIngredients(db)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, "Internal server error", http.StatusInternalServerError)
|
||||||
|
log.Print("Error fetching ingredients: ", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Print("Fetched ", len(ingredients), " ingredients from database")
|
||||||
|
|
||||||
|
WriteJson(w, ingredients)
|
||||||
|
}
|
||||||
|
|
||||||
|
func HandleApiRecipeIngredients(w http.ResponseWriter, r *http.Request, db *sql.DB) {
|
||||||
|
if r.Method != http.MethodGet && r.Method != "" {
|
||||||
|
http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)
|
||||||
|
log.Print("Mismatched method: ", r.Method)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
recipe_name := r.PathValue("name")
|
||||||
|
log.Print("Fetching ingredients for recipe ", recipe_name)
|
||||||
|
|
||||||
|
ingredients, err := GetRecipeIngredients(recipe_name, db)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, "Internal server error", http.StatusInternalServerError)
|
||||||
|
log.Print("Error fetching ingredients: ", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if ingredients == nil {
|
||||||
|
http.NotFound(w, r)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
WriteJson(w, ingredients)
|
||||||
|
}
|
||||||
|
|
||||||
|
func HandleApiIngedientAdd(w http.ResponseWriter, r *http.Request, db *sql.DB) {
|
||||||
|
if r.Method != http.MethodGet && r.Method != "" {
|
||||||
|
http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)
|
||||||
|
log.Print("Mismatched method: ", r.Method)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
|
@ -112,31 +233,22 @@ func main() {
|
||||||
|
|
||||||
db, err := sql.Open("postgres", dbconn)
|
db, err := sql.Open("postgres", dbconn)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
log.Print("connected to database: ", dbconn)
|
log.Print("connected to database: ", dbconn)
|
||||||
|
|
||||||
InitDatabase(db)
|
InitDatabase(db)
|
||||||
|
|
||||||
http.HandleFunc("/version", handleVersion)
|
http.HandleFunc("/version", handleVersion)
|
||||||
|
http.Handle("/", http.FileServer(http.Dir("static")))
|
||||||
http.HandleFunc("/api/recipes", func(w http.ResponseWriter, r *http.Request) {
|
http.HandleFunc("/api/recipes", func(w http.ResponseWriter, r *http.Request) {
|
||||||
rows, err := db.Query("SELECT name, icon FROM recipe")
|
HandleApiRecipes(w, r, db)
|
||||||
if err != nil {
|
})
|
||||||
http.Error(w, "Internal server error", http.StatusInternalServerError)
|
http.HandleFunc("/api/ingredients", func(w http.ResponseWriter, r *http.Request) {
|
||||||
return
|
HandleApiIngredients(w, r, db)
|
||||||
}
|
})
|
||||||
|
http.HandleFunc("/api/recipe/{name}/ingredients", func(w http.ResponseWriter, r *http.Request) {
|
||||||
recipes := struct { Recipes []Recipe `json:recipes`} {
|
HandleApiRecipeIngredients(w, r, db)
|
||||||
Recipes: []Recipe{},
|
|
||||||
}
|
|
||||||
|
|
||||||
for rows.Next() {
|
|
||||||
recipe := Recipe{}
|
|
||||||
rows.Scan(&recipe.Name, &recipe.Icon)
|
|
||||||
recipes.Recipes = append(recipes.Recipes, recipe)
|
|
||||||
}
|
|
||||||
|
|
||||||
WriteJson(w, recipes)
|
|
||||||
})
|
})
|
||||||
|
|
||||||
log.Fatal(http.ListenAndServe(os.Getenv("GREPFOOD_SYNCSERVER_PORT"), nil))
|
log.Fatal(http.ListenAndServe(os.Getenv("GREPFOOD_SYNCSERVER_PORT"), nil))
|
||||||
|
|
|
@ -0,0 +1,14 @@
|
||||||
|
|
||||||
|
INSERT INTO grepfood.public.ingredient (name, icon, price)
|
||||||
|
VALUES
|
||||||
|
('potato', 'potato.png', 399),
|
||||||
|
('tomato', 'tomato.png', 199),
|
||||||
|
('flour', 'flour.png', 050),
|
||||||
|
('water', 'waterglass.png', 005);
|
||||||
|
|
||||||
|
INSERT INTO grepfood.public.recipe (name, icon)
|
||||||
|
VALUES
|
||||||
|
('bread', 'bread.png'),
|
||||||
|
('pizza', 'pizza.png'),
|
||||||
|
('cookie', 'cookie.png'),
|
||||||
|
('pasta', 'pasta.png');
|
Loading…
Reference in New Issue