使用golang开发一个curl命令行工具

 更新时间:2023年11月21日 14:01:49   作者:shura1014  
这篇文章主要为大家详细介绍了如何使用golang开发一个简单的curl命令行工具,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下

本文主要介绍如何使用go编写一个命令行工具,具体一些细节不在本文分析

编写一个简单的curl命令行工具

使用到go的 flag库

例如

第一个参数为命令行参数,第二个默认值GET 第三个usage提示输入信息

flag.String(“X”, “GET”, “GET POST”)

代码如下

package main

import (
	"bufio"
	"bytes"
	"encoding/json"
	"errors"
	"flag"
	"fmt"
	"io"
	"net/http"
	"net/url"
	"time"
)

func main() {

	// 命令行参数 参数名称前面加个 -
    // 第一个参数为命令行参数,第二个默认值GET 第三个usage提示信息 
	method := flag.String("X", "GET", "GET POST")
	// 如果时GET请求不需要写body
	body := flag.String("d", "{}", "json format")
	flag.Parse()

	// 没有 - 的参数,但是需要放到最末尾
	args := flag.Args()
	if len(args) != 1 {
		// 约定只有一个这样子的参数,且必须是url
		fmt.Println("No parameter is passed, the url must be passed at the end")
		return
	}
	// 创建一个httpclient
	client := NewHttpClient()

	switch *method {
	// 只做GET POST支持
	case "POST", "GET":
		// 将body参数转成map 必须要是json格式
		param := make(map[string]any)
		err := json.Unmarshal([]byte(*body), &param)
		if err != nil {
			fmt.Println("Json format is not valid", err.Error())
			return
		}

		data, err := client.Json(*method, args[0], param)
		if err != nil {
			fmt.Println(err.Error())
			return
		}
		// json格式化打印
		result := bytes.Buffer{}
		_ = json.Indent(&result, data, "", "\t")
		fmt.Println(result.String())
	default:
		fmt.Printf("No support %s", *method)
		return
	}
}

// HttpClient http客户端
type HttpClient struct {
	client  http.Client
	Address string
}

func NewHttpClient() *HttpClient {
	client := http.Client{
		Timeout: time.Duration(3) * time.Second,
		Transport: &http.Transport{
			MaxIdleConnsPerHost:   5,
			MaxConnsPerHost:       100,
			IdleConnTimeout:       90 * time.Second,
			TLSHandshakeTimeout:   10 * time.Second,
			ExpectContinueTimeout: 1 * time.Second,
		},
	}
	return &HttpClient{client: client}
}

// Json 接收json格式返回
func (c *HttpClient) Json(method string, url string, args map[string]any) ([]byte, error) {
	jsonStr, _ := json.Marshal(args)
	req, err := http.NewRequest(method, url, bytes.NewReader(jsonStr))
	req.Header.Set("Content-Type", "application/json")
	if err != nil {
		return nil, err
	}
	return c.handleResponse(req)
}

func (c *HttpClient) handleResponse(req *http.Request) ([]byte, error) {
	// go原生方法 执行http请求
	response, err := c.client.Do(req)
	if err != nil {
		return nil, err
	}

	if response.StatusCode != http.StatusOK {
		return nil, errors.New(fmt.Sprintf("response code is %d", response.StatusCode))
	}

	// 以带缓冲字节的方式读取body 与其它语言类似
	buff := make([]byte, 128)
	var body []byte
	reader := bufio.NewReader(response.Body)
	for {
		n, err := reader.Read(buff)

		if err != nil && err != io.EOF {
			return nil, err
		}

		body = append(body, buff[:n]...)

		// 读完了
		if err == io.EOF || n == 0 {
			break
		}

		if n < 128 {
			break
		}

	}
	defer response.Body.Close()
	return body, nil
}

// 将map转成go中的url.Values
func (c *HttpClient) toValues(args map[string]any) string {
	if args != nil && len(args) > 0 {
		params := url.Values{}
		for k, v := range args {
			params.Set(k, fmt.Sprintf("%v", v))
		}
		return params.Encode()
	}
	return ""
}

打开终端命令行执行 go build -o gocurl.exe .

会在当前项目生成 gocurl.exe的windows可执行文件,如果时其它操作系统可以起其它后缀

D:\GoProject\curl> go build  -o gocurl.exe  .                                         

帮助信息 --help 或者 -h

准备两个服务

一个GET请求一个POST,这里需要大家自己准备,我这里准备的两个服务postman调用结果如下

测试

使用终端执行gocurl.exe

PS D:\GoProject\curl> .\gocurl.exe -X=GET http://localhost:7001/product/get?pid=1001
{
        "code": "200",
        "msg": "ok",
        "data": {
                "pid": "1001",
                "pName": "iphone14-7676",
                "count": 0,
                "PType": ""
        }
}
PS D:\GoProject\curl> .\gocurl.exe -X=POST -d='{\"pid\":\"1002\"}' http://localhost:7001/product/post
{
        "code": "200",
        "msg": "ok",
        "data": {
                "pid": "1002",
                "pName": "iphone14-7676",
                "count": 0,
                "PType": ""
        }
}

好了,一个简单的命令行工具已经开法完成。大家也可以尝试着去开发自己的小工具。编译成对应系统的可执行文件。比如开发一个并发测试工具 ab可以并发请求url并且计算qps等信息。

到此这篇关于使用golang开发一个curl命令行工具的文章就介绍到这了,更多相关go命令行工具内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • golang gin 监听rabbitmq队列无限消费的案例代码

    golang gin 监听rabbitmq队列无限消费的案例代码

    这篇文章主要介绍了golang gin 监听rabbitmq队列无限消费,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-12-12
  • Go语言中XML文件的读写操作

    Go语言中XML文件的读写操作

    本文主要介绍了Go语言中XML文件的读写操作,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-04-04
  • gin正确多次读取http request body内容实现详解

    gin正确多次读取http request body内容实现详解

    这篇文章主要为大家介绍了gin正确多次读取http request body内容实现详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-01-01
  • Go语言并发模型的2种编程方案

    Go语言并发模型的2种编程方案

    这篇文章主要介绍了Go语言并发模型的2种编程方案,本文给出共享内存和通过通信的2种解决方案,并给出了实现代码,需要的朋友可以参考下
    2014-10-10
  • 一文带你深入了解Golang中的自旋锁

    一文带你深入了解Golang中的自旋锁

    自旋锁是一种忙等待锁,当一个线程尝试获取一个已经被其它线程持有的锁时,这个线程会持续循环检查锁的状态(即“自旋”) ,直到锁被释放后获得所有权,下面我们就来深入了解下自旋锁的具体操作吧
    2024-01-01
  • Go语言中定时任务库Cron使用方法介绍

    Go语言中定时任务库Cron使用方法介绍

    cron的意思计划任务,说白了就是定时任务。我和系统约个时间,你在几点几分几秒或者每隔几分钟跑一个任务(job),今天通过本文给大家介绍下Go语言中定时任务库Cron使用方法,感兴趣的朋友一起看看吧
    2022-03-03
  • Go语言优雅实现单例模式的多种方式

    Go语言优雅实现单例模式的多种方式

    单例模式(Singleton Pattern)是一种设计模式,旨在保证一个类只有一个实例,并且提供全局访问点,单例模式通常用于需要限制某个对象的实例数量为一个的场景,本文给大家介绍了Go语言实现单例模式的多种方式,需要的朋友可以参考下
    2025-02-02
  • golang中之strconv包的具体使用方法

    golang中之strconv包的具体使用方法

    这篇文章主要介绍了golang中之strconv包的具体使用方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-10-10
  • 关于golang test缓存问题

    关于golang test缓存问题

    这篇文章主要介绍了关于golang test缓存问题,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-09-09
  • Golang中接收者方法语法糖的使用方法详解

    Golang中接收者方法语法糖的使用方法详解

    这篇文章主要为大家详细介绍了Golang中接收者方法语法糖的使用方法,文中的示例代码讲解详细,具有一定的学习价值,感兴趣的可以了解一下
    2023-05-05

最新评论