| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544 |
- package api_test
- import (
- "encoding/json"
- "fmt"
- "net/http"
- "net/http/httptest"
- "net/url"
- "reflect"
- "strings"
- "testing"
- "github.com/porter-dev/porter/internal/helm"
- "helm.sh/helm/v3/pkg/chart"
- "helm.sh/helm/v3/pkg/release"
- "helm.sh/helm/v3/pkg/storage/driver"
- )
- type releaseStub struct {
- name string
- namespace string
- version int
- releaseVersion string
- status release.Status
- }
- // ------------------------- TEST TYPES AND MAIN LOOP ------------------------- //
- type releaseTest struct {
- initializers []func(tester *tester)
- namespace string
- msg string
- method string
- endpoint string
- body string
- expStatus int
- expBody string
- useCookie bool
- validators []func(c *releaseTest, tester *tester, t *testing.T)
- }
- func testReleaseRequests(t *testing.T, tests []*releaseTest, canQuery bool) {
- for _, c := range tests {
- // create a new tester
- tester := newTester(canQuery)
- // if there's an initializer, call it
- for _, init := range c.initializers {
- init(tester)
- }
- tester.app.TestAgents.HelmAgent.ActionConfig.Releases.Driver.(*driver.Memory).SetNamespace(c.namespace)
- req, err := http.NewRequest(
- c.method,
- c.endpoint,
- strings.NewReader(c.body),
- )
- tester.req = req
- if c.useCookie {
- req.AddCookie(tester.cookie)
- }
- if err != nil {
- t.Fatal(err)
- }
- tester.execute()
- rr := tester.rr
- // first, check that the status matches
- if status := rr.Code; status != c.expStatus {
- t.Errorf("%s, handler returned wrong status code: got %v want %v",
- c.msg, status, c.expStatus)
- }
- // if there's a validator, call it
- for _, validate := range c.validators {
- validate(c, tester, t)
- }
- }
- }
- // ------------------------- TEST FIXTURES AND FUNCTIONS ------------------------- //
- var listReleasesTests = []*releaseTest{
- &releaseTest{
- initializers: []func(tester *tester){
- initDefaultReleases,
- },
- msg: "List releases no namespace",
- method: "GET",
- endpoint: "/api/projects/1/releases?" + url.Values{
- "namespace": []string{""},
- "cluster_id": []string{"1"},
- "service_account_id": []string{"1"},
- "storage": []string{"memory"},
- "limit": []string{"20"},
- "skip": []string{"0"},
- "byDate": []string{"false"},
- "statusFilter": []string{"deployed"},
- }.Encode(),
- body: "",
- expStatus: http.StatusOK,
- expBody: releaseStubsToReleaseJSON(sampleReleaseStubs),
- useCookie: true,
- validators: []func(c *releaseTest, tester *tester, t *testing.T){
- releaseReleaseArrBodyValidator,
- },
- },
- &releaseTest{
- initializers: []func(tester *tester){
- initDefaultReleases,
- },
- msg: "List releases with namespace",
- method: "GET",
- namespace: "default",
- endpoint: "/api/projects/1/releases?" + url.Values{
- "namespace": []string{"default"},
- "cluster_id": []string{"1"},
- "service_account_id": []string{"1"},
- "storage": []string{"memory"},
- "limit": []string{"20"},
- "skip": []string{"0"},
- "byDate": []string{"false"},
- "statusFilter": []string{"deployed"},
- }.Encode(),
- body: "",
- expStatus: http.StatusOK,
- expBody: releaseStubsToReleaseJSON([]releaseStub{
- sampleReleaseStubs[0],
- sampleReleaseStubs[2],
- }),
- useCookie: true,
- validators: []func(c *releaseTest, tester *tester, t *testing.T){
- releaseReleaseArrBodyValidator,
- },
- },
- &releaseTest{
- initializers: []func(tester *tester){
- initDefaultReleases,
- },
- msg: "List releases missing required",
- method: "GET",
- namespace: "default",
- endpoint: "/api/projects/1/releases?" + url.Values{
- "service_account_id": []string{"1"},
- "namespace": []string{"default"},
- "storage": []string{"memory"},
- "limit": []string{"20"},
- "skip": []string{"0"},
- "byDate": []string{"false"},
- "statusFilter": []string{"deployed"},
- }.Encode(),
- body: "",
- expStatus: http.StatusUnprocessableEntity,
- expBody: `{"code":601,"errors":["required validation failed"]}`,
- useCookie: true,
- validators: []func(c *releaseTest, tester *tester, t *testing.T){
- releaseBasicBodyValidator,
- },
- },
- }
- func TestHandleListReleases(t *testing.T) {
- testReleaseRequests(t, listReleasesTests, true)
- }
- var getReleaseTests = []*releaseTest{
- &releaseTest{
- initializers: []func(tester *tester){
- initDefaultReleases,
- },
- msg: "Get releases",
- method: "GET",
- namespace: "default",
- endpoint: "/api/projects/1/releases/airwatch/1?" + url.Values{
- "namespace": []string{""},
- "cluster_id": []string{"1"},
- "service_account_id": []string{"1"},
- "storage": []string{"memory"},
- }.Encode(),
- body: "",
- expStatus: http.StatusOK,
- expBody: releaseStubToReleaseJSON(sampleReleaseStubs[0]),
- useCookie: true,
- validators: []func(c *releaseTest, tester *tester, t *testing.T){
- releaseReleaseBodyValidator,
- },
- },
- &releaseTest{
- initializers: []func(tester *tester){
- initDefaultReleases,
- },
- msg: "Release not found",
- method: "GET",
- namespace: "default",
- endpoint: "/api/projects/1/releases/airwatch/5?" + url.Values{
- "namespace": []string{""},
- "cluster_id": []string{"1"},
- "service_account_id": []string{"1"},
- "storage": []string{"memory"},
- }.Encode(),
- body: "",
- expStatus: http.StatusNotFound,
- expBody: `{"code":602,"errors":["release not found"]}`,
- useCookie: true,
- validators: []func(c *releaseTest, tester *tester, t *testing.T){
- releaseBasicBodyValidator,
- },
- },
- }
- func TestHandleGetRelease(t *testing.T) {
- testReleaseRequests(t, getReleaseTests, true)
- }
- var listReleaseHistoryTests = []*releaseTest{
- &releaseTest{
- initializers: []func(tester *tester){
- initHistoryReleases,
- },
- msg: "List release history",
- method: "GET",
- namespace: "default",
- endpoint: "/api/projects/1/releases/wordpress/history?" + url.Values{
- "namespace": []string{""},
- "cluster_id": []string{"1"},
- "service_account_id": []string{"1"},
- "storage": []string{"memory"},
- }.Encode(),
- body: "",
- expStatus: http.StatusOK,
- expBody: releaseStubsToReleaseJSON(historyReleaseStubs),
- useCookie: true,
- validators: []func(c *releaseTest, tester *tester, t *testing.T){
- releaseReleaseArrBodyValidator,
- },
- },
- &releaseTest{
- initializers: []func(tester *tester){
- initDefaultReleases,
- },
- msg: "Release not found",
- method: "GET",
- namespace: "default",
- endpoint: "/api/projects/1/releases/asldfkja/history?" + url.Values{
- "namespace": []string{""},
- "cluster_id": []string{"1"},
- "service_account_id": []string{"1"},
- "storage": []string{"memory"},
- }.Encode(),
- body: "",
- expStatus: http.StatusNotFound,
- expBody: `{"code":602,"errors":["release not found"]}`,
- useCookie: true,
- validators: []func(c *releaseTest, tester *tester, t *testing.T){
- releaseBasicBodyValidator,
- },
- },
- }
- func TestHandleListReleaseHistory(t *testing.T) {
- testReleaseRequests(t, listReleaseHistoryTests, true)
- }
- var upgradeReleaseTests = []*releaseTest{
- &releaseTest{
- initializers: []func(tester *tester){
- initHistoryReleases,
- },
- msg: "Upgrade relase",
- method: "POST",
- namespace: "default",
- endpoint: "/api/projects/1/releases/wordpress/upgrade?" + url.Values{
- "cluster_id": []string{"1"},
- "service_account_id": []string{"1"},
- }.Encode(),
- body: `
- {
- "namespace": "default",
- "storage": "memory",
- "values": "\nfoo: bar\n"
- }
- `,
- expStatus: http.StatusOK,
- expBody: ``,
- useCookie: true,
- validators: []func(c *releaseTest, tester *tester, t *testing.T){
- func(c *releaseTest, tester *tester, t *testing.T) {
- req, err := http.NewRequest(
- "GET",
- "/api/projects/1/releases/wordpress/3?"+url.Values{
- "namespace": []string{"default"},
- "cluster_id": []string{"1"},
- "service_account_id": []string{"1"},
- "storage": []string{"memory"},
- }.Encode(),
- strings.NewReader(""),
- )
- req.AddCookie(tester.cookie)
- if err != nil {
- t.Fatal(err)
- }
- rr2 := httptest.NewRecorder()
- tester.router.ServeHTTP(rr2, req)
- gotBody := &release.Release{}
- expBody := &release.Release{}
- expBodyJSON := releaseStubToReleaseJSON(releaseStub{"wordpress", "default", 3, "1.0.2", release.StatusDeployed})
- json.Unmarshal(rr2.Body.Bytes(), gotBody)
- json.Unmarshal([]byte(expBodyJSON), expBody)
- // just check name and version match, other items will be different
- if gotBody.Name != expBody.Name {
- t.Errorf("%s, validation wrong body: got %v want %v",
- c.msg, gotBody.Name, expBody.Name)
- }
- if gotBody.Version != expBody.Version {
- t.Errorf("%s, validation wrong body: got %v want %v",
- c.msg, gotBody.Version, expBody.Version)
- }
- expConfig := map[string]interface{}{
- "foo": "bar",
- }
- if !reflect.DeepEqual(gotBody.Config, expConfig) {
- t.Errorf("%s, validation wrong config: got %v want %v",
- c.msg, gotBody.Config, expConfig)
- }
- },
- },
- },
- }
- func TestUpgradeRelease(t *testing.T) {
- testReleaseRequests(t, upgradeReleaseTests, true)
- }
- var rollbackReleaseTests = []*releaseTest{
- &releaseTest{
- initializers: []func(tester *tester){
- initHistoryReleases,
- },
- msg: "Rollback relase",
- method: "POST",
- namespace: "default",
- endpoint: "/api/projects/1/releases/wordpress/rollback?" + url.Values{
- "cluster_id": []string{"1"},
- "service_account_id": []string{"1"},
- }.Encode(),
- body: `
- {
- "namespace": "default",
- "storage": "memory",
- "revision": 1
- }
- `,
- expStatus: http.StatusOK,
- expBody: ``,
- useCookie: true,
- validators: []func(c *releaseTest, tester *tester, t *testing.T){
- func(c *releaseTest, tester *tester, t *testing.T) {
- req, err := http.NewRequest(
- "GET",
- "/api/projects/1/releases/wordpress/3?"+url.Values{
- "namespace": []string{"default"},
- "cluster_id": []string{"1"},
- "service_account_id": []string{"1"},
- "storage": []string{"memory"},
- }.Encode(),
- strings.NewReader(""),
- )
- req.AddCookie(tester.cookie)
- if err != nil {
- t.Fatal(err)
- }
- rr2 := httptest.NewRecorder()
- tester.router.ServeHTTP(rr2, req)
- gotBody := &release.Release{}
- expBody := &release.Release{}
- expBodyJSON := releaseStubToReleaseJSON(releaseStub{"wordpress", "default", 3, "1.0.1", release.StatusDeployed})
- fmt.Println(rr2.Body.String())
- json.Unmarshal(rr2.Body.Bytes(), gotBody)
- json.Unmarshal([]byte(expBodyJSON), expBody)
- // just check name and version match, other items will be different
- if gotBody.Name != expBody.Name {
- t.Errorf("%s, validation wrong body: got %v want %v",
- c.msg, gotBody.Name, expBody.Name)
- }
- if gotBody.Version != expBody.Version {
- t.Errorf("%s, validation wrong body: got %v want %v",
- c.msg, gotBody.Version, expBody.Version)
- }
- },
- },
- },
- }
- func TestRollbackRelease(t *testing.T) {
- testReleaseRequests(t, rollbackReleaseTests, true)
- }
- // ------------------------- INITIALIZERS AND VALIDATORS ------------------------- //
- func initDefaultReleases(tester *tester) {
- initUserDefault(tester)
- initProject(tester)
- initProjectSADefault(tester)
- agent := tester.app.TestAgents.HelmAgent
- makeReleases(agent, sampleReleaseStubs)
- // calling agent.ActionConfig.Releases.Create in makeReleases will automatically set the
- // namespace, so we have to reset the namespace of the storage driver
- agent.ActionConfig.Releases.Driver.(*driver.Memory).SetNamespace("")
- }
- func initHistoryReleases(tester *tester) {
- initUserDefault(tester)
- initProject(tester)
- initProjectSADefault(tester)
- agent := tester.app.TestAgents.HelmAgent
- makeReleases(agent, historyReleaseStubs)
- // calling agent.ActionConfig.Releases.Create in makeReleases will automatically set the
- // namespace, so we have to reset the namespace of the storage driver
- agent.ActionConfig.Releases.Driver.(*driver.Memory).SetNamespace("")
- }
- var sampleReleaseStubs = []releaseStub{
- releaseStub{"airwatch", "default", 1, "1.0.0", release.StatusDeployed},
- releaseStub{"not-in-default-namespace", "other", 1, "1.0.1", release.StatusDeployed},
- releaseStub{"wordpress", "default", 1, "1.0.2", release.StatusDeployed},
- }
- var historyReleaseStubs = []releaseStub{
- releaseStub{"wordpress", "default", 1, "1.0.1", release.StatusSuperseded},
- releaseStub{"wordpress", "default", 2, "1.0.2", release.StatusDeployed},
- }
- func releaseStubsToReleaseJSON(rels []releaseStub) string {
- releases := make([]*release.Release, 0)
- for _, r := range rels {
- rel := releaseStubToRelease(r)
- releases = append(releases, rel)
- }
- str, _ := json.Marshal(releases)
- return string(str)
- }
- func releaseStubToReleaseJSON(r releaseStub) string {
- rel := releaseStubToRelease(r)
- str, _ := json.Marshal(rel)
- return string(str)
- }
- func releaseStubToRelease(r releaseStub) *release.Release {
- return &release.Release{
- Name: r.name,
- Namespace: r.namespace,
- Version: r.version,
- Info: &release.Info{
- Status: r.status,
- },
- Chart: &chart.Chart{
- Metadata: &chart.Metadata{
- Version: r.releaseVersion,
- Icon: "https://example.com/icon.png",
- },
- },
- }
- }
- func makeReleases(agent *helm.Agent, rels []releaseStub) {
- storage := agent.ActionConfig.Releases
- for _, r := range rels {
- rel := releaseStubToRelease(r)
- storage.Create(rel)
- }
- }
- func releaseBasicBodyValidator(c *releaseTest, tester *tester, t *testing.T) {
- if body := tester.rr.Body.String(); strings.TrimSpace(body) != strings.TrimSpace(c.expBody) {
- t.Errorf("%s, handler returned wrong body: got %v want %v",
- c.msg, body, c.expBody)
- }
- }
- func releaseReleaseBodyValidator(c *releaseTest, tester *tester, t *testing.T) {
- gotBody := &release.Release{}
- expBody := &release.Release{}
- json.Unmarshal(tester.rr.Body.Bytes(), gotBody)
- json.Unmarshal([]byte(c.expBody), expBody)
- if !reflect.DeepEqual(gotBody, expBody) {
- t.Errorf("%s, handler returned wrong body: got %v want %v",
- c.msg, gotBody, expBody)
- }
- }
- func releaseReleaseArrBodyValidator(c *releaseTest, tester *tester, t *testing.T) {
- gotBody := &[]release.Release{}
- expBody := &[]release.Release{}
- json.Unmarshal(tester.rr.Body.Bytes(), gotBody)
- json.Unmarshal([]byte(c.expBody), expBody)
- if !reflect.DeepEqual(gotBody, expBody) {
- t.Errorf("%s, handler returned wrong body: got %v want %v",
- c.msg, gotBody, expBody)
- }
- }
|