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