Тест Go (Golang) API исходящие запросы без фактического попадания сторонних API

Я работаю над API Go, который переводит между внутренним бэкэндом и несколькими сторонними API. Я пытаюсь понять, как проверить его функциональность, не нажимая на внешний API.

например, вот сервер, который обрабатывает входящие запросы на создание новых песен и отправляет запрос стороннему API:

package main

import (
        "bytes"
        "encoding/json"
        "fmt"
        "net/http"
)

var ThirdPartyApi = "http://www.coolsongssite.api"

type IncomingRequest struct {
        username string         `json:"username"`
        password string         `json:"password"`
        songs    []IncomingSong `json:"songs"`
}

type OutgoingRequest struct {
        username string         `json:"username"`
        password string         `json:"password"`
        songs    []OutgoingSong `json:"songs"`
}

type IncomingSong struct {
        artist string `json:"artist"`
        album  string `json:"album"`
        title  string `json:"title"`
}

type OutgoingSong struct {
        musician string `json:"musician"`
        record   string `json:"record"`
        name     string `json:"name"`
}

func main() {
        http.HandleFunc("/songs/create", createSong)
        http.ListenAndServe(":8080", nil)
}

func createSong(rw http.ResponseWriter, req *http.Request) {
         decoder := json.NewDecoder(req.Body)
         var incomingRequest IncomingRequest
         decoder.Decode(&incomingRequest)
         outgoingRequest := incomingRequestToOutgoingRequest(incomingRequest)

         r, _ := json.Marshal(outgoingRequest)
         request, _ := http.NewRequest("POST", ThirdPartyApi, bytes.NewBuffer(r))
         request.Header.Set("Content-Type", "application/json")

         client := http.Client{}

         response, _ := client.Do(request)
         fmt.Fprintln(rw, response)
}

func incomingRequestToOutgoingRequest(inc IncomingRequest) OutgoingRequest {
        outgoingRequest := OutgoingRequest{
                username: inc.username,
                password: inc.password,
        }

        for _, s := range inc.songs {
                outgoingRequest.songs = append(
                        outgoingRequest.songs, OutgoingSong{
                                musician: s.artist,
                                record:   s.album,
                                name:     s.title,
                        },
                )
        }

        return outgoingRequest
}

так что я мог бы ударить приложение работает на localhost:8080 что-то вроде этого:

curl -X POST http://localhost:8080/songs/new --data \
'{"username": "<my-username>", "password": "<my-password>", "songs": \
["artist": "<song-artist>", "title": "<song-title>", "album": "<song-album>"]}'

мой вопрос is: Как написать тесты, которые проверяют, что запрос, который выходит (в этом случае http://www.coolsongssite.api) правильно, фактически не отправляя его?

должен ли я переписать createSong обработчик, чтобы я мог изолировать то, что происходит в client.Do(request)?

любая помощь/совет/точки в правильном направлении ценятся.

здесь я могу проверить incomingRequestToOutgoingRequest вот так:

package main

import (
        "testing"
)

func TestincomingRequestToOutgoingRequest(t *testing.T) {
        incomingRequest := IncomingRequest{
                username: "myuser",
                password: "mypassword",
        }

        var songs []IncomingSong
        songs = append(
                songs, IncomingSong{
                artist: "White Rabbits",
                album:  "Milk Famous",
                title:  "I'm Not Me",
                },
        )

        outgoingRequest := incomingRequestToOutgoingRequest(incomingRequest)

        if outgoingRequest.songs[0].musician != "White Rabbits" {
                t.Error("Expected musican name to be 'White Rabbits'. Got: ", outgoingRequest.songs[0].musician)
        }
}

1 ответов


можно использовать net/http/httptest.NewServer такой:

ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
    w.Write([]byte(`desired response here`))
}))
defer ts.Close()
ThirdPartyAPI = ts.URL
...
your test here