2017-07-12 18:43:11 +12:00
|
|
|
package yatwiki
|
2017-07-09 11:13:36 +12:00
|
|
|
|
|
|
|
import (
|
2017-07-11 20:19:37 +12:00
|
|
|
"database/sql"
|
2017-07-09 13:00:45 +12:00
|
|
|
"errors"
|
2017-07-09 18:11:39 +12:00
|
|
|
"fmt"
|
2017-07-09 11:13:36 +12:00
|
|
|
"html/template"
|
2017-07-11 18:39:59 +12:00
|
|
|
"math/rand"
|
2017-07-09 11:13:36 +12:00
|
|
|
"net/http"
|
2017-07-09 13:00:45 +12:00
|
|
|
"net/url"
|
2017-07-11 18:31:50 +12:00
|
|
|
"regexp"
|
2017-07-09 13:18:18 +12:00
|
|
|
"strconv"
|
2017-07-09 12:15:49 +12:00
|
|
|
"strings"
|
2017-07-09 17:20:10 +12:00
|
|
|
"time"
|
2017-07-09 11:13:36 +12:00
|
|
|
)
|
|
|
|
|
2017-10-29 13:40:00 +13:00
|
|
|
var SERVER_HEADER string = "YATWiki/0.0.0-devel"
|
|
|
|
|
2017-07-09 11:13:36 +12:00
|
|
|
type WikiServer struct {
|
|
|
|
db *WikiDB
|
|
|
|
opts *ServerOptions
|
|
|
|
pageTmp *template.Template
|
2017-07-09 17:20:10 +12:00
|
|
|
loc *time.Location
|
2017-07-11 18:31:50 +12:00
|
|
|
rxDiff *regexp.Regexp
|
2017-07-11 19:14:26 +12:00
|
|
|
bans []*regexp.Regexp
|
2017-07-09 11:13:36 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewWikiServer(opts *ServerOptions) (*WikiServer, error) {
|
2017-07-09 18:45:28 +12:00
|
|
|
wdb, err := NewWikiDB(opts.DBFilePath, opts.GzipCompressionLevel)
|
2017-07-09 11:13:36 +12:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2017-07-11 20:19:37 +12:00
|
|
|
tr, err := wdb.TotalRevisions()
|
|
|
|
if (err == nil && tr == 0) || err == sql.ErrNoRows {
|
|
|
|
err := wdb.SaveArticle(opts.DefaultPage, `YATWiki3`, "", 0)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-13 18:04:26 +12:00
|
|
|
tmpl := template.New("yatwiki/page")
|
|
|
|
tmpl.Funcs(map[string]interface{}{
|
|
|
|
"pathcomponent": func(s string) string {
|
|
|
|
return url.PathEscape(s)
|
|
|
|
},
|
|
|
|
})
|
|
|
|
_, err = tmpl.Parse(pageTemplate)
|
2017-07-09 11:13:36 +12:00
|
|
|
if err != nil {
|
2017-07-09 17:20:10 +12:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
loc, err := time.LoadLocation(opts.Timezone)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2017-07-09 11:13:36 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
ws := WikiServer{
|
|
|
|
db: wdb,
|
|
|
|
opts: opts,
|
|
|
|
pageTmp: tmpl,
|
2017-07-09 17:20:10 +12:00
|
|
|
loc: loc,
|
2017-07-11 18:31:50 +12:00
|
|
|
rxDiff: regexp.MustCompile(`diff/(\d+)/(\d+)`),
|
2017-07-11 19:14:26 +12:00
|
|
|
bans: make([]*regexp.Regexp, 0, len(opts.BannedUserIPRegexes)),
|
2017-07-09 11:13:36 +12:00
|
|
|
}
|
2017-07-11 19:14:26 +12:00
|
|
|
|
|
|
|
for _, banRx := range opts.BannedUserIPRegexes {
|
|
|
|
rx, err := regexp.Compile(banRx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
ws.bans = append(ws.bans, rx)
|
|
|
|
}
|
|
|
|
|
2017-07-09 11:13:36 +12:00
|
|
|
return &ws, nil
|
|
|
|
}
|
|
|
|
|
2017-11-18 15:04:19 +13:00
|
|
|
func (this *WikiServer) GetBBCodeRenderer() *BBCodeRenderer {
|
|
|
|
return NewBBCodeRenderer(this.opts.ExpectBaseURL, this.opts.ContentedServer, this.opts.ContentedBBCodeTag)
|
|
|
|
}
|
|
|
|
|
2017-07-09 11:13:36 +12:00
|
|
|
func (this *WikiServer) Close() {
|
|
|
|
this.db.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (this *WikiServer) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
2017-10-29 13:40:00 +13:00
|
|
|
w.Header().Set("Server", SERVER_HEADER)
|
2017-07-09 11:13:36 +12:00
|
|
|
|
2017-07-11 19:14:26 +12:00
|
|
|
if len(this.bans) > 0 {
|
|
|
|
remoteIP := RemoteAddrToIPAddress(r.RemoteAddr)
|
|
|
|
for _, ban := range this.bans {
|
|
|
|
if ban.MatchString(remoteIP) {
|
|
|
|
http.Error(w, "Unauthorised", 403)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-11 18:31:50 +12:00
|
|
|
if !strings.HasPrefix(r.URL.Path, this.opts.ExpectBaseURL) {
|
|
|
|
http.Error(w, "Bad request", 400)
|
|
|
|
return
|
|
|
|
}
|
2017-07-11 18:36:08 +12:00
|
|
|
remainingPath := r.URL.Path[len(this.opts.ExpectBaseURL):]
|
2017-07-11 18:31:50 +12:00
|
|
|
|
2017-07-09 12:15:49 +12:00
|
|
|
if r.Method == "GET" {
|
2017-07-11 18:36:08 +12:00
|
|
|
if remainingPath == "wiki.css" {
|
2017-07-09 12:15:49 +12:00
|
|
|
w.Header().Set("Content-Type", "text/css")
|
|
|
|
content, _ := wikiCssBytes()
|
|
|
|
w.Write(content)
|
|
|
|
return
|
|
|
|
|
2017-07-11 18:36:08 +12:00
|
|
|
} else if remainingPath == "highlight.js" {
|
2017-07-09 12:15:49 +12:00
|
|
|
w.Header().Set("Content-Type", "application/javascript")
|
|
|
|
content, _ := highlightJsBytes()
|
|
|
|
w.Write(content)
|
|
|
|
return
|
|
|
|
|
2017-10-29 13:40:15 +13:00
|
|
|
} else if remainingPath == "favicon.ico" {
|
|
|
|
if len(this.opts.FaviconFilePath) > 0 {
|
|
|
|
w.Header().Set("Content-Type", "image/x-icon")
|
|
|
|
http.ServeFile(w, r, this.opts.FaviconFilePath)
|
|
|
|
} else {
|
|
|
|
http.Error(w, "Not found", 404)
|
|
|
|
}
|
2017-07-09 12:14:28 +12:00
|
|
|
return
|
2017-07-09 12:15:49 +12:00
|
|
|
|
2017-07-11 18:36:08 +12:00
|
|
|
} else if remainingPath == "download-database" {
|
2017-07-09 18:11:39 +12:00
|
|
|
w.Header().Set("Content-Type", "application/octet-stream")
|
|
|
|
w.Header().Set("Content-Disposition", `attachment; filename="database-`+fmt.Sprintf("%d", time.Now().Unix())+`.db"`)
|
|
|
|
http.ServeFile(w, r, this.opts.DBFilePath)
|
|
|
|
return
|
|
|
|
|
2017-07-11 18:36:08 +12:00
|
|
|
} else if remainingPath == "formatting" {
|
2017-07-09 13:26:26 +12:00
|
|
|
this.routeFormatting(w, r)
|
2017-07-09 13:00:45 +12:00
|
|
|
return
|
|
|
|
|
2017-07-11 18:36:08 +12:00
|
|
|
} else if remainingPath == "index" {
|
2017-07-09 13:26:26 +12:00
|
|
|
this.routeIndex(w, r)
|
2017-07-09 12:15:49 +12:00
|
|
|
return
|
|
|
|
|
2017-07-11 19:08:22 +12:00
|
|
|
} else if remainingPath == "rss/changes" {
|
|
|
|
this.routeRecentChangesRSS(w, r)
|
|
|
|
return
|
|
|
|
|
2017-07-11 18:36:14 +12:00
|
|
|
} else if remainingPath == "" {
|
2017-08-13 17:51:44 +12:00
|
|
|
this.serveRedirect(w, this.opts.ExpectBaseURL+`view/`+url.PathEscape(this.opts.DefaultPage))
|
2017-07-11 18:36:14 +12:00
|
|
|
return
|
2017-07-11 18:36:08 +12:00
|
|
|
|
2017-07-11 18:39:59 +12:00
|
|
|
} else if remainingPath == "random" {
|
2018-04-02 17:41:47 +12:00
|
|
|
titles, err := this.db.ListTitles(false) // "Random page" mode does not include deleted pages
|
2017-07-11 18:39:59 +12:00
|
|
|
if err != nil {
|
|
|
|
this.serveInternalError(w, r, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
chosenArticle := titles[rand.Intn(len(titles))]
|
2018-04-02 17:41:47 +12:00
|
|
|
this.serveRedirect(w, this.opts.ExpectBaseURL+`view/`+url.PathEscape(chosenArticle.Title))
|
2017-07-11 18:39:59 +12:00
|
|
|
return
|
|
|
|
|
2017-07-11 18:36:08 +12:00
|
|
|
} else if strings.HasPrefix(remainingPath, "view/") {
|
2017-08-13 17:51:44 +12:00
|
|
|
articleTitle, err := url.PathUnescape(remainingPath[len("view/"):])
|
2017-07-09 13:00:11 +12:00
|
|
|
if err != nil {
|
|
|
|
this.serveErrorMessage(w, err)
|
|
|
|
return
|
|
|
|
}
|
2017-07-09 13:26:26 +12:00
|
|
|
this.routeView(w, r, articleTitle)
|
2017-07-09 12:15:49 +12:00
|
|
|
return
|
|
|
|
|
2017-07-11 18:36:08 +12:00
|
|
|
} else if strings.HasPrefix(remainingPath, "modify/") {
|
2017-08-13 17:51:44 +12:00
|
|
|
articleTitle, err := url.PathUnescape(remainingPath[len("modify/"):])
|
2017-07-09 13:50:13 +12:00
|
|
|
if err != nil {
|
|
|
|
this.serveErrorMessage(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
this.routeModify(w, r, articleTitle)
|
|
|
|
return
|
|
|
|
|
2017-07-11 18:36:08 +12:00
|
|
|
} else if strings.HasPrefix(remainingPath, "history/") {
|
2017-08-13 17:51:44 +12:00
|
|
|
articleTitle, err := url.PathUnescape(remainingPath[len("history/"):])
|
2017-07-09 18:05:03 +12:00
|
|
|
if err != nil {
|
|
|
|
this.serveErrorMessage(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
this.routeHistory(w, r, articleTitle)
|
|
|
|
return
|
|
|
|
|
2017-07-11 18:36:08 +12:00
|
|
|
} else if strings.HasPrefix(remainingPath, "raw/") {
|
|
|
|
revId, err := strconv.Atoi(remainingPath[len("raw/"):])
|
2017-07-09 13:18:18 +12:00
|
|
|
if err != nil {
|
|
|
|
this.serveErrorMessage(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-07-09 13:26:26 +12:00
|
|
|
this.routeRawView(w, r, revId)
|
2017-07-09 13:18:18 +12:00
|
|
|
return
|
2017-07-09 17:20:10 +12:00
|
|
|
|
2017-07-11 18:36:08 +12:00
|
|
|
} else if strings.HasPrefix(remainingPath, "archive/") {
|
|
|
|
revId, err := strconv.Atoi(remainingPath[len("archive/"):])
|
2017-07-09 17:20:10 +12:00
|
|
|
if err != nil {
|
|
|
|
this.serveErrorMessage(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
this.routeArchive(w, r, revId)
|
|
|
|
return
|
|
|
|
|
2017-07-11 18:36:08 +12:00
|
|
|
} else if strings.HasPrefix(remainingPath, "recent/") {
|
|
|
|
pageNum, err := strconv.Atoi(remainingPath[len("recent/"):])
|
2017-07-09 18:05:03 +12:00
|
|
|
if err != nil {
|
|
|
|
this.serveErrorMessage(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
this.routeRecentChanges(w, r, pageNum)
|
|
|
|
return
|
|
|
|
|
2017-07-11 18:36:08 +12:00
|
|
|
} else if remainingPath == "diff" {
|
2017-07-11 18:31:50 +12:00
|
|
|
this.serveRedirect(w, this.opts.ExpectBaseURL+`diff/`+r.URL.Query().Get("f")+`/`+r.URL.Query().Get("t"))
|
|
|
|
return
|
|
|
|
|
2017-07-11 19:19:42 +12:00
|
|
|
} else if strings.HasPrefix(remainingPath, "diff/parent/") {
|
|
|
|
toRev, err := strconv.Atoi(remainingPath[len("diff/parent/"):])
|
|
|
|
if err != nil {
|
|
|
|
this.serveErrorMessage(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
fromRev, err := this.db.GetNextOldestRevision(toRev)
|
|
|
|
if err != nil {
|
|
|
|
this.serveErrorMessage(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
this.serveRedirect(w, this.opts.ExpectBaseURL+`diff/`+fmt.Sprintf("%d/%d", fromRev, toRev))
|
|
|
|
return
|
|
|
|
|
2017-07-11 18:36:08 +12:00
|
|
|
} else if match := this.rxDiff.FindStringSubmatch(remainingPath); len(match) == 3 {
|
2017-07-11 17:52:29 +12:00
|
|
|
|
2017-07-11 18:31:50 +12:00
|
|
|
fromRev, err := strconv.Atoi(match[1])
|
2017-07-11 17:52:29 +12:00
|
|
|
if err != nil {
|
|
|
|
this.serveErrorMessage(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-07-11 18:31:50 +12:00
|
|
|
toRev, err := strconv.Atoi(match[2])
|
2017-07-11 17:52:29 +12:00
|
|
|
if err != nil {
|
|
|
|
this.serveErrorMessage(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
this.routeDiff(w, r, fromRev, toRev)
|
|
|
|
return
|
|
|
|
|
2017-07-09 12:15:49 +12:00
|
|
|
}
|
|
|
|
|
2017-07-09 18:45:28 +12:00
|
|
|
} else if r.Method == "POST" {
|
|
|
|
|
|
|
|
if r.URL.Path == this.opts.ExpectBaseURL+"save" {
|
|
|
|
err := r.ParseForm()
|
|
|
|
if err != nil {
|
|
|
|
this.serveErrorMessage(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
title := r.Form.Get("pname")
|
|
|
|
body := r.Form.Get("content")
|
|
|
|
expectRev, err := strconv.Atoi(r.Form.Get("baserev"))
|
|
|
|
if err != nil {
|
|
|
|
this.serveErrorMessage(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-08-13 17:32:54 +12:00
|
|
|
err = this.db.SaveArticle(title, Author(r, this.opts.TrustXForwardedFor), body, int64(expectRev))
|
2017-07-09 18:45:28 +12:00
|
|
|
if err != nil {
|
|
|
|
this.serveErrorMessage(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-08-13 17:51:44 +12:00
|
|
|
this.serveRedirect(w, this.opts.ExpectBaseURL+`view/`+url.PathEscape(title))
|
2017-07-09 18:45:28 +12:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-07-09 12:15:49 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
// No match? Add 'Page not found' to next session response, and redirect to homepage
|
2017-07-09 13:00:45 +12:00
|
|
|
this.serveErrorMessage(w, errors.New("Page not found"))
|
2017-07-09 12:15:49 +12:00
|
|
|
|
2017-07-09 13:00:45 +12:00
|
|
|
}
|