Files
readlist/main.go

218 lines
5.1 KiB
Go

package main
import (
"embed"
"encoding/json"
"fmt"
"html/template"
"log"
"net/http"
"os"
"strconv"
"time"
"github.com/PuerkitoBio/goquery"
"github.com/jmoiron/sqlx"
_ "github.com/mattn/go-sqlite3"
)
type Item struct {
ID int64 `json:"id" db:"id"`
URL string `json:"url" db:"url"`
Title string `json:"title" db:"title"`
Description string `json:"description" db:"description"`
Image *string `json:"image" db:"image"`
CreatedAt time.Time `json:"createdAt" db:"created_at"`
UpdatedAt time.Time `json:"updatedAt" db:"updated_at"`
}
type ArticleMetadata struct {
Title string
Description string
Image *string
}
func getArticleMetadata(url string) (*ArticleMetadata, error) {
resp, err := http.Get(url)
if err != nil {
return nil, fmt.Errorf("failed to query article url: %v", err)
}
defer resp.Body.Close()
if resp.StatusCode < 200 || resp.StatusCode >= 400 {
return &ArticleMetadata{
Title: url,
}, nil
}
doc, err := goquery.NewDocumentFromReader(resp.Body)
if err != nil {
return nil, fmt.Errorf("failed to parse page html: %v", err)
}
title := doc.Find(`head>meta[name="og:title"]`).AttrOr("content", "")
if title == "" {
title = doc.Find(`head>title`).Text()
}
description := doc.Find(`head>meta[name="og:description"]`).AttrOr("content", "")
image := doc.Find(`head>meta[name="og:image"]`).AttrOr("content", "")
if image == "" {
doc.Find("img").Each(func(i int, s *goquery.Selection) {
src := s.AttrOr("src", "")
if src != "" {
image = src
return
}
})
}
return &ArticleMetadata{
Title: title,
Description: description,
Image: &image,
}, nil
}
var (
//go:embed views/*
viewsFS embed.FS
)
func main() {
dbPath := os.Getenv("DB_PATH")
if dbPath == "" {
dbPath = "./sqlite.db"
}
db, err := sqlx.Open("sqlite3", dbPath)
if err != nil {
log.Fatalf("failed to connect to db: %v\n", err)
}
defer db.Close()
db.MustExec(`
CREATE TABLE IF NOT EXISTS items (
id integer primary key not null,
url varchar not null unique,
title varchar not null,
description varchar,
image varchar default null,
created_at datetime default current_timestamp,
updated_at datetime default current_timestamp
);
CREATE TRIGGER IF NOT EXISTS update_items_updated_at
AFTER UPDATE ON items
WHEN old.updated_at <> current_timestamp
BEGIN
UPDATE items
SET updated_at = CURRENT_TIMESTAMP
WHERE id = OLD.id;
END;
`)
tmpl := template.Must(template.ParseFS(viewsFS, "**/*.html"))
mux := http.NewServeMux()
mux.HandleFunc("GET /", func(w http.ResponseWriter, r *http.Request) {
tmpl.ExecuteTemplate(w, "index.html", nil)
})
mux.HandleFunc("POST /items", func(w http.ResponseWriter, r *http.Request) {
pageUrl := r.FormValue("url")
if pageUrl == "" {
http.Error(w, "url field is required", 400)
return
}
meta, err := getArticleMetadata(pageUrl)
if err != nil {
http.Error(w, fmt.Sprintf("failed to fetch page metadata: %v", err), 500)
return
}
if _, err := db.NamedExec("INSERT INTO items (url, title, description, image) VALUES (:url, :title, :description, :image)", map[string]any{
"url": pageUrl,
"title": meta.Title,
"description": meta.Description,
"image": meta.Image,
}); err != nil {
http.Error(w, fmt.Sprintf("failed to add item to db: %v", err), 500)
return
}
http.Redirect(w, r, "/", http.StatusFound)
})
mux.HandleFunc("GET /items", func(w http.ResponseWriter, r *http.Request) {
rows, err := db.Queryx("SELECT * FROM items")
if err != nil {
http.Error(w, fmt.Sprintf("failed to query db for items: %v", err), 500)
return
}
items := []Item{}
for rows.Next() {
item := Item{}
if err := rows.StructScan(&item); err != nil {
continue
}
items = append(items, item)
}
w.Header().Set("Content-Type", "application/json")
if err := json.NewEncoder(w).Encode(items); err != nil {
http.Error(w, err.Error(), 500)
}
})
mux.HandleFunc("GET /items/{id}", func(w http.ResponseWriter, r *http.Request) {
id, err := strconv.Atoi(r.PathValue("id"))
if err != nil {
http.Error(w, err.Error(), 500)
return
}
row := db.QueryRowx("SELECT * FROM items WHERE id = ?", id)
if row.Err() != nil {
http.Error(w, fmt.Sprintf("item not found: %v", row.Err()), 404)
return
}
item := &Item{}
if err := row.StructScan(item); err != nil {
http.Error(w, fmt.Sprintf("failed to scan item to struct: %v", err), 500)
return
}
tmpl.ExecuteTemplate(w, "item.html", struct {
Item *Item
}{
Item: item,
})
})
mux.HandleFunc("POST /items/{id}/delete", func(w http.ResponseWriter, r *http.Request) {
id, err := strconv.Atoi(r.PathValue("id"))
if err != nil {
http.Error(w, err.Error(), 500)
return
}
if _, err := db.Exec("DELETE FROM items WHERE id = ?", id); err != nil {
http.Error(w, fmt.Sprintf("failed to delete item: %v", err), 500)
return
}
http.Redirect(w, r, "/", http.StatusFound)
})
log.Println("starting http server")
if err := http.ListenAndServe(":5000", mux); err != nil {
log.Fatalf("failed to start http server: %v\n", err)
}
}