bee/cmd/commands/run/watch.go

282 lines
6.7 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.
package run
2012-12-15 16:56:28 +00:00
import (
2013-12-15 23:08:51 +00:00
"bytes"
2012-12-15 16:56:28 +00:00
"os"
"os/exec"
"regexp"
"runtime"
2013-07-24 06:58:13 +00:00
"strings"
2013-07-06 07:30:57 +00:00
"sync"
"time"
"github.com/beego/bee/config"
beeLogger "github.com/beego/bee/logger"
"github.com/beego/bee/logger/colors"
"github.com/beego/bee/utils"
2016-12-08 17:31:24 +00:00
"github.com/fsnotify/fsnotify"
2012-12-15 16:56:28 +00:00
)
var (
2017-02-13 21:05:32 +00:00
cmd *exec.Cmd
state sync.Mutex
eventTime = make(map[string]int64)
scheduleTime time.Time
watchExts = config.Conf.WatchExts
watchExtsStatic = config.Conf.WatchExtsStatic
2017-02-13 21:05:32 +00:00
ignoredFilesRegExps = []string{
`.#(\w+).go$`,
`.(\w+).go.swp$`,
`(\w+).go~$`,
`(\w+).tmp$`,
`commentsRouter_controllers.go$`,
2017-02-13 21:05:32 +00:00
}
2012-12-15 16:56:28 +00:00
)
// NewWatcher starts an fsnotify Watcher on the specified paths
func NewWatcher(paths []string, files []string, isgenerate bool) {
2012-12-15 16:56:28 +00:00
watcher, err := fsnotify.NewWatcher()
if err != nil {
beeLogger.Log.Fatalf("Failed to create watcher: %s", err)
2012-12-15 16:56:28 +00:00
}
go func() {
for {
select {
2016-12-08 17:31:24 +00:00
case e := <-watcher.Events:
2017-02-13 21:05:32 +00:00
isBuild := true
2013-07-24 06:58:13 +00:00
if ifStaticFile(e.Name) && config.Conf.EnableReload {
2017-02-13 21:05:32 +00:00
sendReload(e.String())
continue
}
// Skip ignored files
if shouldIgnoreFile(e.Name) {
2013-07-24 06:58:13 +00:00
continue
}
if !shouldWatchFileWithExtension(e.Name) {
2013-07-30 07:23:14 +00:00
continue
}
2013-07-24 06:58:13 +00:00
2017-04-02 21:36:17 +00:00
mt := utils.GetFileModTime(e.Name)
if t := eventTime[e.Name]; mt == t {
beeLogger.Log.Hintf(colors.Bold("Skipping: ")+"%s", e.String())
2017-02-13 21:05:32 +00:00
isBuild = false
2013-07-06 07:30:57 +00:00
}
eventTime[e.Name] = mt
2013-07-06 07:30:57 +00:00
2017-02-13 21:05:32 +00:00
if isBuild {
beeLogger.Log.Hintf("Event fired: %s", e)
go func() {
2017-02-11 19:54:12 +00:00
// Wait 1s before autobuild until there is no file change.
scheduleTime = time.Now().Add(1 * time.Second)
2019-01-08 15:21:19 +00:00
time.Sleep(time.Until(scheduleTime))
AutoBuild(files, isgenerate)
2017-10-19 10:39:59 +00:00
if config.Conf.EnableReload {
// Wait 100ms more before refreshing the browser
time.Sleep(100 * time.Millisecond)
sendReload(e.String())
}
}()
2013-07-06 07:30:57 +00:00
}
2016-12-08 17:31:24 +00:00
case err := <-watcher.Errors:
beeLogger.Log.Warnf("Watcher error: %s", err.Error()) // No need to exit here
2012-12-15 16:56:28 +00:00
}
}
}()
beeLogger.Log.Info("Initializing watcher...")
2012-12-15 16:56:28 +00:00
for _, path := range paths {
beeLogger.Log.Hintf(colors.Bold("Watching: ")+"%s", path)
2016-12-08 17:31:24 +00:00
err = watcher.Add(path)
2012-12-15 16:56:28 +00:00
if err != nil {
beeLogger.Log.Fatalf("Failed to watch directory: %s", err)
2012-12-15 16:56:28 +00:00
}
}
}
// AutoBuild builds the specified set of files
func AutoBuild(files []string, isgenerate bool) {
2013-07-06 08:13:30 +00:00
state.Lock()
defer state.Unlock()
2012-12-15 16:56:28 +00:00
os.Chdir(currpath)
2013-07-25 08:26:54 +00:00
2013-11-04 03:16:15 +00:00
cmdName := "go"
var (
err error
stderr bytes.Buffer
)
2013-07-25 08:26:54 +00:00
// For applications use full import path like "github.com/.../.."
// are able to use "go install" to reduce build time.
if config.Conf.GoInstall {
icmd := exec.Command(cmdName, "install", "-v")
icmd.Stdout = os.Stdout
icmd.Stderr = os.Stderr
icmd.Env = append(os.Environ(), "GOGC=off")
icmd.Run()
}
2013-07-25 08:26:54 +00:00
if isgenerate {
beeLogger.Log.Info("Generating the docs...")
icmd := exec.Command("bee", "generate", "docs")
icmd.Env = append(os.Environ(), "GOGC=off")
err = icmd.Run()
if err != nil {
2017-03-10 17:43:57 +00:00
utils.Notify("", "Failed to generate the docs.")
beeLogger.Log.Errorf("Failed to generate the docs.")
return
}
beeLogger.Log.Success("Docs generated!")
}
appName := appname
2013-07-25 08:26:54 +00:00
if err == nil {
if runtime.GOOS == "windows" {
appName += ".exe"
}
2013-12-15 23:08:51 +00:00
2014-02-21 17:51:18 +00:00
args := []string{"build"}
args = append(args, "-o", appName)
2016-02-10 19:39:02 +00:00
if buildTags != "" {
args = append(args, "-tags", buildTags)
}
if buildLDFlags != "" {
args = append(args, "-ldflags", buildLDFlags)
}
2014-02-21 17:51:18 +00:00
args = append(args, files...)
bcmd := exec.Command(cmdName, args...)
bcmd.Env = append(os.Environ(), "GOGC=off")
bcmd.Stderr = &stderr
2013-12-15 23:08:51 +00:00
err = bcmd.Run()
if err != nil {
2017-03-10 17:43:57 +00:00
utils.Notify(stderr.String(), "Build Failed")
beeLogger.Log.Errorf("Failed to build the application: %s", stderr.String())
return
}
2013-07-25 08:26:54 +00:00
}
2013-07-06 08:13:30 +00:00
beeLogger.Log.Success("Built Successfully!")
Restart(appName)
2012-12-15 16:56:28 +00:00
}
// Kill kills the running command process
2012-12-15 16:56:28 +00:00
func Kill() {
2013-07-06 08:13:30 +00:00
defer func() {
if e := recover(); e != nil {
beeLogger.Log.Infof("Kill recover: %s", e)
2013-07-06 08:13:30 +00:00
}
}()
if cmd != nil && cmd.Process != nil {
// Windows does not support Interrupt
if runtime.GOOS == "windows" {
cmd.Process.Signal(os.Kill)
} else {
cmd.Process.Signal(os.Interrupt)
}
ch := make(chan struct{}, 1)
go func() {
cmd.Wait()
ch <- struct{}{}
}()
select {
case <-ch:
return
case <-time.After(10 * time.Second):
beeLogger.Log.Info("Timeout. Force kill cmd process")
2019-01-04 14:28:36 +00:00
err := cmd.Process.Kill()
if err != nil {
beeLogger.Log.Errorf("Error while killing cmd process: %s", err)
}
return
2013-12-15 23:08:51 +00:00
}
2012-12-15 16:56:28 +00:00
}
}
// Restart kills the running command process and starts it again
2013-07-06 07:30:57 +00:00
func Restart(appname string) {
beeLogger.Log.Debugf("Kill running process", utils.FILE(), utils.LINE())
2013-07-06 07:30:57 +00:00
Kill()
2013-07-06 08:13:30 +00:00
go Start(appname)
2013-07-06 07:30:57 +00:00
}
2013-07-06 08:13:30 +00:00
// Start starts the command process
2012-12-15 16:56:28 +00:00
func Start(appname string) {
beeLogger.Log.Infof("Restarting '%s'...", appname)
if !strings.Contains(appname, "./") {
appname = "./" + appname
}
2013-07-06 07:30:57 +00:00
2012-12-15 16:56:28 +00:00
cmd = exec.Command(appname)
2013-07-06 07:30:57 +00:00
cmd.Stdout = os.Stdout
cmd.Stderr = os.Stderr
2018-06-26 10:07:49 +00:00
if runargs != "" {
r := regexp.MustCompile("'.+'|\".+\"|\\S+")
m := r.FindAllString(runargs, -1)
cmd.Args = append([]string{appname}, m...)
} else {
cmd.Args = append([]string{appname}, config.Conf.CmdArgs...)
}
cmd.Env = append(os.Environ(), config.Conf.Envs...)
2013-07-06 07:30:57 +00:00
2013-07-06 08:13:30 +00:00
go cmd.Run()
beeLogger.Log.Successf("'%s' is running...", appname)
2013-08-24 11:00:08 +00:00
started <- true
2012-12-15 16:56:28 +00:00
}
2013-07-24 06:58:13 +00:00
2017-02-13 21:05:32 +00:00
func ifStaticFile(filename string) bool {
for _, s := range watchExtsStatic {
if strings.HasSuffix(filename, s) {
return true
}
}
return false
}
// shouldIgnoreFile ignores filenames generated by Emacs, Vim or SublimeText.
// It returns true if the file should be ignored, false otherwise.
func shouldIgnoreFile(filename string) bool {
for _, regex := range ignoredFilesRegExps {
r, err := regexp.Compile(regex)
if err != nil {
beeLogger.Log.Fatalf("Could not compile regular expression: %s", err)
}
if r.MatchString(filename) {
return true
}
continue
2013-07-24 06:58:13 +00:00
}
return false
}
2013-07-30 07:23:14 +00:00
// shouldWatchFileWithExtension returns true if the name of the file
// hash a suffix that should be watched.
func shouldWatchFileWithExtension(name string) bool {
2013-09-04 15:23:51 +00:00
for _, s := range watchExts {
if strings.HasSuffix(name, s) {
return true
}
2013-07-30 07:23:14 +00:00
}
return false
}