aliyunpan/main.go

557 lines
14 KiB
Go
Raw Normal View History

2021-10-10 10:48:53 +08:00
// 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 版本号
2021-11-13 17:34:44 +08:00
Version = "v0.0.6"
2021-10-10 10:48:53 +08:00
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
2021-10-30 22:53:08 +08:00
cmdRuneFunc = func(r rune) bool {
switch r {
case '\'', '"':
return true
}
return unicode.IsSpace(r)
}
2021-10-10 10:48:53 +08:00
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 {
2021-10-30 22:53:08 +08:00
wd := "/"
if activeUser.IsFileDriveActive() {
wd = activeUser.Workdir
} else if activeUser.IsAlbumDriveActive() {
wd = activeUser.AlbumWorkdir
}
targetDir = path.Join(wd, targetPath)
2021-10-10 10:48:53 +08:00
if !isDir {
targetDir = path.Dir(targetDir)
}
}
2021-10-30 22:53:08 +08:00
files, err := activeUser.CacheFilesDirectoriesList(targetDir)
if err != nil {
return
}
for _, file := range files {
if file == nil {
continue
}
var (
appendLine string
)
2021-10-10 10:48:53 +08:00
2021-10-30 22:53:08 +08:00
// 已经有的情况
if !closed {
if !strings.HasPrefix(file.Path, path.Clean(path.Join(targetDir, path.Base(targetPath)))) {
if path.Base(targetDir) == path.Base(targetPath) {
appendLine = strings.Join(append(lineArgs[:numArgs-1], escaper.EscapeByRuneFunc(path.Join(targetPath, file.FileName), cmdRuneFunc)), " ")
goto handle
}
continue
}
appendLine = strings.Join(append(lineArgs[:numArgs-1], escaper.EscapeByRuneFunc(path.Clean(path.Join(path.Dir(targetPath), file.FileName)), cmdRuneFunc)), " ")
goto handle
}
// 没有的情况
appendLine = strings.Join(append(lineArgs, escaper.EscapeByRuneFunc(file.FileName, cmdRuneFunc)), " ")
goto handle
handle:
if file.IsFolder() {
s = append(s, appendLine+"/")
continue
}
s = append(s, appendLine+" ")
continue
}
2021-10-10 10:48:53 +08:00
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(),
2021-10-27 22:13:29 +08:00
// Token操作
command.CmdToken(),
2021-10-10 10:48:53 +08:00
// 切换工作目录 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(),
// 手动秒传
2021-10-24 23:06:35 +08:00
//command.CmdRapidUpload(),
2021-10-10 10:48:53 +08:00
// 下载文件/目录 download
command.CmdDownload(),
// 导出文件/目录元数据 export
2021-10-24 23:06:35 +08:00
//command.CmdExport(),
2021-10-10 10:48:53 +08:00
// 导入文件 import
2021-10-24 23:06:35 +08:00
//command.CmdImport(),
2021-10-10 10:48:53 +08:00
2022-01-02 18:44:45 +08:00
// webdav服务
command.CmdWebdav(),
2021-10-10 10:48:53 +08:00
// 回收站
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
2021-12-29 15:41:30 +08:00
{
Name: "debug",
Aliases: []string{"dg"},
Usage: "开发调试用",
Description: "",
Category: "debug",
Before: cmder.ReloadConfigFunc,
Action: func(c *cli.Context) error {
2022-01-02 18:44:45 +08:00
os.Setenv(config.EnvVerbose, "1")
logger.IsVerbose = true
2021-12-29 15:41:30 +08:00
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: "显示调试信息",
},
},
},
2021-10-10 10:48:53 +08:00
}
sort.Sort(cli.FlagsByName(app.Flags))
sort.Sort(cli.CommandsByName(app.Commands))
app.Run(os.Args)
}