123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281 |
- package main
- import (
- "bytes"
- "context"
- "os"
- osexec "os/exec"
- "path/filepath"
- "slices"
- "strings"
- "testing"
- "github.com/google/go-cmp/cmp"
- "github.com/google/go-cmp/cmp/cmpopts"
- "github.com/sqlc-dev/sqlc/internal/cmd"
- "github.com/sqlc-dev/sqlc/internal/config"
- "github.com/sqlc-dev/sqlc/internal/opts"
- )
- func TestExamples(t *testing.T) {
- t.Parallel()
- ctx := context.Background()
- examples, err := filepath.Abs(filepath.Join("..", "..", "examples"))
- if err != nil {
- t.Fatal(err)
- }
- files, err := os.ReadDir(examples)
- if err != nil {
- t.Fatal(err)
- }
- for _, replay := range files {
- if !replay.IsDir() {
- continue
- }
- tc := replay.Name()
- t.Run(tc, func(t *testing.T) {
- t.Parallel()
- path := filepath.Join(examples, tc)
- var stderr bytes.Buffer
- opts := &cmd.Options{
- Env: cmd.Env{},
- Stderr: &stderr,
- }
- output, err := cmd.Generate(ctx, path, "", opts)
- if err != nil {
- t.Fatalf("sqlc generate failed: %s", stderr.String())
- }
- cmpDirectory(t, path, output)
- })
- }
- }
- func BenchmarkExamples(b *testing.B) {
- ctx := context.Background()
- examples, err := filepath.Abs(filepath.Join("..", "..", "examples"))
- if err != nil {
- b.Fatal(err)
- }
- files, err := os.ReadDir(examples)
- if err != nil {
- b.Fatal(err)
- }
- for _, replay := range files {
- if !replay.IsDir() {
- continue
- }
- tc := replay.Name()
- b.Run(tc, func(b *testing.B) {
- path := filepath.Join(examples, tc)
- for i := 0; i < b.N; i++ {
- var stderr bytes.Buffer
- opts := &cmd.Options{
- Env: cmd.Env{},
- Stderr: &stderr,
- }
- cmd.Generate(ctx, path, "", opts)
- }
- })
- }
- }
- type textContext struct {
- Mutate func(*config.Config)
- Enabled func() bool
- }
- func TestReplay(t *testing.T) {
- // Ensure that this environment variable is always set to true when running
- // end-to-end tests
- os.Setenv("SQLC_DUMMY_VALUE", "true")
- // t.Parallel()
- ctx := context.Background()
- contexts := map[string]textContext{
- "base": {
- Mutate: func(c *config.Config) {},
- Enabled: func() bool { return true },
- },
- "managed-db": {
- Mutate: func(c *config.Config) {
- c.Cloud.Project = "01HAQMMECEYQYKFJN8MP16QC41" // TODO: Read from environment
- for i := range c.SQL {
- c.SQL[i].Database = &config.Database{
- Managed: true,
- }
- }
- },
- Enabled: func() bool {
- return len(os.Getenv("SQLC_AUTH_TOKEN")) > 0
- },
- },
- }
- for name, testctx := range contexts {
- name := name
- testctx := testctx
- if !testctx.Enabled() {
- continue
- }
- for _, replay := range FindTests(t, "testdata", name) {
- tc := replay
- t.Run(filepath.Join(name, tc.Name), func(t *testing.T) {
- t.Parallel()
- var stderr bytes.Buffer
- var output map[string]string
- var err error
- path, _ := filepath.Abs(tc.Path)
- args := tc.Exec
- if args == nil {
- args = &Exec{Command: "generate"}
- }
- expected := string(tc.Stderr)
- if args.Process != "" {
- _, err := osexec.LookPath(args.Process)
- if err != nil {
- t.Skipf("executable not found: %s %s", args.Process, err)
- }
- }
- if len(args.Contexts) > 0 {
- if !slices.Contains(args.Contexts, name) {
- t.Skipf("unsupported context: %s", name)
- }
- }
- opts := cmd.Options{
- Env: cmd.Env{
- Debug: opts.DebugFromString(args.Env["SQLCDEBUG"]),
- NoRemote: true,
- },
- Stderr: &stderr,
- MutateConfig: testctx.Mutate,
- }
- switch args.Command {
- case "diff":
- err = cmd.Diff(ctx, path, "", &opts)
- case "generate":
- output, err = cmd.Generate(ctx, path, "", &opts)
- if err == nil {
- cmpDirectory(t, path, output)
- }
- case "vet":
- err = cmd.Vet(ctx, path, "", &opts)
- default:
- t.Fatalf("unknown command")
- }
- if len(expected) == 0 && err != nil {
- t.Fatalf("sqlc %s failed: %s", args.Command, stderr.String())
- }
- diff := cmp.Diff(strings.TrimSpace(expected), strings.TrimSpace(stderr.String()))
- if diff != "" {
- t.Fatalf("stderr differed (-want +got):\n%s", diff)
- }
- })
- }
- }
- }
- func cmpDirectory(t *testing.T, dir string, actual map[string]string) {
- expected := map[string]string{}
- var ff = func(path string, file os.FileInfo, err error) error {
- if err != nil {
- return err
- }
- if file.IsDir() {
- return nil
- }
- if !strings.HasSuffix(path, ".go") && !strings.HasSuffix(path, ".kt") && !strings.HasSuffix(path, ".py") && !strings.HasSuffix(path, ".json") && !strings.HasSuffix(path, ".txt") {
- return nil
- }
- // TODO: Figure out a better way to ignore certain files
- if strings.HasSuffix(path, ".txt") && filepath.Base(path) != "hello.txt" {
- return nil
- }
- if filepath.Base(path) == "sqlc.json" {
- return nil
- }
- if filepath.Base(path) == "exec.json" {
- return nil
- }
- if strings.Contains(path, "/kotlin/build") {
- return nil
- }
- if strings.HasSuffix(path, "_test.go") || strings.Contains(path, "src/test/") {
- return nil
- }
- if strings.Contains(path, "/python/.venv") || strings.Contains(path, "/python/src/tests/") ||
- strings.HasSuffix(path, "__init__.py") || strings.Contains(path, "/python/src/dbtest/") ||
- strings.Contains(path, "/python/.mypy_cache") {
- return nil
- }
- blob, err := os.ReadFile(path)
- if err != nil {
- return err
- }
- expected[path] = string(blob)
- return nil
- }
- if err := filepath.Walk(dir, ff); err != nil {
- t.Fatal(err)
- }
- if !cmp.Equal(expected, actual, cmpopts.EquateEmpty()) {
- t.Errorf("%s contents differ", dir)
- for name, contents := range expected {
- name := name
- if actual[name] == "" {
- t.Errorf("%s is empty", name)
- return
- }
- if diff := cmp.Diff(contents, actual[name]); diff != "" {
- t.Errorf("%s differed (-want +got):\n%s", name, diff)
- }
- }
- }
- }
- func BenchmarkReplay(b *testing.B) {
- ctx := context.Background()
- var dirs []string
- err := filepath.Walk("testdata", func(path string, info os.FileInfo, err error) error {
- if err != nil {
- return err
- }
- if info.Name() == "sqlc.json" || info.Name() == "sqlc.yaml" || info.Name() == "sqlc.yml" {
- dirs = append(dirs, filepath.Dir(path))
- return filepath.SkipDir
- }
- return nil
- })
- if err != nil {
- b.Fatal(err)
- }
- for _, replay := range dirs {
- tc := replay
- b.Run(tc, func(b *testing.B) {
- path, _ := filepath.Abs(tc)
- for i := 0; i < b.N; i++ {
- var stderr bytes.Buffer
- opts := &cmd.Options{
- Env: cmd.Env{},
- Stderr: &stderr,
- }
- cmd.Generate(ctx, path, "", opts)
- }
- })
- }
- }
|