459 lines
12 KiB
Go
459 lines
12 KiB
Go
// Copyright (c) 2014, B3log
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
// Configurations manipulations, all configurations (including user configurations) are stored in wide.json.
|
|
package conf
|
|
|
|
import (
|
|
"encoding/json"
|
|
"io/ioutil"
|
|
"net"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"runtime"
|
|
"strings"
|
|
"text/template"
|
|
"time"
|
|
|
|
"github.com/b3log/wide/event"
|
|
"github.com/b3log/wide/util"
|
|
"github.com/golang/glog"
|
|
)
|
|
|
|
const (
|
|
PathSeparator = string(os.PathSeparator) // OS-specific path separator
|
|
PathListSeparator = string(os.PathListSeparator) // OS-specific path list separator
|
|
)
|
|
|
|
const (
|
|
WideVersion = "1.0.1" // wide version
|
|
CodeMirrorVer = "4.7" // editor version
|
|
)
|
|
|
|
// The latest session content.
|
|
type LatestSessionContent struct {
|
|
FileTree []string // paths of expanding nodes of file tree
|
|
Files []string // paths of files of opening editor tabs
|
|
CurrentFile string // path of file of the current focused editor tab
|
|
}
|
|
|
|
// User configuration.
|
|
type User struct {
|
|
Name string
|
|
Password string
|
|
Workspace string // the GOPATH of this user
|
|
Locale string
|
|
GoFormat string
|
|
FontFamily string
|
|
FontSize string
|
|
Editor *Editor
|
|
LatestSessionContent *LatestSessionContent
|
|
}
|
|
|
|
// Editor configuration of a user.
|
|
type Editor struct {
|
|
FontFamily string
|
|
FontSize string
|
|
LineHeight string
|
|
}
|
|
|
|
// Configuration.
|
|
type conf struct {
|
|
IP string // server ip, ${ip}
|
|
Port string // server port
|
|
Server string // server host and port ({IP}:{Port})
|
|
StaticServer string // static resources server scheme, host and port (http://{IP}:{Port})
|
|
EditorChannel string // editor channel (ws://{IP}:{Port})
|
|
OutputChannel string // output channel (ws://{IP}:{Port})
|
|
ShellChannel string // shell channel(ws://{IP}:{Port})
|
|
SessionChannel string // wide session channel (ws://{IP}:{Port})
|
|
HTTPSessionMaxAge int // HTTP session max age (in seciond)
|
|
StaticResourceVersion string // version of static resources
|
|
MaxProcs int // Go max procs
|
|
RuntimeMode string // runtime mode (dev/prod)
|
|
WD string // current working direcitory, ${pwd}
|
|
Locale string // default locale
|
|
Users []*User // configurations of users
|
|
}
|
|
|
|
// Configuration variable.
|
|
var Wide conf
|
|
|
|
// A raw copy of configuration variable.
|
|
//
|
|
// Save function will use this variable to persist.
|
|
var rawWide conf
|
|
|
|
// FixedTimeCheckEnv checks Wide runtime enviorment periodically (7 minutes).
|
|
//
|
|
// Exits process if found fatal issues (such as not found $GOPATH),
|
|
// Notifies user by notification queue if found warning issues (such as not found gocode).
|
|
func FixedTimeCheckEnv() {
|
|
checkEnv() // check immediately
|
|
|
|
go func() {
|
|
for _ = range time.Tick(time.Minute * 7) {
|
|
checkEnv()
|
|
}
|
|
}()
|
|
}
|
|
|
|
func checkEnv() {
|
|
if "" == os.Getenv("GOPATH") {
|
|
glog.Fatal("Not found $GOPATH, please configure it before running Wide")
|
|
|
|
os.Exit(-1)
|
|
}
|
|
|
|
if "" == os.Getenv("GOROOT") {
|
|
glog.Fatal("Not found $GOROOT, please configure it before running Wide")
|
|
|
|
os.Exit(-1)
|
|
}
|
|
|
|
gocode := Wide.GetExecutableInGOBIN("gocode")
|
|
cmd := exec.Command(gocode, "close")
|
|
_, err := cmd.Output()
|
|
if nil != err {
|
|
event.EventQueue <- &event.Event{Code: event.EvtCodeGocodeNotFound}
|
|
|
|
glog.Warningf("Not found gocode [%s]", gocode)
|
|
}
|
|
|
|
ide_stub := Wide.GetExecutableInGOBIN("ide_stub")
|
|
cmd = exec.Command(ide_stub, "version")
|
|
_, err = cmd.Output()
|
|
if nil != err {
|
|
event.EventQueue <- &event.Event{Code: event.EvtCodeIDEStubNotFound}
|
|
|
|
glog.Warningf("Not found ide_stub [%s]", ide_stub)
|
|
}
|
|
}
|
|
|
|
// FixedTimeSave saves configurations (wide.json) periodically (1 minute).
|
|
//
|
|
// Main goal of this function is to save user session content, for restoring session content while user open Wide next time.
|
|
func FixedTimeSave() {
|
|
go func() {
|
|
for _ = range time.Tick(time.Minute) {
|
|
Save()
|
|
}
|
|
}()
|
|
}
|
|
|
|
// GetUserWorkspace gets workspace path with the specified username, returns "" if not found.
|
|
func (c *conf) GetUserWorkspace(username string) string {
|
|
for _, user := range c.Users {
|
|
if user.Name == username {
|
|
return user.GetWorkspace()
|
|
}
|
|
}
|
|
|
|
return ""
|
|
}
|
|
|
|
// GetGoFmt gets the path of Go format tool, returns "gofmt" if not found.
|
|
func (c *conf) GetGoFmt(username string) string {
|
|
for _, user := range c.Users {
|
|
if user.Name == username {
|
|
switch user.GoFormat {
|
|
case "gofmt":
|
|
return "gofmt"
|
|
case "goimports":
|
|
return c.GetExecutableInGOBIN("goimports")
|
|
default:
|
|
glog.Errorf("Unsupported Go Format tool [%s]", user.GoFormat)
|
|
return "gofmt"
|
|
}
|
|
}
|
|
}
|
|
|
|
return "gofmt"
|
|
}
|
|
|
|
// GetWorkspace gets workspace path of the user.
|
|
//
|
|
// Compared to the use of Wide.Workspace, this function will be processed as follows:
|
|
// 1. Replace {WD} variable with the actual directory path
|
|
// 2. Replace ${GOPATH} with enviorment variable GOPATH
|
|
// 3. Replace "/" with "\\" (Windows)
|
|
func (u *User) GetWorkspace() string {
|
|
w := strings.Replace(u.Workspace, "{WD}", Wide.WD, 1)
|
|
w = strings.Replace(w, "${GOPATH}", os.Getenv("GOPATH"), 1)
|
|
|
|
return filepath.FromSlash(w)
|
|
}
|
|
|
|
// GetUser gets configuration of the user specified by the given username, returns nil if not found.
|
|
func (*conf) GetUser(username string) *User {
|
|
for _, user := range Wide.Users {
|
|
if user.Name == username {
|
|
return user
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// GetExecutableInGOBIN gets executable file under GOBIN path.
|
|
//
|
|
// The specified executable should not with extension, this function will append .exe if on Windows.
|
|
func (*conf) GetExecutableInGOBIN(executable string) string {
|
|
if util.OS.IsWindows() {
|
|
executable += ".exe"
|
|
}
|
|
|
|
gopaths := filepath.SplitList(os.Getenv("GOPATH"))
|
|
|
|
for _, gopath := range gopaths {
|
|
// $GOPATH/bin/$GOOS_$GOARCH/executable
|
|
ret := gopath + PathSeparator + "bin" + PathSeparator +
|
|
os.Getenv("GOOS") + "_" + os.Getenv("GOARCH") + PathSeparator + executable
|
|
if util.File.IsExist(ret) {
|
|
return ret
|
|
}
|
|
|
|
// $GOPATH/bin/{runtime.GOOS}_{runtime.GOARCH}/executable
|
|
ret = gopath + PathSeparator + "bin" + PathSeparator +
|
|
runtime.GOOS + "_" + runtime.GOARCH + PathSeparator + executable
|
|
if util.File.IsExist(ret) {
|
|
return ret
|
|
}
|
|
|
|
// $GOPATH/bin/executable
|
|
ret = gopath + PathSeparator + "bin" + PathSeparator + executable
|
|
if util.File.IsExist(ret) {
|
|
return ret
|
|
}
|
|
}
|
|
|
|
// $GOBIN/executable
|
|
return os.Getenv("GOBIN") + PathSeparator + executable
|
|
}
|
|
|
|
// Save saves Wide configurations.
|
|
func Save() bool {
|
|
// just the Users field are volatile
|
|
rawWide.Users = Wide.Users
|
|
|
|
// format
|
|
bytes, err := json.MarshalIndent(rawWide, "", " ")
|
|
|
|
if nil != err {
|
|
glog.Error(err)
|
|
|
|
return false
|
|
}
|
|
|
|
if err = ioutil.WriteFile("conf/wide.json", bytes, 0644); nil != err {
|
|
glog.Error(err)
|
|
|
|
return false
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
// Load loads the configurations from wide.json.
|
|
func Load(confPath, confIP, confPort, confServer, confChannel string, confDocker bool) {
|
|
bytes, _ := ioutil.ReadFile(confPath)
|
|
|
|
err := json.Unmarshal(bytes, &Wide)
|
|
if err != nil {
|
|
glog.Error("Parses wide.json error: ", err)
|
|
|
|
os.Exit(-1)
|
|
}
|
|
|
|
// keep the raw content
|
|
json.Unmarshal(bytes, &rawWide)
|
|
|
|
// Working Driectory
|
|
Wide.WD = util.OS.Pwd()
|
|
glog.V(5).Infof("${pwd} [%s]", Wide.WD)
|
|
|
|
// IP
|
|
ip, err := util.Net.LocalIP()
|
|
if err != nil {
|
|
glog.Error(err)
|
|
|
|
os.Exit(-1)
|
|
}
|
|
|
|
glog.V(5).Infof("${ip} [%s]", ip)
|
|
|
|
if "" != confIP {
|
|
ip = confIP
|
|
}
|
|
|
|
Wide.IP = strings.Replace(Wide.IP, "${ip}", ip, 1)
|
|
|
|
if "" != confPort {
|
|
Wide.Port = confPort
|
|
}
|
|
|
|
// Server
|
|
Wide.Server = strings.Replace(Wide.Server, "{IP}", Wide.IP, 1)
|
|
if "" != confServer {
|
|
Wide.Server = confServer
|
|
}
|
|
|
|
// Static Server
|
|
Wide.StaticServer = strings.Replace(Wide.StaticServer, "{IP}", Wide.IP, 1)
|
|
|
|
// Channels
|
|
Wide.EditorChannel = strings.Replace(Wide.EditorChannel, "{IP}", Wide.IP, 1)
|
|
if "" != confChannel {
|
|
Wide.EditorChannel = confChannel
|
|
}
|
|
Wide.OutputChannel = strings.Replace(Wide.OutputChannel, "{IP}", Wide.IP, 1)
|
|
if "" != confChannel {
|
|
Wide.OutputChannel = confChannel
|
|
}
|
|
Wide.ShellChannel = strings.Replace(Wide.ShellChannel, "{IP}", Wide.IP, 1)
|
|
if "" != confChannel {
|
|
Wide.ShellChannel = confChannel
|
|
}
|
|
Wide.SessionChannel = strings.Replace(Wide.SessionChannel, "{IP}", Wide.IP, 1)
|
|
if "" != confChannel {
|
|
Wide.SessionChannel = confChannel
|
|
}
|
|
|
|
// Docker
|
|
if confDocker {
|
|
h, err := net.LookupHost("wide")
|
|
if nil != err {
|
|
glog.Error(err)
|
|
|
|
os.Exit(-1)
|
|
}
|
|
|
|
// XXX: secure protocol wss enchance
|
|
channel := "ws://" + h[0] + Wide.Port
|
|
Wide.EditorChannel = channel
|
|
Wide.OutputChannel = channel
|
|
Wide.ShellChannel = channel
|
|
Wide.SessionChannel = channel
|
|
}
|
|
|
|
Wide.Server = strings.Replace(Wide.Server, "{Port}", Wide.Port, 1)
|
|
Wide.StaticServer = strings.Replace(Wide.StaticServer, "{Port}", Wide.Port, 1)
|
|
Wide.EditorChannel = strings.Replace(Wide.EditorChannel, "{Port}", Wide.Port, 1)
|
|
Wide.OutputChannel = strings.Replace(Wide.OutputChannel, "{Port}", Wide.Port, 1)
|
|
Wide.ShellChannel = strings.Replace(Wide.ShellChannel, "{Port}", Wide.Port, 1)
|
|
Wide.SessionChannel = strings.Replace(Wide.SessionChannel, "{Port}", Wide.Port, 1)
|
|
|
|
glog.V(5).Info("Conf: \n" + string(bytes))
|
|
|
|
initWorkspaceDirs()
|
|
initCustomizedConfs()
|
|
}
|
|
|
|
// initCustomizedConfs initializes the user customized configurations.
|
|
func initCustomizedConfs() {
|
|
for _, user := range Wide.Users {
|
|
UpdateCustomizedConf(user.Name)
|
|
}
|
|
}
|
|
|
|
// UpdateCustomizedConf creates (if not exists) or updates user customized configuration files.
|
|
//
|
|
// 1. /static/user/{username}/style.css
|
|
func UpdateCustomizedConf(username string) {
|
|
var u *User = nil
|
|
for _, user := range Wide.Users { // maybe it is a beauty of the trade-off of the another world between design and implementation
|
|
if user.Name == username {
|
|
u = user
|
|
}
|
|
}
|
|
|
|
if nil == u {
|
|
return
|
|
}
|
|
|
|
model := map[string]interface{}{"user": u}
|
|
|
|
t, err := template.ParseFiles("static/user/style.css.tmpl")
|
|
if nil != err {
|
|
glog.Error(err)
|
|
|
|
os.Exit(-1)
|
|
}
|
|
|
|
wd := util.OS.Pwd()
|
|
dir := filepath.Clean(wd + "/static/user/" + u.Name)
|
|
if err := os.MkdirAll(dir, 0755); nil != err {
|
|
glog.Error(err)
|
|
|
|
os.Exit(-1)
|
|
}
|
|
|
|
fout, err := os.Create(dir + PathSeparator + "style.css")
|
|
if nil != err {
|
|
glog.Error(err)
|
|
|
|
os.Exit(-1)
|
|
}
|
|
|
|
defer fout.Close()
|
|
|
|
if err := t.Execute(fout, model); nil != err {
|
|
glog.Error(err)
|
|
|
|
os.Exit(-1)
|
|
}
|
|
}
|
|
|
|
// initWorkspaceDirs initializes the directories of users' workspaces.
|
|
//
|
|
// Creates directories if not found on path of workspace.
|
|
func initWorkspaceDirs() {
|
|
paths := []string{}
|
|
|
|
for _, user := range Wide.Users {
|
|
paths = append(paths, filepath.SplitList(user.GetWorkspace())...)
|
|
}
|
|
|
|
for _, path := range paths {
|
|
CreateWorkspaceDir(path)
|
|
}
|
|
}
|
|
|
|
// CreateWorkspaceDir creates (if not exists) directories on the path.
|
|
//
|
|
// 1. root directory:{path}
|
|
// 2. src directory: {path}/src
|
|
// 3. package directory: {path}/pkg
|
|
// 4. binary directory: {path}/bin
|
|
func CreateWorkspaceDir(path string) {
|
|
createDir(path)
|
|
createDir(path + PathSeparator + "src")
|
|
createDir(path + PathSeparator + "pkg")
|
|
createDir(path + PathSeparator + "bin")
|
|
}
|
|
|
|
// createDir creates a directory on the path if it not exists.
|
|
func createDir(path string) {
|
|
if !util.File.IsExist(path) {
|
|
if err := os.MkdirAll(path, 0775); nil != err {
|
|
glog.Error(err)
|
|
|
|
os.Exit(-1)
|
|
}
|
|
|
|
glog.V(7).Infof("Created a directory [%s]", path)
|
|
}
|
|
}
|