Unlocking the Power of HTTP Requests in Go

When it comes to web development, making HTTP requests is a fundamental aspect of interacting with servers and APIs. In Go, also known as Golang, the net/http package provides a robust way to make HTTP requests with ease. In this article, we’ll dive into the world of HTTP requests in Go, exploring how to make GET and POST requests, manage headers and cookies, and even build a real-world cryptocurrency price checker CLI tool.

Making GET Requests

To make a GET request in Go, we can use the Get function from the net/http package. This function takes a URL string as an argument and returns a response and an error. Let’s take a closer look at an example:
go
resp, err := http.Get("https://jsonplaceholder.typicode.com/posts")
if err!= nil {
log.Fatal(err)
}
defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body)
if err!= nil {
log.Fatal(err)
}
fmt.Println(string(body))

In this example, we make a GET request to https://jsonplaceholder.typicode.com/posts and print the response body to the console.

Making POST Requests

To make a POST request in Go, we can use the Post function from the net/http package. This function takes three arguments: the URL string, the content type of the request body, and the request body itself. Let’s take a closer look at an example:
go
postBody, _ := json.Marshal(map[string]string{
"name": "Toby",
"email": "[email protected]",
})
responseBody := bytes.NewBuffer(postBody)
resp, err := http.Post("https://postman-echo.com/post", "application/json", responseBody)
if err!= nil {
log.Fatal(err)
}
defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body)
if err!= nil {
log.Fatal(err)
}
fmt.Println(string(body))

In this example, we make a POST request to https://postman-echo.com/post with a JSON request body containing a name and email.

Managing HTTP Headers and Cookies

HTTP headers and cookies play a crucial role in network requests, providing additional information about the request and response. In Go, we can use the NewRequest function from the net/http package to specify headers and cookies for our requests. Let’s take a closer look at an example:
go
req, err := http.NewRequest("POST", "https://postman-echo.com/post", responseBody)
if err!= nil {
log.Fatal(err)
}
req.Header.Set("X-Auth-Key", "DEMO_KEY_HERE")
resp, err := http.DefaultClient.Do(req)
if err!= nil {
log.Fatal(err)
}
defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body)
if err!= nil {
log.Fatal(err)
}
fmt.Println(string(body))

In this example, we specify an X-Auth-Key header with the value DEMO_KEY_HERE for our POST request.

Building a Cryptocurrency Price Checker CLI Tool

Let’s put our newfound knowledge of HTTP requests in Go to the test by building a cryptocurrency price checker CLI tool. We’ll use the Nomics API to fetch real-time cryptocurrency prices and display them in a user-friendly format.

First, we’ll create a crypto-model file to define a struct for our cryptocurrency data:
go
type CryptoResponse struct {
Price float64 `json:"price"`
Currency string `json:"currency"`
}

Next, we’ll create a crypto-client file to fetch the cryptocurrency data from the Nomics API:
go
func FetchCrypto(crypto string, fiat string) (*CryptoResponse, error) {
url := fmt.Sprintf("https://api.nomics.com/v1/currencies/ticker?key=YOUR_API_KEY&ids=%s&convert=%s", crypto, fiat)
resp, err := http.Get(url)
if err!= nil {
return nil, err
}
defer resp.Body.Close()
var cryptoResponse CryptoResponse
err = json.NewDecoder(resp.Body).Decode(&cryptoResponse)
if err!= nil {
return nil, err
}
return &cryptoResponse, nil
}

Finally, we’ll create a main file to tie everything together:
go
func main() {
fiatCurrency := flag.String("fiat", "USD", "Fiat currency")
cryptoName := flag.String("crypto", "BTC", "Cryptocurrency name")
flag.Parse()
cryptoResponse, err := FetchCrypto(*cryptoName, *fiatCurrency)
if err!= nil {
log.Fatal(err)
}
fmt.Printf("The current price of %s in %s is %.2f\n", *cryptoName, *fiatCurrency, cryptoResponse.Price)
}

With our CLI tool complete, we can run it from the command line and get the current price of our desired cryptocurrency in our preferred fiat currency:

go run main.go -fiat=EUR -crypto=ETH

Concurrency with Go HTTP Requests

Go’s concurrency features make it an ideal choice for applications that require multiple requests to be made simultaneously. We can use goroutines and WaitGroups to launch multiple requests concurrently and wait for them to finish before continuing.

Let’s take a closer look at an example:
“`go
func retrievePost(id int) {
url := fmt.Sprintf(“https://jsonplaceholder.typicode.com/posts/%d”, id)
resp, err := http.Get(url)
if err!= nil {
log.Fatal(err)
}
defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body)
if err!= nil {
log.Fatal(err)
}
fmt.Println(string(body))
}

func main() {
var wg sync.WaitGroup
for i := 1; i <= 10; i++ {
wg.Add(1)
go func(id int) {
retrievePost(id)
wg.Done()
}(i)
}
wg.Wait()
}
“`
In this example, we launch 10 goroutines to retrieve posts from the JSONPlaceholder API concurrently.

Third-Party Request Libraries for Go

While the net/http package provides a robust way to make HTTP requests in Go, there are also third-party libraries that offer additional features and improvements. Let’s take a closer look at three popular libraries: Resty, Sling, and Gentleman.

Resty

Resty is a feature-rich HTTP client library for Go that provides support for JSON and XML marshaling, retry mechanisms, and request redirect policies.
“`go
import “github.com/go-resty/resty/v2”

func main() {
client := resty.New()
resp, err := client.R().Get(“https://jsonplaceholder.typicode.com/posts”)
if err!= nil {
log.Fatal(err)
}
fmt.Println(resp.String())
}
“`
Sling

Sling is a lightweight HTTP client library for Go that provides a syntax with a preference for function chaining.
“`go
import “github.com/dghubble/sling”

func main() {
client := sling.New().Get(“https://jsonplaceholder.typicode.com/posts”)
resp, err := client.Receive()
if err!= nil {
log.Fatal(err)
}
fmt.Println(resp.Body)
}
“`
Gentleman

Gentleman is another highly extensible HTTP client library for Go that provides support for auto retries, file upload, redirects, logging, and debugging.
“`go
import “github.com/hawky4/gentleman”

func main() {
client := gentleman.New()
req := client.Request().Get(“https://jsonplaceholder.typicode.com/posts”)
resp, err := req.Send()
if err!= nil {
log.Fatal(err)
}
fmt.Println(resp.String())
}
“`
With these libraries, we can simplify our HTTP request code and take advantage of additional features and improvements.

Conclusion

In this article, we’ve explored the world of HTTP requests in Go, from making GET and POST requests to managing headers and cookies, and even building a real-world cryptocurrency price checker CLI tool. We’ve also taken a closer look at concurrency with Go HTTP requests and third-party request libraries like Resty, Sling, and Gentleman. Whether you’re building a web scraper, a CLI tool, or a full-fledged web application, Go’s net/http package and these third-party libraries provide a robust way to make HTTP requests with ease.

Leave a Reply