LiHansiyuan 2019-10-27
轉載請註明出處:https://www.cnblogs.com/funnyzpc/p/11721978.html
内容提要:
首先,写这篇博文的一个出发点是:我无法在window的cmd命令行下实现日期的加减(還有其他問題),当然這不是没有实现的方法,而是实现起来很难维护难度较大,光插件都够我折腾了,另外window自带的powershell也可以实现,不过作为一个java渣来说真的难了点儿,因为又要熟悉powershell语法从零开始 后来,我换了个思路,想用代码+第三方开源插件(依赖)实现以上功能;至于,目前我有对Python、java、js、Go、甚至Rust,这些都有或多或少的涉猎,分析了一遍,发现使用半静态或者脚本语言实现很easy,不过有一个问题:你每部署一台机器都要安装语言环境如Python、java,虽然可以跨平台,不过太臃肿了部署一个几兆的应用要安装一个几百兆的语言环境,实在太浪费了内存,js呢又太弱,需要自己造轮子,可以剔除,Rust速度快,不过编写的难度太大,很难考虑, 最后我选用GO作为以上工具的语言,当然这里不得不说一下使用GO的好处:语法简单、跨平台、代码安全、静态打包:这个很重要,可以直接打windows下的可执行程序,也可以打linux可执行程序[交叉编译],这样就可以在部署的时候不用动手又动脚的安装语言环境,配置环境变量之类的乱七八糟的东西,同时安全度也很复合我的需求,例如打成一个可执行包后就自带破解难度,更牛掰的是还可以使用upx对可执行包进行加壳,加壳有三个好处:几乎无法破解、可执行应用体积大大缩小(比我的一个应用打包后有16MB,加壳后只有3MB左右)、易于分发(当然这个是建立在加壳之上在),一切准备就绪,这一篇我就简单的聊一聊我用GO如何实现这类Tools。
由于本章只是個人實現的工具分享,因為這些工具有一定的灵活性,(若需要改造或實現)这时就需要您有一定的GO语言基础。
这里的我主要用到 github.com/pkg/sftp
与 golang.org/x/crypto/ssh
,一个是执行sftp命令,一个是建立ssh连接的,因为sftp是建立在安全的ssh连接上的
样例中有我实现实现linux日志拉取的功能的完整代码,,这里就不展示具体实现代码(参考样例),就简单说说实现步骤吧:
config := &ssh.ClientConfig{...
client, err := ssh.Dial("tcp", "服务器地址:端口", config)
sftp, err := sftp.NewClient(client)
srcFile, err := sftp.Open("/路径/文件01.log." + time + ".zip")
dstFile, err := os.Create(targetPath + "/文件01.log." + time + ".zip")
srcFile.WriteTo(dstFile)
由于在解决实际问题的时候面临的问题比较复杂,光一个压缩包就有zip、tar、gz还有tar+gz的方式,具体使用的时候用到的依赖有archive/tar
、compress/gzip
、archive/zip
、path/filepath
及GO自带的基本依赖等等...
r, err := zip.OpenReader(fullZipFile)
for _, f := range r.File {...
os.MkdirAll(path, f.Mode())
os.MkdirAll(filepath.Dir(path), f.Mode())
f, err := os.OpenFile(path, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, f.Mode())
_, err = io.Copy(f, rc)
fr, err := os.Open(srcFilePath)
gr, err := gzip.NewReader(fr)
tr := tar.NewReader(gr)
for {...
其實官方給的example中並沒有帶秘密的解壓縮,這個問題困擾了我幾個小時,最終在我碰到有網友寫的這個依賴才得以解決:github.com/yeka/zip"
,再次表示十分
感謝,在此能將example共享出來也算是功德一件哈~
這裏的處理其實十分簡單,其實就是在每次循環zip文件的時候判斷一下IsEncrypted()
,在true的時候SetPassword(password)
,後面使用io之後的文件就是
非加密文件了,so easy ~
r, err := zip.OpenReader(fullZipFile)
for _, f := range r.File {...
f.SetPassword(password)
func writeFile(filePath string, f *zip.File) error {...
其實這是對前幾個功能對一個綜合,我對目的是下載阿里雲的RDS的物理備份並解壓,當然你需要先參閱官方api文檔,在這裏DescribeBackups
其實一開始我並不知道,所有的備份文件都需要查詢到其下載連結後才可下載,獲取到下載連結後其他的事情就按部就班了~,以下是我實現的思路,可參閱~
client, err := rds.NewClientWithAccessKey("cn-hangzhou", "您的accessKeyId", "您的accessKeySecret")
request := rds.CreateDescribeBackupsRequest()
respJsonStr, err := client.DescribeBackups(request)
data := &BackupInfo{}
json.Unmarshal(respJsonStr.GetHttpContentBytes(), data)
downloadUrl := data.Items.BackupItem[0].BackupDownloadURL
backupDate := data.Items.BackupItem[0].BackupStartTime[0:10]
err := util.DownloadFile(downloadUrl, filePath)
err2 := util.UnTarGz(filePath, unTarGzFilePath)
一開始我是想將打包好的tools部署後使用命令行調用,這樣會顯得靈活一些,後來覺得這樣使用太過與零碎了,而且window下的執行環境也是個問題,最後才做決定將一組
功能當堵打包,然後使用windows的計畫任務調用,不過既然作為一種可行的方式(linux下比較可行),所以就參閱了個簡單的Demo,讀者可以根據這個Demo改寫上述功能
這裏是結合著命令行實現了個文件下載功能
args\[0\]
是你打包好的exe可執行程序本身(windows環境下)main_exec.exe https://www.xxx.com/path/xx.zip D:/tmp
os.Args
獲取到的參數有仨 os.Args[0]:main_exec.exe
,os.Args[1]:https://www.xxx.com/path/xx.zip
,os.Args[2]:D:/tmp
url := os.Args[1]
filename := os.Args[2]
err := util.DownloadFile(url, filename)
本章的內容比較零散,望讀者諒解,另外,以上內容的所有代碼(包括已經打包好的exe程序)我已推送至github rds_backup
這些代碼全部使用GO語言實現,當然以上內容可能並不完整,全黨是拋磚引玉,一個解決問題的方式,如有需要改進或優化的地方請留言哈~