After the upgrade command, I implemented my own upgrade command using Go language.
Get the latest version
Let’s take a look at Deno’s source code first:
use deno_runtime::deno_fetch::reqwest::Client; const RELEASE_URL: &str = "https://github.com/denoland/deno/releases"; ... async fn get_latest_release_version( client: &Client, ) -> Result<String, AnyError> { println!("Looking up latest version"); let res = client .get(&format!("{}/latest", RELEASE_URL)) .send() .await?; let version = res.url().path_segments().unwrap().last().unwrap(); Ok(version.replace("v", "")) }
- Analyze the above code Did the following things: Since it is an asynchronous function, use the println function to prompt that it is looking for a new version. Use the GET method to request https://github.com/denoland/deno/releases/latest
https: //github.com/denoland/deno/releases/latest will redirect to the latest release https://github.com/denoland/deno/releases/tag/v1.7.1, and we will take it out through a string operation v1.7.1
-
Call the replace
method of the string to replacev
with""
to get the version number
Let us use Go to implement it:
The code looks similar, such as template string writing and string processing
package release import ( "net/http" "strings" ) func getLatestVersion() string{ releaseURL := fmt.Sprintf("https://github.com/%s/%s/releases/latest", user, repo) resp, _ := http.Get(releaseURL) defer resp.Body.Close() // 为了防止内存泄漏 pathArr := strings.Split(resp.Request.URL.Path, "/") latest := strings.Replace(pathArr[len(pathArr)-1], "v", "", 1) return latest }
Version detection
SemVer is indispensable for version processing. We use go-version to handle version-related work. With version comparison, we can implement a version checking function. The following code is a piece of code in my encapsulated go-release:
type UpdateInfo struct { IsUpdate bool LatestVersion string } func CheckUpdate(user string, repo string, current string) (updateInfo *UpdateInfo, err error) { releaseURL := fmt.Sprintf("https://github.com/%s/%s/releases/latest", user, repo) resp, err := http.Get(releaseURL) if err != nil { return nil, err } defer resp.Body.Close() current = strings.Replace(current, "v", "", 1) pathArr := strings.Split(resp.Request.URL.Path, "/") latest := strings.Replace(pathArr[len(pathArr)-1], "v", "", 1) currentVersion, err := version.NewVersion(current) if err != nil { return nil, err } latestVersion, err := version.NewVersion(latest) if err != nil { return nil, err } updateInfo = &UpdateInfo{ IsUpdate: currentVersion.LessThan(latestVersion), LatestVersion: latest, } return updateInfo, nil }
cobra implements upgrade
In the following code, we are based on go -release implements a simple CLI with the upgrade
command. The core logic is to compare whether the local version is the latest, and if not, install the latest one.
const Version = "0.0.1" func checkUpgrade(current string) { fmt.Println("Looking up latest version") update, err := release.CheckUpdate("youngjuning", "tpc", current) if err != nil { panic(err) } if update.IsUpdate { fmt.Printf("Found latest version %v \n", update.LatestVersion) sh.Command("bash", "-c", "curl -fsSL https://raw.githubusercontent.com/youngjuning/tpc/main/install.sh | sh").Run() } else { fmt.Printf("Local version %v is the most recent release \n", current) } } var rootCmd = &cobra.Command{ Use: "app", Version: Version, } var cmdUpgrade = &cobra.Command{ Use: "upgrade", Run: func(cmd *cobra.Command, args []string) { checkUpgrade(Version, force) }, } func main() { rootCmd.AddCommand(cmdUpgrade) rootCmd.Execute() }
强制更新方案
Node.js 命令行工具检查更新的正确姿势 对这个问题做了很深入的研究并实现了 Node 版的工具。核心逻辑就是每次执行命令时异步去执行检查更新。这句话翻译成 Go 就是用协程去执行检查更新的动作,这样执行命令还是会立马得到反馈,程序则会在后台执行检查更新,当检测到有新版本则强制更新。
完整代码在go-release/example,对 Cobra 不熟悉的同学可以看一下 Cobra 中文文档。
执行 shell 命令推荐 go-sh,它对 exec 包做了封装,提供了更好地使用体验( PS:还不是因为我菜)
// rootCmd 代表没有调用子命令时的基础命令 var rootCmd = &cobra.Command{ Use: "tpc", Version: Version, Run: func(cmd *cobra.Command, args []string) { sh.Command("tpc", "-h").Run() }, // 每次执行完命令后去执行检查更新,Start 表示不阻塞原来的任务,还有一个 Run 方法则是会阻塞 PersistentPostRun: func(cmd *cobra.Command, args []string) { sh.Command("bash", "-c", "tpc upgrade --force=false").Start() }, }
go-release 的诞生
go-release 核心代码翻译自 Deno 的
upgrade
,开源项目,点赞是我维护的动力,在此求个 Star。
The above is the detailed content of Implement Deno upgrade based on Go. For more information, please follow other related articles on the PHP Chinese website!

Goisidealforbuildingscalablesystemsduetoitssimplicity,efficiency,andbuilt-inconcurrencysupport.1)Go'scleansyntaxandminimalisticdesignenhanceproductivityandreduceerrors.2)Itsgoroutinesandchannelsenableefficientconcurrentprogramming,distributingworkloa

InitfunctionsinGorunautomaticallybeforemain()andareusefulforsettingupenvironmentsandinitializingvariables.Usethemforsimpletasks,avoidsideeffects,andbecautiouswithtestingandloggingtomaintaincodeclarityandtestability.

Goinitializespackagesintheordertheyareimported,thenexecutesinitfunctionswithinapackageintheirdefinitionorder,andfilenamesdeterminetheorderacrossmultiplefiles.Thisprocesscanbeinfluencedbydependenciesbetweenpackages,whichmayleadtocomplexinitializations

CustominterfacesinGoarecrucialforwritingflexible,maintainable,andtestablecode.Theyenabledeveloperstofocusonbehavioroverimplementation,enhancingmodularityandrobustness.Bydefiningmethodsignaturesthattypesmustimplement,interfacesallowforcodereusabilitya

The reason for using interfaces for simulation and testing is that the interface allows the definition of contracts without specifying implementations, making the tests more isolated and easy to maintain. 1) Implicit implementation of the interface makes it simple to create mock objects, which can replace real implementations in testing. 2) Using interfaces can easily replace the real implementation of the service in unit tests, reducing test complexity and time. 3) The flexibility provided by the interface allows for changes in simulated behavior for different test cases. 4) Interfaces help design testable code from the beginning, improving the modularity and maintainability of the code.

In Go, the init function is used for package initialization. 1) The init function is automatically called when package initialization, and is suitable for initializing global variables, setting connections and loading configuration files. 2) There can be multiple init functions that can be executed in file order. 3) When using it, the execution order, test difficulty and performance impact should be considered. 4) It is recommended to reduce side effects, use dependency injection and delay initialization to optimize the use of init functions.

Go'sselectstatementstreamlinesconcurrentprogrammingbymultiplexingoperations.1)Itallowswaitingonmultiplechanneloperations,executingthefirstreadyone.2)Thedefaultcasepreventsdeadlocksbyallowingtheprogramtoproceedifnooperationisready.3)Itcanbeusedforsend

ContextandWaitGroupsarecrucialinGoformanaginggoroutineseffectively.1)ContextallowssignalingcancellationanddeadlinesacrossAPIboundaries,ensuringgoroutinescanbestoppedgracefully.2)WaitGroupssynchronizegoroutines,ensuringallcompletebeforeproceeding,prev


Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

Video Face Swap
Swap faces in any video effortlessly with our completely free AI face swap tool!

Hot Article

Hot Tools

ZendStudio 13.5.1 Mac
Powerful PHP integrated development environment

MantisBT
Mantis is an easy-to-deploy web-based defect tracking tool designed to aid in product defect tracking. It requires PHP, MySQL and a web server. Check out our demo and hosting services.

SublimeText3 Chinese version
Chinese version, very easy to use

EditPlus Chinese cracked version
Small size, syntax highlighting, does not support code prompt function

Atom editor mac version download
The most popular open source editor
