|  | // Copyright 2017 Google LLC. | 
|  | // Use of this source code is governed by a BSD-style | 
|  | // license that can be found in the LICENSE file. | 
|  |  | 
|  | package main | 
|  |  | 
|  | import ( | 
|  | "bytes" | 
|  | "flag" | 
|  | "fmt" | 
|  | "io/ioutil" | 
|  | "path/filepath" | 
|  | "strings" | 
|  | "testing" | 
|  |  | 
|  | "google.golang.org/api/google-api-go-generator/internal/disco" | 
|  | "google.golang.org/api/internal/version" | 
|  | ) | 
|  |  | 
|  | var updateGolden = flag.Bool("update_golden", false, "If true, causes TestAPIs to update golden files") | 
|  |  | 
|  | func TestAPIs(t *testing.T) { | 
|  | *copyrightYear = "YEAR" | 
|  |  | 
|  | names := []string{ | 
|  | "any", | 
|  | "arrayofarray-1", | 
|  | "arrayofenum", | 
|  | "arrayofmapofobjects", | 
|  | "arrayofmapofstrings", | 
|  | "blogger-3", | 
|  | "floats", | 
|  | "getwithoutbody", | 
|  | "http-body", | 
|  | "json-body", | 
|  | "mapofany", | 
|  | "mapofarrayofobjects", | 
|  | "mapofint64strings", | 
|  | "mapofobjects", | 
|  | "mapofstrings-1", | 
|  | "param-rename", | 
|  | "quotednum", | 
|  | "repeated", | 
|  | "required-query", | 
|  | "resource-named-service", // appengine/v1/appengine-api.json | 
|  | "unfortunatedefaults", | 
|  | "variants", | 
|  | "wrapnewlines", | 
|  | } | 
|  | for _, name := range names { | 
|  | t.Run(name, func(t *testing.T) { | 
|  | api, err := apiFromFile(filepath.Join("testdata", name+".json")) | 
|  | if err != nil { | 
|  | t.Fatalf("Error loading API testdata/%s.json: %v", name, err) | 
|  | } | 
|  | clean, err := api.GenerateCode() | 
|  | if err != nil { | 
|  | t.Fatalf("Error generating code for %s: %v", name, err) | 
|  | } | 
|  | goldenFile := filepath.Join("testdata", name+".want") | 
|  | if *updateGolden { | 
|  | clean := strings.Replace(string(clean), fmt.Sprintf("gdcl/%s", version.Repo), "gdcl/00000000", -1) | 
|  | if err := ioutil.WriteFile(goldenFile, []byte(clean), 0644); err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  | } | 
|  | want, err := ioutil.ReadFile(goldenFile) | 
|  | if err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  | wantStr := strings.Replace(string(want), "gdcl/00000000", fmt.Sprintf("gdcl/%s", version.Repo), -1) | 
|  | if !bytes.Equal([]byte(wantStr), clean) { | 
|  | tf, _ := ioutil.TempFile("", "api-"+name+"-got-json.") | 
|  | if _, err := tf.Write(clean); err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  | if err := tf.Close(); err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  | // NOTE: update golden files with `go test -update_golden` | 
|  | t.Errorf("Output for API %s differs: diff -u %s %s", name, goldenFile, tf.Name()) | 
|  | } | 
|  | }) | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestScope(t *testing.T) { | 
|  | tests := [][]string{ | 
|  | { | 
|  | "https://www.googleapis.com/auth/somescope", | 
|  | "SomescopeScope", | 
|  | }, | 
|  | { | 
|  | "https://mail.google.com/somescope", | 
|  | "MailGoogleComSomescopeScope", | 
|  | }, | 
|  | { | 
|  | "https://mail.google.com/", | 
|  | "MailGoogleComScope", | 
|  | }, | 
|  | } | 
|  | for _, test := range tests { | 
|  | if got := scopeIdentifier(disco.Scope{ID: test[0]}); got != test[1] { | 
|  | t.Errorf("scopeIdentifier(%q) got %q, want %q", test[0], got, test[1]) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestDepunct(t *testing.T) { | 
|  | tests := []struct { | 
|  | needCap  bool | 
|  | in, want string | 
|  | }{ | 
|  | { | 
|  | needCap: true, | 
|  | in:      "part__description", | 
|  | want:    "Part__description", | 
|  | }, | 
|  | { | 
|  | needCap: true, | 
|  | in:      "Part_description", | 
|  | want:    "PartDescription", | 
|  | }, | 
|  | { | 
|  | needCap: false, | 
|  | in:      "part_description", | 
|  | want:    "partDescription", | 
|  | }, | 
|  | { | 
|  | needCap: false, | 
|  | in:      "part-description", | 
|  | want:    "partDescription", | 
|  | }, | 
|  | { | 
|  | needCap: false, | 
|  | in:      "part.description", | 
|  | want:    "partDescription", | 
|  | }, | 
|  | { | 
|  | needCap: false, | 
|  | in:      "part$description", | 
|  | want:    "partDescription", | 
|  | }, | 
|  | { | 
|  | needCap: false, | 
|  | in:      "part/description", | 
|  | want:    "partDescription", | 
|  | }, | 
|  | { | 
|  | needCap: true, | 
|  | in:      "Part__description_name", | 
|  | want:    "Part__descriptionName", | 
|  | }, | 
|  | { | 
|  | needCap: true, | 
|  | in:      "Part_description_name", | 
|  | want:    "PartDescriptionName", | 
|  | }, | 
|  | { | 
|  | needCap: true, | 
|  | in:      "Part__description__name", | 
|  | want:    "Part__description__name", | 
|  | }, | 
|  | { | 
|  | needCap: true, | 
|  | in:      "Part_description__name", | 
|  | want:    "PartDescription__name", | 
|  | }, | 
|  | } | 
|  | for _, test := range tests { | 
|  | if got := depunct(test.in, test.needCap); got != test.want { | 
|  | t.Errorf("depunct(%q,%v) = %q; want %q", test.in, test.needCap, got, test.want) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestRenameVersion(t *testing.T) { | 
|  | tests := []struct { | 
|  | version, want string | 
|  | }{ | 
|  | { | 
|  | version: "directory_v1", | 
|  | want:    "directory/v1", | 
|  | }, | 
|  | { | 
|  | version: "email_migration_v1", | 
|  | want:    "email_migration/v1", | 
|  | }, | 
|  | { | 
|  | version: "my_api_v1.2", | 
|  | want:    "my_api/v1.2", | 
|  | }, | 
|  | } | 
|  | for _, test := range tests { | 
|  | if got := renameVersion(test.version); got != test.want { | 
|  | t.Errorf("renameVersion(%q) = %q; want %q", test.version, got, test.want) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestSupportsPaging(t *testing.T) { | 
|  | api, err := apiFromFile(filepath.Join("testdata", "paging.json")) | 
|  | if err != nil { | 
|  | t.Fatalf("Error loading API testdata/paging.json: %v", err) | 
|  | } | 
|  | api.PopulateSchemas() | 
|  | res := api.doc.Resources[0] | 
|  | for _, meth := range api.resourceMethods(res) { | 
|  | _, _, got := meth.supportsPaging() | 
|  | want := strings.HasPrefix(meth.m.Name, "yes") | 
|  | if got != want { | 
|  | t.Errorf("method %s supports paging: got %t, want %t", meth.m.Name, got, want) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestIsNewerRevision(t *testing.T) { | 
|  | olderBytesPath, newerBytesPath := filepath.Join("testdata", "rev20200415.json"), filepath.Join("testdata", "rev20200416.json") | 
|  | olderBytes, err := ioutil.ReadFile(olderBytesPath) | 
|  | if err != nil { | 
|  | t.Fatalf("ioutil.ReadFile(%q) = %v; want nil", olderBytesPath, err) | 
|  | } | 
|  | newerBytes, err := ioutil.ReadFile(newerBytesPath) | 
|  | if err != nil { | 
|  | t.Fatalf("ioutil.ReadFile(%q) = %v; want nil", newerBytesPath, err) | 
|  | } | 
|  |  | 
|  | // newBytes > oldBytes | 
|  | if err := isNewerRevision(olderBytes, newerBytes); err != nil { | 
|  | t.Fatalf("isNewerRevision(%q, %q) = %v; want nil", string(olderBytes), string(newerBytes), err) | 
|  | } | 
|  | // newBytes == newBytes | 
|  | if err := isNewerRevision(newerBytes, newerBytes); err != nil { | 
|  | t.Fatalf("isNewerRevision(%q, %q) = %v; want nil", string(newerBytes), string(newerBytes), err) | 
|  | } | 
|  | // newBytes < newBytes | 
|  | err = isNewerRevision(newerBytes, olderBytes) | 
|  | if err == nil { | 
|  | t.Fatalf("isNewerRevision(%q, %q) = nil; want %v", string(newerBytes), string(olderBytes), errOldRevision) | 
|  | } | 
|  | if err != errOldRevision { | 
|  | t.Fatalf("got %v, want %v", err, errOldRevision) | 
|  | } | 
|  | } |