Logo

dev-resources.site

for different kinds of informations.

Build your own curl in Golang

Published at
3/27/2024
Categories
curl
go
Author
ericbsantana
Categories
2 categories in total
curl
open
go
open
Author
12 person written this
ericbsantana
open
Build your own curl in Golang

In this tutorial, we'll walk through the process of creating a simple command-line tool similar to curl using Go and Cobra, a CLI library for Go.

Disclaimer: Here we are going to use the net package instead of the http that is available natively in Go. The reason for using net is to get a bit into the basics of creating a HTTP request to a server from scrach. You could easily use http package to enjoy all that stuff that is a pain to make from scratch. For instance, http already handles HTTPS requests.

It is a challenge because handling TCP connections and HTTP requests can be complex, but we'll keep it simple and focus on the basics. But will be a good start to understand how curl works under the hood and how to build a simple HTTP client using Go.

Prerequisites

Before we begin, make sure you have the following installed:

  • Go (version 1.22)

Setting Up Your Project

First, let's create a new Go module for our project:

mkdir build-your-own-curl
cd build-your-own-curl
go mod init build-your-own-curl
Enter fullscreen mode Exit fullscreen mode

Next, let's install Cobra:

go get -u github.com/spf13/cobra/cobra
Enter fullscreen mode Exit fullscreen mode

Now, let's initialize Cobra in our project:

cobra-cli init
Enter fullscreen mode Exit fullscreen mode

This will create the structure for our CLI application with the following files:

├── cmd/
│   ├── root.go
├── go.mod
├── go.sum
├── main.go
├── LICENSE
Enter fullscreen mode Exit fullscreen mode

The cmd directory contains the root command file, which is where we'll define our whole application. It is also possible to create subcommands in separate files within the cmd directory. For now, we'll keep it simple and define everything in the root.go file.

Building the Root Command

Now that we have our project set up, let's define our CLI commands using Cobra. We'll create a simple command to make HTTP GET requests.

To achieve this, we need to parse the URL provided as an argument, extract the hostname, port, and path, and then make an HTTP GET request to the specified URL. Firstly, we should parse the URL and extract the necessary information to create our TCP connection.

// cmd/root.go
package cmd

import (
    "net/url"
    "os"

    "github.com/spf13/cobra"
)

var rootCmd = &cobra.Command{
    Use:   "build-your-own-curl",
    Short: "A brief description of your application",
    Long: `A longer description that spans multiple lines and likely contains
examples and usage of using your application. For example:
Cobra is a CLI library for Go that empowers applications.
This application is a tool to generate the needed files
to quickly create a Cobra application.`,
    Args: cobra.ExactArgs(1),

    Run: func(cmd *cobra.Command, args []string) {
        u, err := url.Parse(args[0])

        if err != nil {
            panic(err)
        }

        host := u.Hostname()
        port := u.Port()
        path := u.Path

        println("Host:", host)
        println("Port:", port)
        println("Path:", path)
    },
}

// rest of the code
Enter fullscreen mode Exit fullscreen mode

The Args field specifies the number of arguments the command expects. In this case, we expect exactly one argument, which is the URL we want to make a request to. You may want to add custom validators or other Cobra built-in validators if you want to expand this functionality. If you run the application without an argument, you should see an error message printed to the console.

go run main.go

Error: accepts 1 arg(s), received 0
Usage:
  build-your-own-curl [flags]

Flags:
  -h, --help     help for build-your-own-curl
  -t, --toggle   Help message for toggle

exit status 1
Enter fullscreen mode Exit fullscreen mode

But if you run the application with an URL as an argument, you should see that hostname, port, and path are printed to the console.

go run main.go https://example.com/get

Host: example.com
Port:
Path: /get
Enter fullscreen mode Exit fullscreen mode

As you probably have noticed, the port is not being extracted correctly. This is because the url.Parse function does not return the port if it is not specified in the URL.

Most of us do not specify the port when making a day-to-day HTTP request using curl or a browser. To make our UX better, let's set the default port to 80 if it is not specified in the URL. In this tutorial, I will not handle HTTPS requests, which is why we are going to use only port HTTP (80) for now.

// cmd/root.go
// ...
  Run: func(cmd *cobra.Command, args []string) {
    u, err := url.Parse(args[0])

    if err != nil {
      panic(err)
    }

    host := u.Hostname()
    port := u.Port()

    if port == "" {
      port = "80"
    }

    path := u.Path

    println("Host:", host)
    println("Port:", port)
    println("Path:", path)
  },
}
// ...
Enter fullscreen mode Exit fullscreen mode

Run the application to see the default port bring printed to the console.

go run main.go https://example.com/get

Host: example.com
Port: 80
Path: /get
Enter fullscreen mode Exit fullscreen mode

Now that we have the necessary information to create a TCP connection, let's make an HTTP GET request to the specified URL.

A basic HTTP GET request header consists of the following:

  • Request line: GET /path HTTP/1.0
  • Host header: Host: hostname

We will send this request to the server using net.Dial and read the response. We will then print the response to the console.

// cmd/root.go
package cmd

import (
    "fmt"
    "net"
    "net/url"
    "os"

    "github.com/spf13/cobra"
)

var rootCmd = &cobra.Command{
    Use:   "build-your-own-curl",
    Short: "A brief description of your application",
    Long: `A longer description that spans multiple lines and likely contains
examples and usage of using your application. For example:
Cobra is a CLI library for Go that empowers applications.
This application is a tool to generate the needed files
to quickly create a Cobra application.`,
    Args: cobra.ExactArgs(1),

    Run: func(cmd *cobra.Command, args []string) {
        u, err := url.Parse(args[0])

        if err != nil {
            panic(err)
        }

        host := u.Hostname()
        port := u.Port()
        path := u.Path

        if port == "" {
            port = "80"
        }

        conn, err := net.Dial("tcp", fmt.Sprintf("%s:%s", host, port))

        if err != nil {
            panic(err)
        }

        defer conn.Close()

        fmt.Fprintf(conn, "GET %s HTTP/1.0\r\nHost: %s\r\n\r\n", path, host)

        buf := make([]byte, 1024)
        n, err := conn.Read(buf)

        if err != nil {
            panic(err)
        }

        fmt.Println(string(buf[:n]))
    },
}

// ...
Enter fullscreen mode Exit fullscreen mode

In the code above, we create a TCP connection to the specified host and port. We then send an HTTP GET request to the server and read the response. Finally, we print the response to the console. If you run the application with a valid URL, you should see the HTTP response printed to the console.

go run main.go http://eu.httpbin.org/get
HTTP/1.1 200 OK
Date: Wed, 27 Mar 2024 22:35:13 GMT
Content-Type: application/json
Content-Length: 203
Connection: close
Server: gunicorn/19.9.0
Access-Control-Allow-Origin: *
Access-Control-Allow-Credentials: true

{
  "args": {},
  "headers": {
    "Host": "eu.httpbin.org",
    "X-Amzn-Trace-Id": "Root=1-66049f21-305d0735393fd4ae2bc554a0"
  },
  "url": "http://eu.httpbin.org/get"
}
Enter fullscreen mode Exit fullscreen mode

And you have it! A command-line tool to make GET requests to any URL using Go and Cobra. Additional changes can be made to handle different HTTP methods, headers, and more.

A challenge for you:

  • Add support for different HTTP methods (e.g., POST, PUT, DELETE).
  • Add support for custom headers.
  • Add support for HTTPS requests.

The first and second were made in my project gurl. You can check it out for more inspiration or even to contribute and improve it!

Another project that worth to mention here is go-curling from @chriswiegand

It uses http package instead of net which shows a more simple and fast way to implement a cURL client in Go without making requests from scratch using net.Dial.

Conclusion

Congratulations! You've built a simple command-line tool similar to curl using Go and Cobra. Feel free to expand upon this project by adding more features like handling different HTTP methods, headers, and more.

I have made a project called gurl that is a simple CLI tool that can make HTTP requests using Go. You can check it out for more inspiration.

curl Article's
30 articles in total
Favicon
How to Ignore cURL SSL Errors
Favicon
How to Use cURL to Download Files?
Favicon
Unlocking the Power of cURL Set Headers for Web Development
Favicon
The Essential Guide to cURL Set Headers for Developers
Favicon
What is HTTP 405 Error? (Method Not Allowed)
Favicon
Boost Your Network Control with Curl SOCKS5 Proxies
Favicon
How to Use cURL GET Requests
Favicon
Unlock Efficient IP Management with Curl Proxy
Favicon
How Does Curl Work and Enhance File Transfers Across Platforms
Favicon
How Does Curl Work to Simplify Data Transfers and Testing
Favicon
cURL vs Wget: Key Differences Explained
Favicon
How to Make DELETE Requests Using the curl_init() Function in PHP
Favicon
Using htmlq to filter web data
Favicon
Harder HTB: Using only the terminal
Favicon
Vault CLI in Containers
Favicon
How to Route cURL Requests Through a Proxy Server
Favicon
TLS Fingerprint äŋč­·įš„įķēįŦ™
Favicon
Manage Telegram Webhooks Using curl
Favicon
Introducing CurlDock: Simplify API Testing with Docker and Curl
Favicon
CURL - All methods and Usage ✅
Favicon
uploading to s3 with bash
Favicon
Curl on FTP
Favicon
Build your own curl in Golang
Favicon
Build Your Own curl - Rust
Favicon
Download file using curl
Favicon
How to Use cURL For Web Scraping
Favicon
Maintain a Healthy Sense of Caution Whenever Running a `curl|bash` Command
Favicon
How to Use a Proxy in PHP with cURL
Favicon
Curl: Redirect output to file
Favicon
How to use cURL in PHP

Featured ones: