mirror of
https://github.com/tickstep/aliyunpan.git
synced 2025-01-23 22:42:15 +08:00
508 lines
12 KiB
Go
508 lines
12 KiB
Go
|
// Copyright (c) 2020 tickstep.
|
|||
|
//
|
|||
|
// 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 main
|
|||
|
|
|||
|
import (
|
|||
|
"fmt"
|
|||
|
"github.com/tickstep/aliyunpan-api/aliyunpan"
|
|||
|
"github.com/tickstep/aliyunpan/cmder"
|
|||
|
"os"
|
|||
|
"path"
|
|||
|
"path/filepath"
|
|||
|
"runtime"
|
|||
|
"sort"
|
|||
|
"strings"
|
|||
|
"time"
|
|||
|
"unicode"
|
|||
|
|
|||
|
"github.com/peterh/liner"
|
|||
|
"github.com/tickstep/aliyunpan/cmder/cmdliner"
|
|||
|
"github.com/tickstep/aliyunpan/cmder/cmdliner/args"
|
|||
|
"github.com/tickstep/aliyunpan/cmder/cmdutil"
|
|||
|
"github.com/tickstep/aliyunpan/cmder/cmdutil/escaper"
|
|||
|
"github.com/tickstep/aliyunpan/internal/command"
|
|||
|
"github.com/tickstep/aliyunpan/internal/config"
|
|||
|
"github.com/tickstep/aliyunpan/internal/panupdate"
|
|||
|
"github.com/tickstep/aliyunpan/internal/utils"
|
|||
|
"github.com/tickstep/library-go/converter"
|
|||
|
"github.com/tickstep/library-go/logger"
|
|||
|
"github.com/urfave/cli"
|
|||
|
)
|
|||
|
|
|||
|
const (
|
|||
|
// NameShortDisplayNum 文件名缩略显示长度
|
|||
|
NameShortDisplayNum = 16
|
|||
|
)
|
|||
|
|
|||
|
var (
|
|||
|
// Version 版本号
|
|||
|
Version = "v0.0.2"
|
|||
|
|
|||
|
historyFilePath = filepath.Join(config.GetConfigDir(), "aliyunpan_command_history.txt")
|
|||
|
|
|||
|
isCli bool
|
|||
|
)
|
|||
|
|
|||
|
func init() {
|
|||
|
config.AppVersion = Version
|
|||
|
cmdutil.ChWorkDir()
|
|||
|
|
|||
|
err := config.Config.Init()
|
|||
|
switch err {
|
|||
|
case nil:
|
|||
|
case config.ErrConfigFileNoPermission, config.ErrConfigContentsParseError:
|
|||
|
fmt.Fprintf(os.Stderr, "FATAL ERROR: config file error: %s\n", err)
|
|||
|
os.Exit(1)
|
|||
|
default:
|
|||
|
fmt.Printf("WARNING: config init error: %s\n", err)
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
func checkLoginExpiredAndRelogin() {
|
|||
|
cmder.ReloadConfigFunc(nil)
|
|||
|
activeUser := config.Config.ActiveUser()
|
|||
|
if activeUser == nil || activeUser.UserId == "" {
|
|||
|
// maybe expired, try to login
|
|||
|
cmder.TryLogin()
|
|||
|
} else {
|
|||
|
// refresh expired token
|
|||
|
if activeUser.PanClient() != nil {
|
|||
|
if len(activeUser.WebToken.RefreshToken) > 0 {
|
|||
|
cz := time.FixedZone("CST", 8*3600) // 东8区
|
|||
|
expiredTime, _ := time.ParseInLocation("2006-01-02 15:04:05", activeUser.WebToken.ExpireTime, cz)
|
|||
|
now := time.Now()
|
|||
|
if (expiredTime.Unix() - now.Unix()) <= (10 * 60) {
|
|||
|
// need refresh token
|
|||
|
logger.Verboseln("access token expired, get new from refresh token")
|
|||
|
if wt, er := aliyunpan.GetAccessTokenFromRefreshToken(activeUser.RefreshToken); er == nil {
|
|||
|
activeUser.WebToken = *wt
|
|||
|
activeUser.PanClient().UpdateToken(*wt)
|
|||
|
logger.Verboseln("get new access token success")
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
cmder.SaveConfigFunc(nil)
|
|||
|
}
|
|||
|
|
|||
|
func main() {
|
|||
|
defer config.Config.Close()
|
|||
|
|
|||
|
// check & relogin
|
|||
|
checkLoginExpiredAndRelogin()
|
|||
|
|
|||
|
// check token expired task
|
|||
|
go func() {
|
|||
|
for {
|
|||
|
time.Sleep(time.Duration(5) * time.Minute)
|
|||
|
//time.Sleep(time.Duration(5) * time.Second)
|
|||
|
checkLoginExpiredAndRelogin()
|
|||
|
}
|
|||
|
}()
|
|||
|
|
|||
|
app := cli.NewApp()
|
|||
|
cmder.SetApp(app)
|
|||
|
|
|||
|
app.Name = "aliyunpan"
|
|||
|
app.Version = Version
|
|||
|
app.Author = "tickstep/aliyunpan: https://github.com/tickstep/aliyunpan"
|
|||
|
app.Copyright = "(c) 2021 tickstep."
|
|||
|
app.Usage = "阿里云盘客户端 for " + runtime.GOOS + "/" + runtime.GOARCH
|
|||
|
app.Description = `aliyunpan 使用Go语言编写的阿里云盘命令行客户端, 为操作阿里云盘, 提供实用功能.
|
|||
|
具体功能, 参见 COMMANDS 列表
|
|||
|
|
|||
|
------------------------------------------------------------------------------
|
|||
|
前往 https://github.com/tickstep/aliyunpan 以获取更多帮助信息!
|
|||
|
前往 https://github.com/tickstep/aliyunpan/releases 以获取程序更新信息!
|
|||
|
------------------------------------------------------------------------------
|
|||
|
|
|||
|
交流反馈:
|
|||
|
提交Issue: https://github.com/tickstep/aliyunpan/issues
|
|||
|
联系邮箱: tickstep@outlook.com`
|
|||
|
|
|||
|
// 全局options
|
|||
|
app.Flags = []cli.Flag{
|
|||
|
cli.BoolFlag{
|
|||
|
Name: "verbose",
|
|||
|
Usage: "启用调试",
|
|||
|
EnvVar: config.EnvVerbose,
|
|||
|
Destination: &logger.IsVerbose,
|
|||
|
},
|
|||
|
}
|
|||
|
|
|||
|
// 进入交互CLI命令行界面
|
|||
|
app.Action = func(c *cli.Context) {
|
|||
|
if c.NArg() != 0 {
|
|||
|
fmt.Printf("未找到命令: %s\n运行命令 %s help 获取帮助\n", c.Args().Get(0), app.Name)
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
os.Setenv(config.EnvVerbose, c.String("verbose"))
|
|||
|
isCli = true
|
|||
|
logger.Verbosef("提示: 你已经开启VERBOSE调试日志\n\n")
|
|||
|
|
|||
|
var (
|
|||
|
line = cmdliner.NewLiner()
|
|||
|
err error
|
|||
|
)
|
|||
|
|
|||
|
line.History, err = cmdliner.NewLineHistory(historyFilePath)
|
|||
|
if err != nil {
|
|||
|
fmt.Printf("警告: 读取历史命令文件错误, %s\n", err)
|
|||
|
}
|
|||
|
|
|||
|
line.ReadHistory()
|
|||
|
defer func() {
|
|||
|
line.DoWriteHistory()
|
|||
|
line.Close()
|
|||
|
}()
|
|||
|
|
|||
|
// tab 自动补全命令
|
|||
|
line.State.SetCompleter(func(line string) (s []string) {
|
|||
|
var (
|
|||
|
lineArgs = args.Parse(line)
|
|||
|
numArgs = len(lineArgs)
|
|||
|
acceptCompleteFileCommands = []string{
|
|||
|
"cd", "cp", "xcp", "download", "ls", "mkdir", "mv", "pwd", "rename", "rm", "share", "upload", "login", "loglist", "logout",
|
|||
|
"clear", "quit", "exit", "quota", "who", "sign", "update", "who", "su", "config",
|
|||
|
"drive", "export", "import", "backup",
|
|||
|
}
|
|||
|
closed = strings.LastIndex(line, " ") == len(line)-1
|
|||
|
)
|
|||
|
|
|||
|
for _, cmd := range app.Commands {
|
|||
|
for _, name := range cmd.Names() {
|
|||
|
if !strings.HasPrefix(name, line) {
|
|||
|
continue
|
|||
|
}
|
|||
|
|
|||
|
s = append(s, name+" ")
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
switch numArgs {
|
|||
|
case 0:
|
|||
|
return
|
|||
|
case 1:
|
|||
|
if !closed {
|
|||
|
return
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
thisCmd := app.Command(lineArgs[0])
|
|||
|
if thisCmd == nil {
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
if !cmdutil.ContainsString(acceptCompleteFileCommands, thisCmd.FullName()) {
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
var (
|
|||
|
activeUser = config.Config.ActiveUser()
|
|||
|
runeFunc = unicode.IsSpace
|
|||
|
//cmdRuneFunc = func(r rune) bool {
|
|||
|
// switch r {
|
|||
|
// case '\'', '"':
|
|||
|
// return true
|
|||
|
// }
|
|||
|
// return unicode.IsSpace(r)
|
|||
|
//}
|
|||
|
targetPath string
|
|||
|
)
|
|||
|
|
|||
|
if !closed {
|
|||
|
targetPath = lineArgs[numArgs-1]
|
|||
|
escaper.EscapeStringsByRuneFunc(lineArgs[:numArgs-1], runeFunc) // 转义
|
|||
|
} else {
|
|||
|
escaper.EscapeStringsByRuneFunc(lineArgs, runeFunc)
|
|||
|
}
|
|||
|
|
|||
|
switch {
|
|||
|
case targetPath == "." || strings.HasSuffix(targetPath, "/."):
|
|||
|
s = append(s, line+"/")
|
|||
|
return
|
|||
|
case targetPath == ".." || strings.HasSuffix(targetPath, "/.."):
|
|||
|
s = append(s, line+"/")
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
var (
|
|||
|
targetDir string
|
|||
|
isAbs = path.IsAbs(targetPath)
|
|||
|
isDir = strings.LastIndex(targetPath, "/") == len(targetPath)-1
|
|||
|
)
|
|||
|
|
|||
|
if isAbs {
|
|||
|
targetDir = path.Dir(targetPath)
|
|||
|
} else {
|
|||
|
targetDir = path.Join(activeUser.Workdir, targetPath)
|
|||
|
if !isDir {
|
|||
|
targetDir = path.Dir(targetDir)
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return
|
|||
|
})
|
|||
|
|
|||
|
fmt.Printf("提示: 方向键上下可切换历史命令.\n")
|
|||
|
fmt.Printf("提示: Ctrl + A / E 跳转命令 首 / 尾.\n")
|
|||
|
fmt.Printf("提示: 输入 help 获取帮助.\n")
|
|||
|
|
|||
|
// check update
|
|||
|
cmder.ReloadConfigFunc(c)
|
|||
|
if config.Config.UpdateCheckInfo.LatestVer != "" {
|
|||
|
if utils.ParseVersionNum(config.Config.UpdateCheckInfo.LatestVer) > utils.ParseVersionNum(config.AppVersion) {
|
|||
|
fmt.Printf("\n当前的软件版本为:%s, 现在有新版本 %s 可供更新,强烈推荐进行更新!(可以输入 update 命令进行更新)\n\n",
|
|||
|
config.AppVersion, config.Config.UpdateCheckInfo.LatestVer)
|
|||
|
}
|
|||
|
}
|
|||
|
go func() {
|
|||
|
latestCheckTime := config.Config.UpdateCheckInfo.CheckTime
|
|||
|
nowTime := time.Now().Unix()
|
|||
|
secsOf12Hour := int64(43200)
|
|||
|
if (nowTime - latestCheckTime) > secsOf12Hour {
|
|||
|
releaseInfo := panupdate.GetLatestReleaseInfo(false)
|
|||
|
if releaseInfo == nil {
|
|||
|
logger.Verboseln("获取版本信息失败!")
|
|||
|
return
|
|||
|
}
|
|||
|
config.Config.UpdateCheckInfo.LatestVer = releaseInfo.TagName
|
|||
|
config.Config.UpdateCheckInfo.CheckTime = nowTime
|
|||
|
|
|||
|
// save
|
|||
|
cmder.SaveConfigFunc(c)
|
|||
|
}
|
|||
|
}()
|
|||
|
|
|||
|
for {
|
|||
|
var (
|
|||
|
prompt string
|
|||
|
activeUser = config.Config.ActiveUser()
|
|||
|
)
|
|||
|
|
|||
|
if activeUser == nil {
|
|||
|
activeUser = cmder.TryLogin()
|
|||
|
}
|
|||
|
|
|||
|
if activeUser != nil && activeUser.Nickname != "" {
|
|||
|
// 格式: aliyunpan:<工作目录> <UserName>$
|
|||
|
// 工作目录太长时, 会自动缩略
|
|||
|
wd := "/"
|
|||
|
if activeUser.IsFileDriveActive() {
|
|||
|
wd = activeUser.Workdir
|
|||
|
prompt = app.Name + ":" + converter.ShortDisplay(path.Base(wd), NameShortDisplayNum) + " " + activeUser.Nickname + "$ "
|
|||
|
} else if activeUser.IsAlbumDriveActive() {
|
|||
|
wd = activeUser.AlbumWorkdir
|
|||
|
prompt = app.Name + ":" + converter.ShortDisplay(path.Base(wd), NameShortDisplayNum) + " " + activeUser.Nickname + "(相册)$ "
|
|||
|
}
|
|||
|
|
|||
|
} else {
|
|||
|
// aliyunpan >
|
|||
|
prompt = app.Name + " > "
|
|||
|
}
|
|||
|
|
|||
|
commandLine, err := line.State.Prompt(prompt)
|
|||
|
switch err {
|
|||
|
case liner.ErrPromptAborted:
|
|||
|
return
|
|||
|
case nil:
|
|||
|
// continue
|
|||
|
default:
|
|||
|
fmt.Println(err)
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
line.State.AppendHistory(commandLine)
|
|||
|
|
|||
|
cmdArgs := args.Parse(commandLine)
|
|||
|
if len(cmdArgs) == 0 {
|
|||
|
continue
|
|||
|
}
|
|||
|
|
|||
|
s := []string{os.Args[0]}
|
|||
|
s = append(s, cmdArgs...)
|
|||
|
|
|||
|
// 恢复原始终端状态
|
|||
|
// 防止运行命令时程序被结束, 终端出现异常
|
|||
|
line.Pause()
|
|||
|
c.App.Run(s)
|
|||
|
line.Resume()
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// 命令配置和对应的处理func
|
|||
|
app.Commands = []cli.Command{
|
|||
|
// 登录账号 login
|
|||
|
command.CmdLogin(),
|
|||
|
|
|||
|
// 退出登录帐号 logout
|
|||
|
command.CmdLogout(),
|
|||
|
|
|||
|
// 列出帐号列表 loglist
|
|||
|
command.CmdLoglist(),
|
|||
|
|
|||
|
// 切换网盘 drive
|
|||
|
command.CmdDrive(),
|
|||
|
|
|||
|
// 切换阿里账号 su
|
|||
|
command.CmdSu(),
|
|||
|
|
|||
|
// 获取当前帐号 who
|
|||
|
command.CmdWho(),
|
|||
|
|
|||
|
// 获取当前帐号空间配额 quota
|
|||
|
command.CmdQuota(),
|
|||
|
|
|||
|
// 切换工作目录 cd
|
|||
|
command.CmdCd(),
|
|||
|
|
|||
|
// 输出工作目录 pwd
|
|||
|
command.CmdPwd(),
|
|||
|
|
|||
|
// 列出目录 ls
|
|||
|
command.CmdLs(),
|
|||
|
|
|||
|
// 创建目录 mkdir
|
|||
|
command.CmdMkdir(),
|
|||
|
|
|||
|
// 删除文件/目录 rm
|
|||
|
command.CmdRm(),
|
|||
|
|
|||
|
//// 拷贝文件/目录 cp
|
|||
|
//command.CmdCp(),
|
|||
|
//
|
|||
|
//// 拷贝文件/目录到个人云/家庭云 xcp
|
|||
|
//command.CmdXcp(),
|
|||
|
|
|||
|
// 移动文件/目录 mv
|
|||
|
command.CmdMv(),
|
|||
|
|
|||
|
// 重命名文件 rename
|
|||
|
command.CmdRename(),
|
|||
|
|
|||
|
// 分享文件/目录 share
|
|||
|
command.CmdShare(),
|
|||
|
|
|||
|
// 备份 backup
|
|||
|
command.CmdBackup(),
|
|||
|
|
|||
|
// 上传文件/目录 upload
|
|||
|
command.CmdUpload(),
|
|||
|
|
|||
|
// 手动秒传
|
|||
|
command.CmdRapidUpload(),
|
|||
|
|
|||
|
// 下载文件/目录 download
|
|||
|
command.CmdDownload(),
|
|||
|
|
|||
|
// 导出文件/目录元数据 export
|
|||
|
command.CmdExport(),
|
|||
|
|
|||
|
// 导入文件 import
|
|||
|
command.CmdImport(),
|
|||
|
|
|||
|
// 回收站
|
|||
|
command.CmdRecycle(),
|
|||
|
|
|||
|
// 显示和修改程序配置项 config
|
|||
|
command.CmdConfig(),
|
|||
|
|
|||
|
// 工具箱 tool
|
|||
|
command.CmdTool(),
|
|||
|
|
|||
|
// 清空控制台 clear
|
|||
|
{
|
|||
|
Name: "clear",
|
|||
|
Aliases: []string{"cls"},
|
|||
|
Usage: "清空控制台",
|
|||
|
UsageText: app.Name + " clear",
|
|||
|
Description: "清空控制台屏幕",
|
|||
|
Category: "其他",
|
|||
|
Action: func(c *cli.Context) error {
|
|||
|
cmdliner.ClearScreen()
|
|||
|
return nil
|
|||
|
},
|
|||
|
},
|
|||
|
|
|||
|
// 检测程序更新 update
|
|||
|
{
|
|||
|
Name: "update",
|
|||
|
Usage: "检测程序更新",
|
|||
|
Category: "其他",
|
|||
|
Action: func(c *cli.Context) error {
|
|||
|
if c.IsSet("y") {
|
|||
|
if !c.Bool("y") {
|
|||
|
return nil
|
|||
|
}
|
|||
|
}
|
|||
|
panupdate.CheckUpdate(app.Version, c.Bool("y"))
|
|||
|
return nil
|
|||
|
},
|
|||
|
Flags: []cli.Flag{
|
|||
|
cli.BoolFlag{
|
|||
|
Name: "y",
|
|||
|
Usage: "确认更新",
|
|||
|
},
|
|||
|
},
|
|||
|
},
|
|||
|
|
|||
|
// 退出程序 quit
|
|||
|
{
|
|||
|
Name: "quit",
|
|||
|
Aliases: []string{"exit"},
|
|||
|
Usage: "退出程序",
|
|||
|
Description: "退出程序",
|
|||
|
Category: "其他",
|
|||
|
Action: func(c *cli.Context) error {
|
|||
|
return cli.NewExitError("", 0)
|
|||
|
},
|
|||
|
Hidden: true,
|
|||
|
HideHelp: true,
|
|||
|
},
|
|||
|
|
|||
|
// 调试用 debug
|
|||
|
//{
|
|||
|
// Name: "debug",
|
|||
|
// Aliases: []string{"dg"},
|
|||
|
// Usage: "开发调试用",
|
|||
|
// Description: "",
|
|||
|
// Category: "debug",
|
|||
|
// Before: cmder.ReloadConfigFunc,
|
|||
|
// Action: func(c *cli.Context) error {
|
|||
|
// os.Setenv(config.EnvVerbose, c.String("verbose"))
|
|||
|
// fmt.Println("显示调试日志", logger.IsVerbose)
|
|||
|
// return nil
|
|||
|
// },
|
|||
|
// Flags: []cli.Flag{
|
|||
|
// cli.StringFlag{
|
|||
|
// Name: "param",
|
|||
|
// Usage: "参数",
|
|||
|
// },
|
|||
|
// cli.BoolFlag{
|
|||
|
// Name: "verbose",
|
|||
|
// Destination: &logger.IsVerbose,
|
|||
|
// EnvVar: config.EnvVerbose,
|
|||
|
// Usage: "显示调试信息",
|
|||
|
// },
|
|||
|
// },
|
|||
|
//},
|
|||
|
}
|
|||
|
|
|||
|
sort.Sort(cli.FlagsByName(app.Flags))
|
|||
|
sort.Sort(cli.CommandsByName(app.Commands))
|
|||
|
app.Run(os.Args)
|
|||
|
}
|