diff --git a/gowiki/about.txt b/gowiki/about.txt new file mode 100644 index 0000000..ec147cc --- /dev/null +++ b/gowiki/about.txt @@ -0,0 +1,3 @@ +This is a simple wiki created in Go following an official guide from the Go developers. + +https://golang.org/doc/articles/wiki/ \ No newline at end of file diff --git a/gowiki/edit.html b/gowiki/edit.html new file mode 100644 index 0000000..044c3be --- /dev/null +++ b/gowiki/edit.html @@ -0,0 +1,6 @@ +

Editing {{.Title}}

+ +
+
+
+
diff --git a/gowiki/go.mod b/gowiki/go.mod new file mode 100644 index 0000000..d6c89dd --- /dev/null +++ b/gowiki/go.mod @@ -0,0 +1,3 @@ +module example.com/gowiki + +go 1.17 diff --git a/gowiki/view.html b/gowiki/view.html new file mode 100644 index 0000000..b1e87ef --- /dev/null +++ b/gowiki/view.html @@ -0,0 +1,5 @@ +

{{.Title}}

+ +

[edit]

+ +
{{printf "%s" .Body}}
diff --git a/gowiki/wiki b/gowiki/wiki new file mode 100755 index 0000000..6e879e8 Binary files /dev/null and b/gowiki/wiki differ diff --git a/gowiki/wiki.go b/gowiki/wiki.go new file mode 100644 index 0000000..cddef56 --- /dev/null +++ b/gowiki/wiki.go @@ -0,0 +1,95 @@ +package main + +import ( + "errors" + "html/template" + "io/ioutil" + "log" + "net/http" + "regexp" +) + +type Page struct { + Title string + Body []byte +} + +var templates = template.Must(template.ParseFiles("edit.html", "view.html")) +var validPath = regexp.MustCompile("^/(edit|save|view)/([a-zA-Z0-9]+)$") + +func (p *Page) save() error { + filename := p.Title + ".txt" + return ioutil.WriteFile(filename, p.Body, 0600) +} + +func getTitle(w http.ResponseWriter, r *http.Request) (string, error) { + m := validPath.FindStringSubmatch(r.URL.Path) + if m == nil { + http.NotFound(w, r) + return "", errors.New("invalid Page Title") + } + return m[2], nil // The title is the second subexpression. +} + +func loadPage(title string) (*Page, error) { + filename := title + ".txt" + body, err := ioutil.ReadFile(filename) + if err != nil { + return nil, err + } + return &Page{Title: title, Body: body}, nil +} + +func renderTemplate(w http.ResponseWriter, tmpl string, p *Page) { + err := templates.ExecuteTemplate(w, tmpl+".html", p) + if err != nil { + http.Error(w, err.Error(), http.StatusInternalServerError) + } +} + +func makeHandler(fn func(http.ResponseWriter, *http.Request, string)) http.HandlerFunc { + return func(w http.ResponseWriter, r *http.Request) { + m := validPath.FindStringSubmatch(r.URL.Path) + if m == nil { + http.NotFound(w, r) + return + } + fn(w, r, m[2]) + } +} + +func viewHandler(w http.ResponseWriter, r *http.Request, title string) { + p, err := loadPage(title) + if err != nil { + http.Redirect(w, r, "/edit/"+title, http.StatusFound) + return + } + renderTemplate(w, "view", p) +} + +func editHandler(w http.ResponseWriter, r *http.Request, title string) { + p, err := loadPage(title) + if err != nil { + p = &Page{Title: title} + } + renderTemplate(w, "edit", p) +} + +func saveHandler(w http.ResponseWriter, r *http.Request, title string) { + body := r.FormValue("body") + p := &Page{Title: title, Body: []byte(body)} + err := p.save() + if err != nil { + http.Error(w, err.Error(), http.StatusInternalServerError) + return + } + http.Redirect(w, r, "/view/"+title, http.StatusFound) +} + +func main() { + http.HandleFunc("/view/", makeHandler(viewHandler)) + http.HandleFunc("/edit/", makeHandler(editHandler)) + http.HandleFunc("/save/", makeHandler(saveHandler)) + + log.Fatal(http.ListenAndServe(":8173", nil)) +}