2020-08-04 14:28:25 +08:00
package main
import (
"bufio"
2020-09-25 20:05:25 +08:00
"bytes"
2021-02-11 15:18:46 +08:00
"context"
2020-08-04 14:28:25 +08:00
"database/sql"
2021-07-17 11:12:17 +08:00
"encoding/json"
2021-12-18 11:33:17 +08:00
"errors"
2021-01-07 18:32:17 +08:00
"flag"
2020-08-04 14:28:25 +08:00
"fmt"
2021-07-17 11:12:17 +08:00
"github.com/hashicorp/go-version"
2021-12-14 17:42:12 +08:00
"github.com/zhenorzz/goploy/config"
2020-08-04 14:28:25 +08:00
"github.com/zhenorzz/goploy/core"
"github.com/zhenorzz/goploy/model"
"github.com/zhenorzz/goploy/route"
"github.com/zhenorzz/goploy/task"
"github.com/zhenorzz/goploy/utils"
"github.com/zhenorzz/goploy/ws"
2021-02-11 15:18:46 +08:00
"io/ioutil"
2020-08-04 14:28:25 +08:00
"log"
"net/http"
"os"
2020-09-25 20:05:25 +08:00
"os/exec"
2021-02-11 15:18:46 +08:00
"os/signal"
"path"
2021-12-17 17:55:26 +08:00
"runtime"
2021-02-11 15:18:46 +08:00
"strconv"
"syscall"
2020-08-04 14:28:25 +08:00
"time"
_ "github.com/go-sql-driver/mysql"
)
2021-01-07 18:32:17 +08:00
var (
2021-07-17 11:12:17 +08:00
help bool
v bool
s string
2021-01-07 18:32:17 +08:00
)
2022-01-26 17:00:53 +08:00
const appVersion = "1.4.5"
2021-07-17 11:12:17 +08:00
2021-01-07 18:32:17 +08:00
func init ( ) {
flag . StringVar ( & core . AssetDir , "asset-dir" , "" , "default: ./" )
2021-02-11 15:18:46 +08:00
flag . StringVar ( & s , "s" , "" , "stop" )
flag . BoolVar ( & help , "help" , false , "list available subcommands and some concept guides" )
2021-07-17 11:12:17 +08:00
flag . BoolVar ( & v , "version" , false , "show goploy version" )
2021-01-07 18:32:17 +08:00
// 改变默认的 Usage
flag . Usage = usage
}
func usage ( ) {
2021-07-17 11:12:17 +08:00
fmt . Fprintf ( os . Stderr , "Options:\n" )
2021-01-07 18:32:17 +08:00
flag . PrintDefaults ( )
}
2020-08-04 14:28:25 +08:00
func main ( ) {
2021-01-07 18:32:17 +08:00
flag . Parse ( )
if help {
flag . Usage ( )
return
}
2021-07-17 11:12:17 +08:00
if v {
println ( appVersion )
2021-02-26 16:45:13 +08:00
return
}
2021-02-11 15:18:46 +08:00
handleClientSignal ( )
2020-08-15 13:38:06 +08:00
println ( `
______ __
/ ____ / ___ ____ / / ___ __ __
/ / __ / __ \ / __ \ / / __ \ / / / /
/ / _ / / / _ / / / _ / / / / _ / / / _ / /
\ ____ / \ ____ / . ___ / _ / \ ____ / \ __ , /
2021-07-17 11:12:17 +08:00
/ _ / / ____ / ` + appVersion + "\n" )
2020-08-15 16:35:00 +08:00
install ( )
2021-12-14 17:42:12 +08:00
config . Create ( core . GetConfigFile ( ) )
2021-08-20 11:26:37 +08:00
model . Init ( )
if err := model . Update ( appVersion ) ; err != nil {
println ( err . Error ( ) )
}
pid := strconv . Itoa ( os . Getpid ( ) )
2021-07-17 11:12:17 +08:00
_ = ioutil . WriteFile ( path . Join ( core . GetAssetDir ( ) , "goploy.pid" ) , [ ] byte ( pid ) , 0755 )
2021-01-13 17:12:40 +08:00
println ( "Start at " + time . Now ( ) . String ( ) )
2021-02-26 16:45:13 +08:00
println ( "goploy -h for more help" )
println ( "Current pid: " + pid )
2021-12-14 17:42:12 +08:00
println ( "Config Loaded: " + core . GetConfigFile ( ) )
2021-12-17 17:55:26 +08:00
println ( "Env: " + config . Toml . Env )
2021-12-14 17:42:12 +08:00
println ( "Log: " + config . Toml . Log . Path )
println ( "Listen: " + config . Toml . Web . Port )
2021-01-13 17:12:40 +08:00
println ( "Running..." )
2020-08-04 14:28:25 +08:00
core . CreateValidator ( )
ws . Init ( )
route . Init ( )
task . Init ( )
2022-01-08 21:35:57 +08:00
go checkUpdate ( )
2021-02-11 15:18:46 +08:00
// server
srv := http . Server {
2021-12-14 17:42:12 +08:00
Addr : ":" + config . Toml . Web . Port ,
2021-02-11 15:18:46 +08:00
}
go func ( ) {
c := make ( chan os . Signal , 1 )
2022-01-08 21:35:57 +08:00
// SIGINT Ctrl+C
// SIGTERM A generic signal used to cause program termination
signal . Notify ( c , syscall . SIGINT , syscall . SIGTERM )
2021-02-11 15:18:46 +08:00
println ( "Received the signal: " + ( <- c ) . String ( ) )
2022-01-08 21:35:57 +08:00
2021-02-11 15:18:46 +08:00
println ( "Server is trying to shutdown, wait for a minute" )
2022-01-08 21:35:57 +08:00
ctx , cancel := context . WithTimeout ( context . Background ( ) , config . Toml . APP . ShutdownTimeout * time . Second )
2021-02-11 15:18:46 +08:00
defer cancel ( )
if err := srv . Shutdown ( ctx ) ; err != nil {
2022-01-08 21:35:57 +08:00
println ( "Server shutdown timeout, err: %v\n" , err )
2021-02-11 15:18:46 +08:00
}
println ( "Server shutdown gracefully" )
2021-02-12 10:47:16 +08:00
println ( "Task is trying to shutdown, wait for a minute" )
if err := task . Shutdown ( ctx ) ; err != nil {
2022-01-08 21:35:57 +08:00
println ( "Task shutdown timeout, err: %v\n" , err )
2021-02-12 10:47:16 +08:00
}
println ( "Task shutdown gracefully" )
2021-02-11 15:18:46 +08:00
} ( )
2021-02-12 16:21:05 +08:00
if err := srv . ListenAndServe ( ) ; err != nil && err != http . ErrServerClosed {
2021-02-11 15:18:46 +08:00
log . Fatal ( "ListenAndServe: " , err . Error ( ) )
}
2021-07-17 11:12:17 +08:00
_ = os . Remove ( path . Join ( core . GetAssetDir ( ) , "goploy.pid" ) )
2022-01-08 21:35:57 +08:00
println ( "shutdown success" )
2021-02-11 15:18:46 +08:00
return
2020-08-04 14:28:25 +08:00
}
func install ( ) {
2021-12-14 17:42:12 +08:00
_ , err := os . Stat ( core . GetConfigFile ( ) )
2020-08-04 14:28:25 +08:00
if err == nil || os . IsExist ( err ) {
2021-01-13 17:12:40 +08:00
println ( "The configuration file already exists, no need to reinstall (if you need to reinstall, please back up the database `goploy` first, delete the .env file, then restart.)" )
2020-08-04 14:28:25 +08:00
return
}
2021-12-14 17:42:12 +08:00
cfg := config . Config {
Env : "production" ,
2022-01-08 21:35:57 +08:00
APP : config . APPConfig { DeployLimit : int32 ( runtime . NumCPU ( ) ) , ShutdownTimeout : 10 } ,
2021-12-14 17:42:12 +08:00
Cookie : config . CookieConfig { Name : "goploy_token" , Expire : 86400 } ,
JWT : config . JWTConfig { Key : time . Now ( ) . String ( ) } ,
2021-12-25 16:39:23 +08:00
DB : config . DBConfig { Type : "mysql" , Host : "127.0.0.1" , Port : "3306" , Database : "goploy" } ,
2021-12-14 17:42:12 +08:00
Log : config . LogConfig { Path : "stdout" } ,
Web : config . WebConfig { Port : "80" } ,
}
2021-01-13 17:12:40 +08:00
println ( "Installation guide ↓" )
2020-09-25 20:05:25 +08:00
var stdout bytes . Buffer
var stderr bytes . Buffer
cmd := exec . Command ( "rsync" , "--version" )
cmd . Stdout = & stdout
cmd . Stderr = & stderr
if err := cmd . Run ( ) ; err != nil {
println ( err . Error ( ) + ", detail: " + stderr . String ( ) )
panic ( "Please check if rsync is installed correctly, see https://rsync.samba.org/download.html" )
}
git := utils . GIT { }
if err := git . Run ( "--version" ) ; err != nil {
println ( err . Error ( ) + ", detail: " + git . Err . String ( ) )
panic ( "Please check if git is installed correctly, see https://git-scm.com/downloads" )
}
2020-08-04 14:28:25 +08:00
inputReader := bufio . NewReader ( os . Stdin )
2020-08-15 13:38:06 +08:00
println ( "Installation guidelines (Enter to confirm input)" )
println ( "Please enter the mysql user:" )
2020-08-04 14:28:25 +08:00
mysqlUser , err := inputReader . ReadString ( '\n' )
if err != nil {
panic ( "There were errors reading, exiting program." )
}
2021-12-14 17:42:12 +08:00
cfg . DB . User = utils . ClearNewline ( mysqlUser )
2020-08-15 13:38:06 +08:00
println ( "Please enter the mysql password:" )
2020-08-04 14:28:25 +08:00
mysqlPassword , err := inputReader . ReadString ( '\n' )
if err != nil {
panic ( "There were errors reading, exiting program." )
}
mysqlPassword = utils . ClearNewline ( mysqlPassword )
if len ( mysqlPassword ) != 0 {
2021-12-14 17:42:12 +08:00
cfg . DB . Password = mysqlPassword
2020-08-04 14:28:25 +08:00
}
2020-08-15 13:38:06 +08:00
println ( "Please enter the mysql host(default 127.0.0.1, without port):" )
2020-08-04 14:28:25 +08:00
mysqlHost , err := inputReader . ReadString ( '\n' )
if err != nil {
panic ( "There were errors reading, exiting program." )
}
mysqlHost = utils . ClearNewline ( mysqlHost )
2021-12-14 17:42:12 +08:00
if len ( mysqlHost ) != 0 {
cfg . DB . Host = mysqlHost
2020-08-04 14:28:25 +08:00
}
2020-08-15 13:38:06 +08:00
println ( "Please enter the mysql port(default 3306):" )
2020-08-04 14:28:25 +08:00
mysqlPort , err := inputReader . ReadString ( '\n' )
if err != nil {
panic ( "There were errors reading, exiting program." )
}
mysqlPort = utils . ClearNewline ( mysqlPort )
2021-12-14 17:42:12 +08:00
if len ( mysqlPort ) != 0 {
2021-12-14 19:04:07 +08:00
cfg . DB . Port = mysqlPort
2020-08-04 14:28:25 +08:00
}
2021-05-30 20:21:18 +08:00
println ( "Please enter the absolute path of the log directory(default stdout):" )
2020-08-04 14:28:25 +08:00
logPath , err := inputReader . ReadString ( '\n' )
if err != nil {
panic ( "There were errors reading, exiting program." )
}
logPath = utils . ClearNewline ( logPath )
2021-12-14 17:42:12 +08:00
if len ( logPath ) != 0 {
cfg . Log . Path = logPath
2020-08-04 14:28:25 +08:00
}
2020-08-15 13:38:06 +08:00
println ( "Please enter the listening port(default 80):" )
2020-08-04 14:28:25 +08:00
port , err := inputReader . ReadString ( '\n' )
if err != nil {
panic ( "There were errors reading, exiting program." )
}
port = utils . ClearNewline ( port )
2021-12-14 17:42:12 +08:00
if len ( port ) != 0 {
cfg . Web . Port = port
2020-08-04 14:28:25 +08:00
}
2020-08-15 13:38:06 +08:00
println ( "Start to install the database..." )
2020-08-04 14:28:25 +08:00
2021-12-14 17:42:12 +08:00
db , err := sql . Open ( cfg . DB . Type , fmt . Sprintf (
2021-12-14 19:04:07 +08:00
"%s:%s@(%s:%s)/?charset=utf8mb4,utf8\n" ,
2021-12-14 17:42:12 +08:00
cfg . DB . User ,
cfg . DB . Password ,
cfg . DB . Host ,
cfg . DB . Port ) )
2020-08-04 14:28:25 +08:00
if err != nil {
panic ( err )
}
defer db . Close ( )
2021-07-17 11:12:17 +08:00
if err := model . ImportSQL ( db , "sql/goploy.sql" ) ; err != nil {
2020-08-04 14:28:25 +08:00
panic ( err )
}
2020-08-15 13:38:06 +08:00
println ( "Database installation is complete" )
println ( "Start writing configuration file..." )
2021-12-14 17:42:12 +08:00
err = config . Write ( core . GetConfigFile ( ) , cfg )
2020-08-04 14:28:25 +08:00
if err != nil {
2021-12-14 17:42:12 +08:00
panic ( "Write config file error, " + err . Error ( ) )
2020-08-04 14:28:25 +08:00
}
2020-08-15 13:38:06 +08:00
println ( "Write configuration file completed" )
2020-08-04 14:28:25 +08:00
}
2021-02-11 15:18:46 +08:00
func handleClientSignal ( ) {
switch s {
case "stop" :
2021-12-18 11:33:17 +08:00
pidFile := path . Join ( core . GetAssetDir ( ) , "goploy.pid" )
pidStr , err := ioutil . ReadFile ( pidFile )
2021-02-11 15:18:46 +08:00
if err != nil {
log . Fatal ( "handle stop, " , err . Error ( ) , ", may be the server not start" )
}
pid , _ := strconv . Atoi ( string ( pidStr ) )
process , err := os . FindProcess ( pid )
if err != nil {
log . Fatal ( "handle stop, " , err . Error ( ) , ", may be the server not start" )
}
err = process . Signal ( syscall . SIGTERM )
if err != nil {
log . Fatal ( "handle stop, " , err . Error ( ) )
}
2021-12-18 11:33:17 +08:00
println ( "App is trying to shutdown, wait for a minute" )
2021-12-18 14:14:56 +08:00
for i := 0 ; i < 5 ; i ++ {
2021-12-18 11:33:17 +08:00
time . Sleep ( time . Second )
if _ , err := os . Stat ( pidFile ) ; errors . Is ( err , os . ErrNotExist ) {
println ( "Success" )
break
} else if err != nil {
log . Fatal ( "handle stop, " , err . Error ( ) )
}
}
2021-02-11 15:18:46 +08:00
os . Exit ( 1 )
}
}
2021-07-17 11:12:17 +08:00
func checkUpdate ( ) {
resp , err := http . Get ( "https://api.github.com/repos/zhenorzz/goploy/releases/latest" )
if err != nil {
println ( "Check failed" )
println ( err . Error ( ) )
return
}
defer resp . Body . Close ( )
body , err := ioutil . ReadAll ( resp . Body )
if err != nil {
println ( "Check failed" )
println ( err . Error ( ) )
return
}
var result map [ string ] interface { }
if err := json . Unmarshal ( body , & result ) ; err != nil {
println ( "Check failed" )
println ( err . Error ( ) )
return
}
tagName := result [ "tag_name" ] . ( string )
tagVer , err := version . NewVersion ( tagName )
if err != nil {
println ( "Check version error" )
println ( err . Error ( ) )
return
}
currentVer , _ := version . NewVersion ( appVersion )
if tagVer . GreaterThan ( currentVer ) {
println ( "New release available" )
println ( result [ "html_url" ] . ( string ) )
}
}