Initial commit
This commit is contained in:
15
error.go
Normal file
15
error.go
Normal file
@@ -0,0 +1,15 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"net/http"
|
||||
)
|
||||
|
||||
type Error struct {
|
||||
Message string `json:"message"`
|
||||
}
|
||||
|
||||
func sendError(w http.ResponseWriter, code int, msg string, args ...any) {
|
||||
w.WriteHeader(code)
|
||||
sendJSON(w, Error{Message: fmt.Sprintf(msg, args...)})
|
||||
}
|
||||
5
go.mod
Normal file
5
go.mod
Normal file
@@ -0,0 +1,5 @@
|
||||
module github.com/gopatchy/t
|
||||
|
||||
go 1.23.3
|
||||
|
||||
require github.com/lib/pq v1.10.9
|
||||
2
go.sum
Normal file
2
go.sum
Normal file
@@ -0,0 +1,2 @@
|
||||
github.com/lib/pq v1.10.9 h1:YXG7RB+JIjhP29X+OtkiDnYaXQwpS4JEWq7dtCCRUEw=
|
||||
github.com/lib/pq v1.10.9/go.mod h1:AlVN5x4E4T544tWzH6hKfbfQvm3HdbOxrmggDNAPY9o=
|
||||
14
json.go
Normal file
14
json.go
Normal file
@@ -0,0 +1,14 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"net/http"
|
||||
)
|
||||
|
||||
func sendJSON(w http.ResponseWriter, v any) {
|
||||
w.Header().Set("Content-Type", "application/json")
|
||||
|
||||
enc := json.NewEncoder(w)
|
||||
enc.SetEscapeHTML(false)
|
||||
_ = enc.Encode(v)
|
||||
}
|
||||
187
main.go
Normal file
187
main.go
Normal file
@@ -0,0 +1,187 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"database/sql"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"html/template"
|
||||
"log"
|
||||
"net/http"
|
||||
"os"
|
||||
"strings"
|
||||
|
||||
_ "github.com/lib/pq"
|
||||
)
|
||||
|
||||
type Tasks struct {
|
||||
tmpl *template.Template
|
||||
mux *http.ServeMux
|
||||
db *sql.DB
|
||||
}
|
||||
|
||||
func NewTasks(db *sql.DB) (*Tasks, error) {
|
||||
funcMap := template.FuncMap{
|
||||
"lower": strings.ToLower,
|
||||
"join": strings.Join,
|
||||
}
|
||||
|
||||
tmpl, err := template.New("index.html").Funcs(funcMap).ParseFiles("static/index.html")
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("static/index.html: %w", err)
|
||||
}
|
||||
|
||||
t := &Tasks{
|
||||
tmpl: tmpl,
|
||||
mux: http.NewServeMux(),
|
||||
db: db,
|
||||
}
|
||||
|
||||
t.mux.HandleFunc("GET /{$}", t.serveRoot)
|
||||
|
||||
return t, nil
|
||||
}
|
||||
|
||||
func (t *Tasks) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
||||
t.mux.ServeHTTP(w, r)
|
||||
}
|
||||
|
||||
func (t *Tasks) serveRoot(w http.ResponseWriter, r *http.Request) {
|
||||
err := t.initRequest(w, r)
|
||||
if err != nil {
|
||||
sendError(w, http.StatusBadRequest, "init request: %s", err)
|
||||
return
|
||||
}
|
||||
|
||||
err = t.tmpl.Execute(w, map[string]any{})
|
||||
if err != nil {
|
||||
sendError(w, http.StatusInternalServerError, "error executing template: %s", err)
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
func (t *Tasks) initRequest(w http.ResponseWriter, r *http.Request) error {
|
||||
w.Header().Set("Access-Control-Allow-Origin", "*")
|
||||
w.Header().Set("Access-Control-Allow-Methods", "GET, POST, QUERY, OPTIONS")
|
||||
w.Header().Set("Access-Control-Allow-Headers", "*")
|
||||
|
||||
defer r.Body.Close()
|
||||
|
||||
err := r.ParseForm()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if r.Header.Get("Content-Type") == "application/json" {
|
||||
dec := json.NewDecoder(r.Body)
|
||||
js := map[string]any{}
|
||||
err := dec.Decode(&js)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
for k, v := range js {
|
||||
switch v := v.(type) {
|
||||
case []any:
|
||||
for _, s := range v {
|
||||
r.Form.Add(k, fmt.Sprintf("%v", s))
|
||||
}
|
||||
|
||||
default:
|
||||
r.Form.Set(k, fmt.Sprintf("%v", v))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
log.Printf("%s %s %s %s %s %#v", r.RemoteAddr, r.Method, r.Host, r.URL, r.Form)
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func main() {
|
||||
port := os.Getenv("PORT")
|
||||
if port == "" {
|
||||
log.Fatalf("please set PORT")
|
||||
}
|
||||
|
||||
pgConn := os.Getenv("PGCONN")
|
||||
if pgConn == "" {
|
||||
log.Fatalf("please set PGCONN")
|
||||
}
|
||||
|
||||
db, err := sql.Open("postgres", pgConn)
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
|
||||
/*
|
||||
stmts := []string{
|
||||
`
|
||||
CREATE TABLE IF NOT EXISTS links (
|
||||
short VARCHAR(100) NOT NULL,
|
||||
long TEXT NOT NULL,
|
||||
domain VARCHAR(255) NOT NULL,
|
||||
generated BOOLEAN NOT NULL,
|
||||
PRIMARY KEY (short, domain)
|
||||
);
|
||||
`,
|
||||
|
||||
`
|
||||
CREATE TABLE IF NOT EXISTS links_history (
|
||||
short VARCHAR(100),
|
||||
long TEXT NOT NULL,
|
||||
domain VARCHAR(255) NOT NULL,
|
||||
generated BOOLEAN NOT NULL,
|
||||
until TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP
|
||||
);
|
||||
`,
|
||||
|
||||
`
|
||||
CREATE OR REPLACE FUNCTION update_link(
|
||||
_short VARCHAR(100),
|
||||
_long TEXT,
|
||||
_domain VARCHAR(255),
|
||||
_generated BOOLEAN
|
||||
) RETURNS void AS $$
|
||||
DECLARE
|
||||
old RECORD;
|
||||
BEGIN
|
||||
SELECT * INTO old FROM links WHERE short = _short AND domain = _domain;
|
||||
|
||||
IF old IS NOT NULL THEN
|
||||
INSERT INTO links_history (short, long, domain, generated)
|
||||
VALUES (old.short, old.long, old.domain, old.generated);
|
||||
|
||||
UPDATE links
|
||||
SET long = _long, generated = _generated
|
||||
WHERE short = _short AND domain = _domain;
|
||||
ELSE
|
||||
INSERT INTO links (short, long, domain, generated)
|
||||
VALUES (_short, _long, _domain, _generated);
|
||||
END IF;
|
||||
END;
|
||||
$$ LANGUAGE plpgsql;
|
||||
`,
|
||||
}
|
||||
|
||||
for _, stmt := range stmts {
|
||||
_, err := db.Exec(stmt)
|
||||
if err != nil {
|
||||
log.Fatalf("Failed to create tables & functions: %v", err)
|
||||
}
|
||||
}
|
||||
*/
|
||||
|
||||
t, err := NewTasks(db)
|
||||
if err != nil {
|
||||
log.Fatalf("failed to create tasks: %v", err)
|
||||
}
|
||||
|
||||
http.Handle("/", t)
|
||||
|
||||
bind := fmt.Sprintf(":%s", port)
|
||||
log.Printf("listening on %s", bind)
|
||||
|
||||
if err := http.ListenAndServe(bind, nil); err != nil {
|
||||
log.Fatalf("listen: %s", err)
|
||||
}
|
||||
}
|
||||
0
static/index.html
Normal file
0
static/index.html
Normal file
Reference in New Issue
Block a user