Add Library Pages
This commit is contained in:
parent
566b9d8667
commit
d6fa1fc97a
5 changed files with 235 additions and 0 deletions
80
server/libraries.go
Normal file
80
server/libraries.go
Normal file
|
@ -0,0 +1,80 @@
|
|||
package server
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"fmt"
|
||||
"log/slog"
|
||||
"net/http"
|
||||
|
||||
"git.lastassault.de/speatzle/morffix/constants"
|
||||
"github.com/jackc/pgx/v5"
|
||||
)
|
||||
|
||||
type LibrariesData struct {
|
||||
Libraries []Library
|
||||
}
|
||||
|
||||
type Library struct {
|
||||
ID string `db:"id"`
|
||||
Name string `db:"name"`
|
||||
Path string `db:"path"`
|
||||
Enable bool `db:"enable"`
|
||||
}
|
||||
|
||||
func handleLibraries(w http.ResponseWriter, r *http.Request) {
|
||||
data := LibrariesData{}
|
||||
|
||||
if r.Method == "POST" {
|
||||
err := createLibrary(r)
|
||||
if err != nil {
|
||||
slog.ErrorContext(r.Context(), "Create Library", "err", err)
|
||||
http.Error(w, "Error Create Library: "+err.Error(), http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
rows, err := db.Query(r.Context(), "SELECT id, name, path, enable FROM libraries")
|
||||
if err != nil {
|
||||
slog.ErrorContext(r.Context(), "Query Libraries", "err", err)
|
||||
http.Error(w, "Error Query Libraries: "+err.Error(), http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
libraries, err := pgx.CollectRows[Library](rows, pgx.RowToStructByName[Library])
|
||||
if err != nil {
|
||||
slog.ErrorContext(r.Context(), "Collect Rows", "err", err)
|
||||
http.Error(w, "Error Query Libraries: "+err.Error(), http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
data.Libraries = libraries
|
||||
|
||||
buf := bytes.Buffer{}
|
||||
err = templates.ExecuteTemplate(&buf, constants.LIBRARIES_TEMPLATE_NAME, data)
|
||||
if err != nil {
|
||||
slog.ErrorContext(r.Context(), "Executing Libraries Template", "err", err)
|
||||
http.Error(w, "Error Executing Template: "+err.Error(), http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
|
||||
_, err = w.Write(buf.Bytes())
|
||||
if err != nil {
|
||||
slog.ErrorContext(r.Context(), "Writing http Response", "err", err)
|
||||
}
|
||||
}
|
||||
|
||||
func createLibrary(r *http.Request) error {
|
||||
err := r.ParseForm()
|
||||
if err != nil {
|
||||
return fmt.Errorf("Parseing Form: %w", err)
|
||||
}
|
||||
name := r.FormValue("name")
|
||||
path := r.FormValue("path")
|
||||
enable := r.FormValue("enable") == "on"
|
||||
slog.Info("Got Library Create", "name", name, "path", path, "enable", enable)
|
||||
|
||||
_, err = db.Exec(r.Context(), "INSERT INTO Libraries (name, path, enable) VALUES ($1,$2,$3)", name, path, enable)
|
||||
if err != nil {
|
||||
return fmt.Errorf("Inserting Library: %w", err)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
79
server/library.go
Normal file
79
server/library.go
Normal file
|
@ -0,0 +1,79 @@
|
|||
package server
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"log/slog"
|
||||
"net/http"
|
||||
|
||||
"git.lastassault.de/speatzle/morffix/constants"
|
||||
"github.com/jackc/pgx/v5"
|
||||
)
|
||||
|
||||
type LibraryData struct {
|
||||
Library Library
|
||||
Files []File
|
||||
}
|
||||
|
||||
type File struct {
|
||||
ID int `db:"id"`
|
||||
Path string `db:"path"`
|
||||
Size int64 `db:"size"`
|
||||
Missing bool `db:"missing"`
|
||||
}
|
||||
|
||||
func handleLibrary(w http.ResponseWriter, r *http.Request) {
|
||||
id := r.PathValue("id")
|
||||
if id == "" {
|
||||
handleLibraries(w, r)
|
||||
return
|
||||
}
|
||||
|
||||
data := LibraryData{}
|
||||
|
||||
var name string
|
||||
var path string
|
||||
var enabled bool
|
||||
err := db.QueryRow(r.Context(), "SELECT name, path, enable FROM libraries WHERE id = $1", id).Scan(&name, &path, &enabled)
|
||||
if err != nil {
|
||||
slog.ErrorContext(r.Context(), "Get Library", "err", err)
|
||||
http.Error(w, "Error Get Library: "+err.Error(), http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
data.Library = Library{
|
||||
ID: id,
|
||||
Name: name,
|
||||
Path: path,
|
||||
Enable: enabled,
|
||||
}
|
||||
|
||||
if r.Method == "PUT" {
|
||||
// TODO
|
||||
}
|
||||
|
||||
rows, err := db.Query(r.Context(), "SELECT id, path, size, missing FROM files where library_id = $1", id)
|
||||
if err != nil {
|
||||
slog.ErrorContext(r.Context(), "Query Files", "err", err)
|
||||
http.Error(w, "Error Query Files: "+err.Error(), http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
files, err := pgx.CollectRows[File](rows, pgx.RowToStructByName[File])
|
||||
if err != nil {
|
||||
slog.ErrorContext(r.Context(), "Collect Rows", "err", err)
|
||||
http.Error(w, "Error Query Files: "+err.Error(), http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
data.Files = files
|
||||
|
||||
buf := bytes.Buffer{}
|
||||
err = templates.ExecuteTemplate(&buf, constants.LIBRARY_TEMPLATE_NAME, data)
|
||||
if err != nil {
|
||||
slog.ErrorContext(r.Context(), "Executing Library Template", "err", err)
|
||||
http.Error(w, "Error Executing Template: "+err.Error(), http.StatusInternalServerError)
|
||||
return
|
||||
}
|
||||
|
||||
_, err = w.Write(buf.Bytes())
|
||||
if err != nil {
|
||||
slog.ErrorContext(r.Context(), "Writing http Response", "err", err)
|
||||
}
|
||||
}
|
|
@ -71,6 +71,8 @@ func Start(_conf config.Config, tmplFS embed.FS, staticFS embed.FS, migrationsFS
|
|||
mux := http.NewServeMux()
|
||||
mux.HandleFunc("/worker", handleWorkerWebsocket)
|
||||
mux.Handle("/static/", fs)
|
||||
mux.HandleFunc("/libraries/{id}", handleLibrary)
|
||||
mux.HandleFunc("/libraries", handleLibraries)
|
||||
mux.HandleFunc("/", handleIndex)
|
||||
|
||||
server := &http.Server{
|
||||
|
|
40
tmpl/libraries.tmpl
Normal file
40
tmpl/libraries.tmpl
Normal file
|
@ -0,0 +1,40 @@
|
|||
{{template "head"}}
|
||||
<h2>Libraries</h2>
|
||||
<div class="library-list">
|
||||
<table>
|
||||
<tr>
|
||||
<th>ID</th>
|
||||
<th>Name</th>
|
||||
<th>Path</th>
|
||||
<th>Enable</th>
|
||||
</tr>
|
||||
{{range $l := .Libraries}}
|
||||
<tr onclick="window.location='/libraries/{{ $l.ID }}';">
|
||||
<td>
|
||||
{{ $l.ID }}
|
||||
</td>
|
||||
<td>
|
||||
{{ $l.Name }}
|
||||
</td>
|
||||
<td>
|
||||
{{ $l.Path }}
|
||||
</td>
|
||||
<td>
|
||||
{{ $l.Enable }}
|
||||
</td>
|
||||
</tr>
|
||||
{{end}}
|
||||
</table>
|
||||
</div>
|
||||
|
||||
<h2>New Library</h2>
|
||||
<form method="POST">
|
||||
<label>Name:</label><br />
|
||||
<input type="text" name="name"><br />
|
||||
<label>Path:</label><br />
|
||||
<input type="text" name="path"><br />
|
||||
<label>Enable:</label><br />
|
||||
<input type="checkbox" name="enable"><br />
|
||||
<input type="submit">
|
||||
</form>
|
||||
{{template "tail"}}
|
34
tmpl/library.tmpl
Normal file
34
tmpl/library.tmpl
Normal file
|
@ -0,0 +1,34 @@
|
|||
{{template "head"}}
|
||||
<h2>Library {{.Library.Name}}</h2>
|
||||
<p>Enabled {{.Library.Enable}}
|
||||
<h2>Files</h2>
|
||||
<form method="POST" action="/scan/{{.Library.ID}}">
|
||||
<input value="Scan" type="submit">
|
||||
</form>
|
||||
<div class="file-list">
|
||||
<table>
|
||||
<tr>
|
||||
<th>ID</th>
|
||||
<th>Path</th>
|
||||
<th>Size</th>
|
||||
<th>Missing</th>
|
||||
</tr>
|
||||
{{range $f := .Files}}
|
||||
<tr>
|
||||
<td>
|
||||
{{ $f.ID }}
|
||||
</td>
|
||||
<td>
|
||||
{{ $f.Path }}
|
||||
</td>
|
||||
<td>
|
||||
{{ $f.Size }}
|
||||
</td>
|
||||
<td>
|
||||
{{ $f.Missing }}
|
||||
</td>
|
||||
</tr>
|
||||
{{end}}
|
||||
</table>
|
||||
</div>
|
||||
{{template "tail"}}
|
Loading…
Add table
Reference in a new issue