1
0
mirror of https://github.com/beego/bee.git synced 2024-11-27 03:01:29 +00:00
bee/run.go

238 lines
6.2 KiB
Go
Raw Normal View History

2013-09-03 17:23:58 +00:00
// Copyright 2013 bee authors
//
// 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.
2013-07-24 12:01:14 +00:00
package main
import (
"bytes"
"fmt"
"io/ioutil"
"log"
"net"
2013-07-24 12:01:14 +00:00
"os"
"os/exec"
2013-07-24 12:01:14 +00:00
path "path/filepath"
"runtime"
2013-09-11 23:06:04 +00:00
"strings"
2013-07-24 12:01:14 +00:00
)
var cmdRun = &Command{
UsageLine: "run [appname] [watchall] [-main=*.go] [-downdoc=true] [-gendoc=true] [-e=Godeps -e=folderToExclude] [-tags=goBuildTags] [-rip=serverIp] [-rpassword=serverPassword]",
Short: "run the app and start a Web server for development",
2013-07-24 12:01:14 +00:00
Long: `
Run command will supervise the file system of the beego project using inotify,
it will recompile and restart the app after any modifications.
2013-07-24 12:01:14 +00:00
`,
}
2014-02-21 17:51:18 +00:00
var mainFiles ListOpts
var downdoc docValue
2014-08-08 16:55:55 +00:00
var gendoc docValue
2015-06-17 06:59:59 +00:00
// The flags list of the paths excluded from watching
var excludedPaths strFlags
2016-02-10 19:39:02 +00:00
// Pass through to -tags arg of "go build"
var buildTags string
// Info about restart notifications server
var rnsAddress string
var rnsPassword string
2013-07-24 12:01:14 +00:00
func init() {
cmdRun.Run = runApp
2014-02-21 17:51:18 +00:00
cmdRun.Flag.Var(&mainFiles, "main", "specify main go files")
2014-08-08 16:55:55 +00:00
cmdRun.Flag.Var(&gendoc, "gendoc", "auto generate the docs")
cmdRun.Flag.Var(&downdoc, "downdoc", "auto download swagger file when not exist")
2015-06-17 06:59:59 +00:00
cmdRun.Flag.Var(&excludedPaths, "e", "Excluded paths[].")
2016-02-10 19:39:02 +00:00
cmdRun.Flag.StringVar(&buildTags, "tags", "", "Build tags (https://golang.org/pkg/go/build/)")
cmdRun.Flag.StringVar(&rnsAddress, "rip", "", "Address of server which receive signal to restart application")
cmdRun.Flag.StringVar(&rnsPassword, "rpassword", "", "Password which should be passed to server to restart application")
2013-07-24 12:01:14 +00:00
}
var appname string
2014-08-15 09:38:51 +00:00
func runApp(cmd *Command, args []string) int {
fmt.Println("bee :" + version)
fmt.Println("beego :" + getbeegoVersion())
goversion, err := exec.Command("go", "version").Output()
if err != nil {
log.Fatal(err)
}
fmt.Println("Go :" + string(goversion))
2013-08-09 09:49:14 +00:00
exit := make(chan bool)
2013-09-11 22:35:24 +00:00
crupath, _ := os.Getwd()
if len(args) == 0 || args[0] == "watchall" {
2013-09-11 22:35:24 +00:00
appname = path.Base(crupath)
2013-10-30 23:39:44 +00:00
ColorLog("[INFO] Uses '%s' as 'appname'\n", appname)
2013-09-11 22:35:24 +00:00
} else {
appname = args[0]
2015-05-09 07:31:55 +00:00
ColorLog("[INFO] Uses '%s' as 'appname'\n", appname)
if strings.HasSuffix(appname, ".go") && isExist(path.Join(crupath, appname)) {
ColorLog("[WARN] The appname has conflic with crupath's file, do you want to build appname as %s\n", appname)
ColorLog("[INFO] Do you want to overwrite it? [yes|no]] ")
if !askForConfirmation() {
return 0
}
}
2013-07-24 12:01:14 +00:00
}
Debugf("current path:%s\n", crupath)
err = loadConfig()
2013-07-24 12:01:14 +00:00
if err != nil {
2013-10-30 23:39:44 +00:00
ColorLog("[ERRO] Fail to parse bee.json[ %s ]\n", err)
2013-07-24 12:01:14 +00:00
}
2013-11-04 03:16:15 +00:00
2013-07-24 12:01:14 +00:00
var paths []string
readAppDirectories(crupath, &paths)
// Because monitor files has some issues, we watch current directory
// and ignore non-go files.
2013-10-30 23:39:44 +00:00
gps := GetGOPATHs()
2013-09-11 23:06:04 +00:00
if len(gps) == 0 {
2013-10-30 23:39:44 +00:00
ColorLog("[ERRO] Fail to start[ %s ]\n", "$GOPATH is not set or empty")
2013-09-11 23:06:04 +00:00
os.Exit(2)
}
gopath := gps[0]
for _, p := range conf.DirStruct.Others {
paths = append(paths, strings.Replace(p, "$GOPATH", gopath, -1))
}
2013-07-24 12:01:14 +00:00
2014-02-21 17:51:18 +00:00
files := []string{}
for _, arg := range mainFiles {
if len(arg) > 0 {
files = append(files, arg)
}
}
isgenerate := gendoc == "true"
if rnsAddress != "" {
NewRestartNotificationServer(rnsAddress, []byte(rnsPassword), files, isgenerate)
}
NewWatcher(paths, files, isgenerate)
Autobuild(files, isgenerate)
if downdoc == "true" {
if _, err := os.Stat(path.Join(crupath, "swagger")); err != nil {
if os.IsNotExist(err) {
downloadFromUrl(swaggerlink, "swagger.zip")
2014-06-23 15:03:10 +00:00
unzipAndDelete("swagger.zip", "swagger")
}
}
}
2013-07-24 12:01:14 +00:00
for {
2013-08-09 09:49:14 +00:00
select {
case <-exit:
runtime.Goexit()
}
2013-07-24 12:01:14 +00:00
}
}
func readAppDirectories(directory string, paths *[]string) {
fileInfos, err := ioutil.ReadDir(directory)
if err != nil {
return
}
useDirectory := false
for _, fileInfo := range fileInfos {
if strings.HasSuffix(fileInfo.Name(), "docs") {
continue
}
2015-06-17 06:59:59 +00:00
if isExcluded(path.Join(directory, fileInfo.Name())) {
2015-06-17 06:59:59 +00:00
continue
}
if fileInfo.IsDir() == true && fileInfo.Name()[0] != '.' {
readAppDirectories(directory+"/"+fileInfo.Name(), paths)
continue
}
if useDirectory == true {
continue
}
if path.Ext(fileInfo.Name()) == ".go" {
*paths = append(*paths, directory)
useDirectory = true
}
}
return
}
2015-06-17 06:59:59 +00:00
// If a file is excluded
func isExcluded(filePath string) bool {
2015-06-17 06:59:59 +00:00
for _, p := range excludedPaths {
absP, err := path.Abs(p)
if err != nil {
ColorLog("[ERROR] Can not get absolute path of [ %s ]\n", p)
continue
}
absFilePath, err := path.Abs(filePath)
if err != nil {
ColorLog("[ERROR] Can not get absolute path of [ %s ]\n", filePath)
break
}
if strings.HasPrefix(absFilePath, absP) {
ColorLog("[INFO] Excluding from watching [ %s ]\n", filePath)
2015-06-17 06:59:59 +00:00
return true
}
}
return false
}
// NewRestartNotificationServer Creates tcp server which receive signal to restart app
func NewRestartNotificationServer(addr string, password []byte, files []string, isgenerate bool) error {
ColorLog("[INFO] Starting restart notification server on addres %s\n", addr)
ln, err := net.Listen("tcp", addr)
if err != nil {
ColorLog("[ERRO] Failt to start restart notification server [%s]\n", err)
return err
}
go func() {
for {
client, err := ln.Accept()
if err != nil {
ColorLog("[ERRO] Fail to accept client[ %s ]\n", err)
continue
}
if len(password) == 0 {
Autobuild(files, isgenerate)
client.Close()
continue
}
data, err := ioutil.ReadAll(client)
if err != nil {
ColorLog("[ERRO] Fail to read data from client[ %s ]\n", err)
}
data = bytes.TrimSpace(data)
if bytes.Compare(data, password) != 0 {
ColorLog("[ERRO] Invalid password to restart app: %v\n", string(data))
continue
}
Autobuild(files, isgenerate)
client.Close()
}
}()
return nil
}