commit f796973fbdac84b750977c5b278d609b9eaf2ec9 Author: qvalentin Date: Tue Mar 4 20:22:07 2025 +0100 feat: init project diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..ac76687 --- /dev/null +++ b/.gitignore @@ -0,0 +1,8 @@ +wormspace +wormspace.db +/bin +/dist +__debug_bin* +.vscode + +.coverage diff --git a/go.mod b/go.mod new file mode 100644 index 0000000..ea424fb --- /dev/null +++ b/go.mod @@ -0,0 +1,8 @@ +module tea.filefighter.de/qvalentin/wormspace + +go 1.23.5 + +require ( + github.com/mattn/go-sqlite3 v1.14.24 // indirect + golang.org/x/net v0.35.0 +) diff --git a/go.sum b/go.sum new file mode 100644 index 0000000..f8711d8 --- /dev/null +++ b/go.sum @@ -0,0 +1,4 @@ +github.com/mattn/go-sqlite3 v1.14.24 h1:tpSp2G2KyMnnQu99ngJ47EIkWVmliIizyZBfPrBWDRM= +github.com/mattn/go-sqlite3 v1.14.24/go.mod h1:Uh1q+B4BYcTPb+yiD3kU8Ct7aC0hY9fxUwlHK0RXw+Y= +golang.org/x/net v0.35.0 h1:T5GQRQb2y08kTAByq9L4/bz8cipCdA8FbRTXewonqY8= +golang.org/x/net v0.35.0/go.mod h1:EglIi67kWsHKlRzzVMUD93VMSWGFOMSZgxFjparz1Qk= diff --git a/main.go b/main.go new file mode 100644 index 0000000..b35ac08 --- /dev/null +++ b/main.go @@ -0,0 +1,298 @@ +package main + +import ( + "bytes" + "database/sql" + "fmt" + "strings" + "unicode" + + // "html/template" + "net/http" + "text/template" + + _ "github.com/mattn/go-sqlite3" + "golang.org/x/net/html" +) + +type User struct { + ID int + Name string + Posts []string + Heros []string +} + +var db *sql.DB + +func initDB() { + var err error + db, err = sql.Open("sqlite3", "wormspace.db") + if err != nil { + panic(err) + } + + _, err = db.Exec(`CREATE TABLE IF NOT EXISTS users ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + name TEXT UNIQUE + );`) + if err != nil { + panic(err) + } + + _, err = db.Exec(`CREATE TABLE IF NOT EXISTS posts ( + id INTEGER PRIMARY KEY AUTOINCREMENT, + user_id INTEGER, + content TEXT, + FOREIGN KEY(user_id) REFERENCES users(id) + );`) + if err != nil { + panic(err) + } + + _, err = db.Exec(`CREATE TABLE IF NOT EXISTS heros ( + user_id INTEGER, + hero_id INTEGER, + PRIMARY KEY(user_id, hero_id), + FOREIGN KEY(user_id) REFERENCES users(id), + FOREIGN KEY(hero_id) REFERENCES users(id) + );`) + if err != nil { + panic(err) + } +} + +func setUser(w http.ResponseWriter, r *http.Request) { + username := r.FormValue("username") + if username == "" { + http.Redirect(w, r, "/", http.StatusSeeOther) + return + } + + for _, c := range username { + if !unicode.IsLetter(c) { + http.Error(w, "Invalid username", http.StatusBadRequest) + return + } + } + + _, err := db.Exec("INSERT OR IGNORE INTO users (name) VALUES (?)", username) + if err != nil { + http.Error(w, "Database error", http.StatusInternalServerError) + return + } + + http.SetCookie(w, &http.Cookie{Name: "username", Value: username, Path: "/"}) + http.Redirect(w, r, "/home", http.StatusSeeOther) +} + +func getUser(r *http.Request) *User { + cookie, err := r.Cookie("username") + if err != nil { + return nil + } + return &User{Name: cookie.Value} +} + +func getPosts(username string) []string { + rows, _ := db.Query("SELECT content FROM posts WHERE user_id = (SELECT id FROM users WHERE name = ?)", username) + var posts []string + for rows.Next() { + var content string + rows.Scan(&content) + posts = append(posts, content) + } + return posts +} + +func getHeros(username string) []string { + rows, _ := db.Query("SELECT name FROM users WHERE id IN (SELECT hero_id FROM heros WHERE user_id = (SELECT id FROM users WHERE name = ?))", username) + var heros []string + for rows.Next() { + var name string + rows.Scan(&name) + heros = append(heros, name) + } + return heros +} + +// sanitizeHTML filters the input, allowing only and tags with "style" and "onload" attributes. +func sanitizeHTML(input string) string { + doc, err := html.Parse(strings.NewReader(input)) + if err != nil { + return "" + } + + var buf bytes.Buffer + processNode(&buf, doc) + str := buf.String() + cleaned := strings.ReplaceAll(str, "onreadystatechange", "") + + return cleaned +} + +// processNode recursively processes nodes, allowing only specific elements and attributes. +func processNode(buf *bytes.Buffer, n *html.Node) { + if n.Type == html.ElementNode { + if n.Data != "img" && n.Data != "a" { + // Skip non-allowed tags but still process children + for c := n.FirstChild; c != nil; c = c.NextSibling { + processNode(buf, c) + } + return + } + + // Start tag + buf.WriteString("<" + n.Data) + + // Filter attributes + for _, attr := range n.Attr { + if attr.Key == "onerror" || attr.Key == "src" || attr.Key == "href" { + buf.WriteString(fmt.Sprintf(` %s="%s"`, attr.Key, attr.Val)) + } + } + + buf.WriteString(">") + + // Process child nodes (for which can have text) + for c := n.FirstChild; c != nil; c = c.NextSibling { + processNode(buf, c) + } + + // Close tag + buf.WriteString("") + } else if n.Type == html.TextNode { + // Preserve text inside tags + buf.WriteString(n.Data) + } + + // Process other children + for c := n.FirstChild; c != nil; c = c.NextSibling { + processNode(buf, c) + } +} + +func addPost(w http.ResponseWriter, r *http.Request) { + user := getUser(r) + if user == nil { + http.Redirect(w, r, "/", http.StatusSeeOther) + return + } + + content := r.FormValue("post") + content = sanitizeHTML(content) + if content == "" { + http.Redirect(w, r, "/home", http.StatusSeeOther) + return + } + + _, err := db.Exec("INSERT INTO posts (user_id, content) VALUES ((SELECT id FROM users WHERE name = ?), ?)", user.Name, content) + if err != nil { + http.Error(w, "Database error", http.StatusInternalServerError) + return + } + + http.Redirect(w, r, "/home", http.StatusSeeOther) +} + +func addHero(w http.ResponseWriter, r *http.Request) { + user := getUser(r) + if user == nil { + http.Redirect(w, r, "/", http.StatusSeeOther) + return + } + + hero := r.URL.Query().Get("hero") + if hero == "" { + http.Redirect(w, r, "/home", http.StatusSeeOther) + return + } + + _, err := db.Exec("INSERT OR IGNORE INTO heros (user_id, hero_id) VALUES ((SELECT id FROM users WHERE name = ?), (SELECT id FROM users WHERE name = ?))", user.Name, hero) + if err != nil { + http.Error(w, "Database error", http.StatusInternalServerError) + return + } + + http.Redirect(w, r, "/profile?name="+hero, http.StatusSeeOther) +} + +func profilePage(w http.ResponseWriter, r *http.Request) { + name := r.URL.Query().Get("name") + if name == "" { + http.NotFound(w, r) + return + } + + user := getUser(r) + posts := getPosts(name) + heros := getHeros(name) + + tmpl := template.Must(template.New("profile").Parse(` + + +

{{.Name}}'s Profile

+ Add as Hero +

Posts:

+ +

Heros:

+ + `)) + tmpl.Execute(w, map[string]interface{}{"User": user, "Name": name, "Posts": posts, "Heros": heros}) +} + +func homePage(w http.ResponseWriter, r *http.Request) { + user := getUser(r) + if user == nil { + http.Redirect(w, r, "/", http.StatusSeeOther) + return + } + + rows, _ := db.Query("SELECT name FROM users") + var users []string + for rows.Next() { + var name string + rows.Scan(&name) + users = append(users, name) + } + + tmpl := template.Must(template.New("home").Parse(` + + +

Welcome {{.User.Name}}

+
+ + +
+

Users:

+ + `)) + tmpl.Execute(w, map[string]interface{}{"User": user, "Users": users}) +} + +func indexPage(w http.ResponseWriter, r *http.Request) { + tmpl := template.Must(template.New("index").Parse(` + +

Welcome to Wormspace

+
+ + +
+ `)) + tmpl.Execute(w, nil) +} + +func main() { + initDB() + http.HandleFunc("/", indexPage) + http.HandleFunc("/setuser", setUser) + http.HandleFunc("/home", homePage) + http.HandleFunc("/profile", profilePage) + http.HandleFunc("/addpost", addPost) + http.HandleFunc("/addhero", addHero) + + http.ListenAndServe(":8080", nil) +}