wide/main.go

384 lines
10 KiB
Go
Raw Normal View History

2014-08-18 17:45:43 +04:00
package main
import (
2014-09-21 16:31:36 +04:00
"encoding/json"
2014-09-02 18:57:30 +04:00
"flag"
2014-09-07 13:07:25 +04:00
"html/template"
"math/rand"
"mime"
2014-09-07 13:07:25 +04:00
"net/http"
"runtime"
"strconv"
2014-09-24 10:36:34 +04:00
"time"
2014-09-07 13:07:25 +04:00
2014-08-18 17:51:03 +04:00
"github.com/b3log/wide/conf"
"github.com/b3log/wide/editor"
2014-09-15 10:24:40 +04:00
"github.com/b3log/wide/event"
2014-08-18 17:51:03 +04:00
"github.com/b3log/wide/file"
2014-08-25 18:17:02 +04:00
"github.com/b3log/wide/i18n"
2014-09-15 14:03:52 +04:00
"github.com/b3log/wide/notification"
2014-08-18 17:51:03 +04:00
"github.com/b3log/wide/output"
2014-09-17 10:35:48 +04:00
"github.com/b3log/wide/session"
2014-08-18 17:51:03 +04:00
"github.com/b3log/wide/shell"
2014-09-21 16:31:36 +04:00
"github.com/b3log/wide/util"
2014-08-18 17:45:43 +04:00
"github.com/golang/glog"
)
2014-10-13 08:22:50 +04:00
const (
2014-10-29 09:05:10 +03:00
Ver = "1.0.1" // wide version
CodeMirrorVer = "4.7" // editor version
2014-10-13 08:22:50 +04:00
)
2014-10-10 12:49:31 +04:00
2014-10-29 09:05:10 +03:00
// The only one init function in Wide.
2014-09-02 18:57:30 +04:00
func init() {
2014-10-29 09:05:10 +03:00
// TODO: args
2014-09-02 18:57:30 +04:00
flag.Set("logtostderr", "true")
2014-09-16 13:32:55 +04:00
flag.Set("v", "3")
2014-09-15 10:24:40 +04:00
flag.Parse()
2014-09-13 12:50:18 +04:00
2014-09-15 10:24:40 +04:00
event.Load()
2014-09-02 18:57:30 +04:00
2014-09-15 10:24:40 +04:00
conf.Load()
2014-09-22 19:13:07 +04:00
conf.FixedTimeCheckEnv()
conf.FixedTimeSave()
2014-09-23 17:03:44 +04:00
session.FixedTimeRelease()
2014-09-02 18:57:30 +04:00
}
2014-10-29 09:05:10 +03:00
// loginHandler handles request of user login.
2014-09-21 16:31:36 +04:00
func loginHandler(w http.ResponseWriter, r *http.Request) {
2014-10-12 18:58:08 +04:00
if "GET" == r.Method {
2014-10-29 09:05:10 +03:00
// show the login page
2014-09-21 16:31:36 +04:00
2014-10-23 17:43:35 +04:00
model := map[string]interface{}{"conf": conf.Wide, "i18n": i18n.GetAll(conf.Wide.Locale),
"locale": conf.Wide.Locale, "ver": Ver}
2014-09-21 16:31:36 +04:00
2014-10-24 07:41:23 +04:00
t, err := template.ParseFiles("views/login.html")
2014-09-21 16:31:36 +04:00
if nil != err {
glog.Error(err)
http.Error(w, err.Error(), 500)
return
}
t.Execute(w, model)
return
}
2014-10-29 09:05:10 +03:00
// non-GET request as login request
2014-09-21 16:31:36 +04:00
succ := false
data := map[string]interface{}{"succ": &succ}
defer util.RetJSON(w, r, data)
2014-09-23 18:29:53 +04:00
args := struct {
Username string
Password string
}{}
2014-09-21 16:31:36 +04:00
if err := json.NewDecoder(r.Body).Decode(&args); err != nil {
glog.Error(err)
succ = true
return
}
for _, user := range conf.Wide.Users {
2014-09-23 18:29:53 +04:00
if user.Name == args.Username && user.Password == args.Password {
2014-09-21 16:31:36 +04:00
succ = true
}
}
if !succ {
return
}
2014-10-29 09:05:10 +03:00
// create a HTTP session
2014-09-21 16:31:36 +04:00
httpSession, _ := session.HTTPSession.Get(r, "wide-session")
2014-09-23 18:29:53 +04:00
httpSession.Values["username"] = args.Username
2014-09-21 16:31:36 +04:00
httpSession.Values["id"] = strconv.Itoa(rand.Int())
2014-10-13 08:22:50 +04:00
httpSession.Options.MaxAge = conf.Wide.HTTPSessionMaxAge
2014-09-21 16:31:36 +04:00
httpSession.Save(r, w)
2014-09-23 18:29:53 +04:00
glog.Infof("Created a HTTP session [%s] for user [%s]", httpSession.Values["id"].(string), args.Username)
2014-09-21 16:31:36 +04:00
}
2014-10-29 09:05:10 +03:00
// logoutHandler handles request of user logout (exit).
2014-10-15 17:50:28 +04:00
func logoutHandler(w http.ResponseWriter, r *http.Request) {
data := map[string]interface{}{"succ": true}
defer util.RetJSON(w, r, data)
httpSession, _ := session.HTTPSession.Get(r, "wide-session")
2014-10-20 14:02:41 +04:00
httpSession.Options.MaxAge = -1
2014-10-15 17:50:28 +04:00
httpSession.Save(r, w)
}
2014-10-29 09:05:10 +03:00
// indexHandler handles request of Wide index.
2014-08-18 17:45:43 +04:00
func indexHandler(w http.ResponseWriter, r *http.Request) {
2014-09-17 10:35:48 +04:00
httpSession, _ := session.HTTPSession.Get(r, "wide-session")
2014-08-18 17:45:43 +04:00
2014-09-16 19:58:52 +04:00
if httpSession.IsNew {
2014-10-20 14:02:41 +04:00
http.Redirect(w, r, "/login", http.StatusFound)
2014-09-01 14:55:11 +04:00
2014-09-21 16:31:36 +04:00
return
2014-09-01 14:55:11 +04:00
}
2014-08-18 17:45:43 +04:00
2014-10-13 08:22:50 +04:00
httpSession.Options.MaxAge = conf.Wide.HTTPSessionMaxAge
2014-09-16 19:58:52 +04:00
httpSession.Save(r, w)
2014-08-18 17:45:43 +04:00
2014-10-29 09:05:10 +03:00
// create a Wide session
2014-09-17 10:35:48 +04:00
wideSession := session.WideSessions.New(httpSession)
2014-09-21 16:31:36 +04:00
username := httpSession.Values["username"].(string)
2014-10-23 17:43:35 +04:00
locale := conf.Wide.GetUser(username).Locale
2014-09-24 07:00:33 +04:00
wideSessions := session.WideSessions.GetByUsername(username)
2014-09-23 18:29:53 +04:00
userConf := conf.Wide.GetUser(username)
2014-10-23 17:43:35 +04:00
model := map[string]interface{}{"conf": conf.Wide, "i18n": i18n.GetAll(locale), "locale": locale,
2014-10-13 10:40:55 +04:00
"session": wideSession, "latestSessionContent": userConf.LatestSessionContent,
2014-10-27 11:00:02 +03:00
"pathSeparator": conf.PathSeparator, "codeMirrorVer": CodeMirrorVer}
2014-09-23 18:29:53 +04:00
2014-09-17 07:17:25 +04:00
glog.V(3).Infof("User [%s] has [%d] sessions", username, len(wideSessions))
2014-10-24 07:41:23 +04:00
t, err := template.ParseFiles("views/index.html")
2014-08-18 17:45:43 +04:00
if nil != err {
glog.Error(err)
http.Error(w, err.Error(), 500)
return
}
2014-08-25 18:17:02 +04:00
t.Execute(w, model)
2014-08-18 17:45:43 +04:00
}
2014-10-29 09:05:10 +03:00
// serveSingle registers the handler function for the given pattern and filename.
2014-10-12 18:00:26 +04:00
func serveSingle(pattern string, filename string) {
http.HandleFunc(pattern, func(w http.ResponseWriter, r *http.Request) {
http.ServeFile(w, r, filename)
})
2014-09-17 07:17:25 +04:00
}
2014-10-29 09:05:10 +03:00
// startHandler handles request of start page.
2014-10-19 14:41:01 +04:00
func startHandler(w http.ResponseWriter, r *http.Request) {
httpSession, _ := session.HTTPSession.Get(r, "wide-session")
if httpSession.IsNew {
2014-10-20 14:02:41 +04:00
http.Redirect(w, r, "/login", http.StatusFound)
2014-10-19 14:41:01 +04:00
return
}
httpSession.Options.MaxAge = conf.Wide.HTTPSessionMaxAge
httpSession.Save(r, w)
username := httpSession.Values["username"].(string)
2014-10-23 17:43:35 +04:00
locale := conf.Wide.GetUser(username).Locale
2014-10-21 13:41:42 +04:00
userWorkspace := conf.Wide.GetUserWorkspace(username)
2014-10-19 14:41:01 +04:00
2014-10-23 17:43:35 +04:00
model := map[string]interface{}{"conf": conf.Wide, "i18n": i18n.GetAll(locale), "locale": locale,
2014-10-21 13:41:42 +04:00
"username": username, "workspace": userWorkspace, "ver": Ver}
2014-10-19 14:41:01 +04:00
2014-10-24 07:41:23 +04:00
t, err := template.ParseFiles("views/start.html")
2014-10-19 14:41:01 +04:00
if nil != err {
glog.Error(err)
http.Error(w, err.Error(), 500)
return
}
t.Execute(w, model)
}
2014-10-29 09:05:10 +03:00
// keyboardShortcutsHandler handles request of keyboard shortcuts page.
2014-10-21 18:48:33 +04:00
func keyboardShortcutsHandler(w http.ResponseWriter, r *http.Request) {
2014-10-23 17:43:35 +04:00
httpSession, _ := session.HTTPSession.Get(r, "wide-session")
if httpSession.IsNew {
http.Redirect(w, r, "/login", http.StatusFound)
return
}
httpSession.Options.MaxAge = conf.Wide.HTTPSessionMaxAge
httpSession.Save(r, w)
username := httpSession.Values["username"].(string)
locale := conf.Wide.GetUser(username).Locale
model := map[string]interface{}{"conf": conf.Wide, "i18n": i18n.GetAll(locale), "locale": locale}
2014-10-21 18:48:33 +04:00
2014-10-24 07:41:23 +04:00
t, err := template.ParseFiles("views/keyboard_shortcuts.html")
2014-10-21 18:48:33 +04:00
if nil != err {
glog.Error(err)
http.Error(w, err.Error(), 500)
return
}
t.Execute(w, model)
}
2014-10-29 09:05:10 +03:00
// aboutHandle handles request of about page.
2014-10-20 18:27:19 +04:00
func aboutHandler(w http.ResponseWriter, r *http.Request) {
2014-10-23 17:43:35 +04:00
httpSession, _ := session.HTTPSession.Get(r, "wide-session")
if httpSession.IsNew {
http.Redirect(w, r, "/login", http.StatusFound)
return
}
httpSession.Options.MaxAge = conf.Wide.HTTPSessionMaxAge
httpSession.Save(r, w)
username := httpSession.Values["username"].(string)
locale := conf.Wide.GetUser(username).Locale
2014-10-23 19:14:34 +04:00
model := map[string]interface{}{"conf": conf.Wide, "i18n": i18n.GetAll(locale), "locale": locale, "ver": Ver,
"goos": runtime.GOOS, "goarch": runtime.GOARCH, "gover": runtime.Version()}
2014-10-20 18:27:19 +04:00
2014-10-24 07:41:23 +04:00
t, err := template.ParseFiles("views/about.html")
2014-10-20 18:27:19 +04:00
if nil != err {
glog.Error(err)
http.Error(w, err.Error(), 500)
return
}
t.Execute(w, model)
}
2014-10-29 09:05:10 +03:00
// Main.
2014-08-18 17:45:43 +04:00
func main() {
2014-09-02 18:57:30 +04:00
runtime.GOMAXPROCS(conf.Wide.MaxProcs)
initMime()
2014-09-02 18:57:30 +04:00
defer glog.Flush()
2014-10-12 18:00:26 +04:00
// IDE
http.HandleFunc("/login", handlerWrapper(loginHandler))
2014-10-15 17:50:28 +04:00
http.HandleFunc("/logout", handlerWrapper(logoutHandler))
2014-10-12 18:00:26 +04:00
http.HandleFunc("/", handlerWrapper(indexHandler))
2014-10-19 14:41:01 +04:00
http.HandleFunc("/start", handlerWrapper(startHandler))
2014-10-20 18:27:19 +04:00
http.HandleFunc("/about", handlerWrapper(aboutHandler))
2014-10-21 18:48:33 +04:00
http.HandleFunc("/keyboard_shortcuts", handlerWrapper(keyboardShortcutsHandler))
2014-10-12 18:00:26 +04:00
2014-10-29 09:05:10 +03:00
// static resources
2014-08-18 17:45:43 +04:00
http.Handle("/static/", http.StripPrefix("/static/", http.FileServer(http.Dir("static"))))
2014-10-12 18:00:26 +04:00
serveSingle("/favicon.ico", "./static/favicon.ico")
2014-08-18 17:45:43 +04:00
2014-10-29 09:05:10 +03:00
// workspaces
2014-09-01 17:40:10 +04:00
http.Handle("/data/", http.StripPrefix("/data/", http.FileServer(http.Dir("data"))))
2014-10-29 09:05:10 +03:00
// session
2014-09-24 10:36:34 +04:00
http.HandleFunc("/session/ws", handlerWrapper(session.WSHandler))
http.HandleFunc("/session/save", handlerWrapper(session.SaveContent))
2014-08-18 17:45:43 +04:00
2014-10-29 09:05:10 +03:00
// run
2014-09-24 10:36:34 +04:00
http.HandleFunc("/build", handlerWrapper(output.BuildHandler))
http.HandleFunc("/run", handlerWrapper(output.RunHandler))
http.HandleFunc("/stop", handlerWrapper(output.StopHandler))
2014-10-27 18:02:15 +03:00
http.HandleFunc("/go/test", handlerWrapper(output.GoTestHandler))
2014-09-24 10:36:34 +04:00
http.HandleFunc("/go/get", handlerWrapper(output.GoGetHandler))
http.HandleFunc("/go/install", handlerWrapper(output.GoInstallHandler))
http.HandleFunc("/output/ws", handlerWrapper(output.WSHandler))
2014-08-18 17:45:43 +04:00
2014-10-29 09:05:10 +03:00
// file tree
2014-09-24 10:36:34 +04:00
http.HandleFunc("/files", handlerWrapper(file.GetFiles))
http.HandleFunc("/file", handlerWrapper(file.GetFile))
http.HandleFunc("/file/save", handlerWrapper(file.SaveFile))
http.HandleFunc("/file/new", handlerWrapper(file.NewFile))
http.HandleFunc("/file/remove", handlerWrapper(file.RemoveFile))
2014-10-11 10:59:38 +04:00
http.HandleFunc("/file/search/text", handlerWrapper(file.SearchText))
2014-08-18 17:45:43 +04:00
2014-10-29 09:05:10 +03:00
// editor
2014-09-24 10:36:34 +04:00
http.HandleFunc("/editor/ws", handlerWrapper(editor.WSHandler))
http.HandleFunc("/go/fmt", handlerWrapper(editor.GoFmtHandler))
http.HandleFunc("/autocomplete", handlerWrapper(editor.AutocompleteHandler))
2014-10-10 07:18:52 +04:00
http.HandleFunc("/exprinfo", handlerWrapper(editor.GetExprInfoHandler))
2014-09-24 10:36:34 +04:00
http.HandleFunc("/find/decl", handlerWrapper(editor.FindDeclarationHandler))
http.HandleFunc("/find/usages", handlerWrapper(editor.FindUsagesHandler))
http.HandleFunc("/html/fmt", handlerWrapper(editor.HTMLFmtHandler))
2014-08-18 17:45:43 +04:00
2014-10-29 09:05:10 +03:00
// shell
2014-09-24 10:36:34 +04:00
http.HandleFunc("/shell/ws", handlerWrapper(shell.WSHandler))
http.HandleFunc("/shell", handlerWrapper(shell.IndexHandler))
2014-08-18 17:45:43 +04:00
2014-10-29 09:05:10 +03:00
// notification
2014-09-24 10:36:34 +04:00
http.HandleFunc("/notification/ws", handlerWrapper(notification.WSHandler))
2014-09-15 14:03:52 +04:00
2014-10-29 09:05:10 +03:00
// user
2014-09-24 10:36:34 +04:00
http.HandleFunc("/user/new", handlerWrapper(session.AddUser))
http.HandleFunc("/user/repos/init", handlerWrapper(session.InitGitRepos))
2014-08-31 14:50:38 +04:00
2014-09-08 07:37:34 +04:00
glog.V(0).Infof("Wide is running [%s]", conf.Wide.Server)
2014-08-31 13:23:14 +04:00
2014-08-18 17:45:43 +04:00
err := http.ListenAndServe(conf.Wide.Server, nil)
if err != nil {
glog.Fatal(err)
}
}
2014-09-24 07:58:02 +04:00
2014-10-29 09:05:10 +03:00
// handlerWrapper wraps the HTTP Handler for some common processes.
2014-09-25 09:29:04 +04:00
//
2014-10-08 09:54:16 +04:00
// 1. panic recover
2014-10-29 09:05:10 +03:00
// 2. request stopwatch
// 3. i18n
2014-09-24 10:36:34 +04:00
func handlerWrapper(f func(w http.ResponseWriter, r *http.Request)) func(w http.ResponseWriter, r *http.Request) {
handler := panicRecover(f)
handler = stopwatch(handler)
2014-10-26 06:44:13 +03:00
handler = i18nLoad(handler)
2014-09-24 10:36:34 +04:00
return handler
}
2014-10-29 09:05:10 +03:00
// i18nLoad wraps the i18n process.
2014-10-26 06:44:13 +03:00
func i18nLoad(handler func(w http.ResponseWriter, r *http.Request)) func(w http.ResponseWriter, r *http.Request) {
return func(w http.ResponseWriter, r *http.Request) {
i18n.Load()
handler(w, r)
}
}
2014-10-29 09:05:10 +03:00
// stopwatch wraps the request stopwatch process.
2014-09-24 10:36:34 +04:00
func stopwatch(handler func(w http.ResponseWriter, r *http.Request)) func(w http.ResponseWriter, r *http.Request) {
return func(w http.ResponseWriter, r *http.Request) {
start := time.Now()
defer func() {
glog.V(5).Infof("[%s] [%s]", r.RequestURI, time.Since(start))
}()
handler(w, r)
}
}
2014-10-29 09:05:10 +03:00
// panicRecover wraps the panic recover process.
2014-09-24 10:36:34 +04:00
func panicRecover(handler func(w http.ResponseWriter, r *http.Request)) func(w http.ResponseWriter, r *http.Request) {
2014-09-24 07:58:02 +04:00
return func(w http.ResponseWriter, r *http.Request) {
2014-09-26 13:36:12 +04:00
defer util.Recover()
2014-09-24 07:58:02 +04:00
2014-09-24 10:36:34 +04:00
handler(w, r)
2014-09-24 07:58:02 +04:00
}
}
2014-10-29 09:05:10 +03:00
// initMime initializes mime types.
//
2014-10-29 09:05:10 +03:00
// We can't get the mime types on some OS (such as Windows XP) by default, so initializes them here.
func initMime() {
mime.AddExtensionType(".css", "text/css")
mime.AddExtensionType(".js", "application/x-javascript")
mime.AddExtensionType(".json", "application/json")
}