goctl add plugin support (#243)
* add plugin support * add plugin support * add plugin support * add plugin support * add plugin support * add plugin support * add plugin support * add plugin support * add plugin support * add plugin support * add plugin support * remove no need * add plugin support * rename * rename * add plugin support * refactor * update plugin * refactor * refactor * refactor * update plugin * newline Co-authored-by: anqiansong <anqiansong@xiaoheiban.cn>master
parent
7cd0463953
commit
cecd4b1b75
@ -0,0 +1,20 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/tal-tech/go-zero/tools/goctl/plugin"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
plugin, err := plugin.NewPlugin()
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if plugin.Api != nil {
|
||||||
|
fmt.Printf("api: %+v \n", plugin.Api)
|
||||||
|
}
|
||||||
|
fmt.Printf("dir: %s \n", plugin.Dir)
|
||||||
|
fmt.Println("Enjoy anything you want.")
|
||||||
|
}
|
@ -0,0 +1,166 @@
|
|||||||
|
package plugin
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/tal-tech/go-zero/tools/goctl/api/parser"
|
||||||
|
"github.com/tal-tech/go-zero/tools/goctl/api/spec"
|
||||||
|
"github.com/tal-tech/go-zero/tools/goctl/rpc/execx"
|
||||||
|
"github.com/tal-tech/go-zero/tools/goctl/util"
|
||||||
|
"github.com/urfave/cli"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
pluginArg = "_plugin"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Plugin struct {
|
||||||
|
Api *spec.ApiSpec
|
||||||
|
Style string
|
||||||
|
Dir string
|
||||||
|
}
|
||||||
|
|
||||||
|
func PluginCommand(c *cli.Context) error {
|
||||||
|
ex, err := os.Executable()
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var plugin = c.String("plugin")
|
||||||
|
if len(plugin) == 0 {
|
||||||
|
return errors.New("missing plugin")
|
||||||
|
}
|
||||||
|
|
||||||
|
transferData, err := prepareArgs(c)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
bin, download, err := getCommand(plugin)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if download {
|
||||||
|
defer func() {
|
||||||
|
_ = os.Remove(bin)
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
|
||||||
|
content, err := execx.Run(bin, filepath.Dir(ex), bytes.NewBuffer(transferData))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Println(content)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func prepareArgs(c *cli.Context) ([]byte, error) {
|
||||||
|
apiPath := c.String("api")
|
||||||
|
|
||||||
|
var transferData Plugin
|
||||||
|
if len(apiPath) > 0 && util.FileExists(apiPath) {
|
||||||
|
p, err := parser.NewParser(apiPath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
api, err := p.Parse()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
transferData.Api = api
|
||||||
|
}
|
||||||
|
|
||||||
|
dirAbs, err := filepath.Abs(c.String("dir"))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
transferData.Dir = dirAbs
|
||||||
|
transferData.Style = c.String("style")
|
||||||
|
data, err := json.Marshal(transferData)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return data, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func getCommand(arg string) (string, bool, error) {
|
||||||
|
p, err := exec.LookPath(arg)
|
||||||
|
if err == nil {
|
||||||
|
abs, err := filepath.Abs(p)
|
||||||
|
if err != nil {
|
||||||
|
return "", false, err
|
||||||
|
}
|
||||||
|
return abs, false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var defaultErr = errors.New("invalid plugin value " + arg)
|
||||||
|
if strings.HasPrefix(arg, "http") {
|
||||||
|
items := strings.Split(arg, "/")
|
||||||
|
if len(items) == 0 {
|
||||||
|
return "", false, defaultErr
|
||||||
|
}
|
||||||
|
|
||||||
|
filename, err := filepath.Abs(pluginArg + items[len(items)-1])
|
||||||
|
if err != nil {
|
||||||
|
return "", false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = downloadFile(filename, arg)
|
||||||
|
if err != nil {
|
||||||
|
return "", false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
os.Chmod(filename, os.ModePerm)
|
||||||
|
return filename, true, nil
|
||||||
|
}
|
||||||
|
return arg, false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func downloadFile(filepath string, url string) error {
|
||||||
|
resp, err := http.Get(url)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
_ = resp.Body.Close()
|
||||||
|
}()
|
||||||
|
|
||||||
|
out, err := os.Create(filepath)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
_ = out.Close()
|
||||||
|
}()
|
||||||
|
|
||||||
|
_, err = io.Copy(out, resp.Body)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewPlugin() (*Plugin, error) {
|
||||||
|
var plugin Plugin
|
||||||
|
content, err := ioutil.ReadAll(os.Stdin)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
err = json.Unmarshal(content, &plugin)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &plugin, nil
|
||||||
|
}
|
Loading…
Reference in New Issue