summaryrefslogtreecommitdiffstats
path: root/src/cmd/api/run.go
blob: 1ae629a032230498910ecaa877a59078d482a94b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
// Copyright 2013 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.

//go:build ignore

// The run program is invoked via the dist tool.
// To invoke manually: go tool dist test -run api --no-rebuild
package main

import (
	"errors"
	"fmt"
	"internal/goversion"
	"io/fs"
	"log"
	"os"
	"os/exec"
	"path/filepath"
	"runtime"
	"strconv"
	"strings"
)

func goCmd() string {
	var exeSuffix string
	if runtime.GOOS == "windows" {
		exeSuffix = ".exe"
	}
	path := filepath.Join(runtime.GOROOT(), "bin", "go"+exeSuffix)
	if _, err := os.Stat(path); err == nil {
		return path
	}
	return "go"
}

var goroot string

func main() {
	log.SetFlags(0)
	goroot = os.Getenv("GOROOT") // should be set by run.{bash,bat}
	if goroot == "" {
		log.Fatal("No $GOROOT set.")
	}
	if err := os.Chdir(filepath.Join(goroot, "api")); err != nil {
		log.Fatal(err)
	}

	files, err := filepath.Glob("go1*.txt")
	if err != nil {
		log.Fatal(err)
	}
	next, err := filepath.Glob(filepath.Join("next", "*.txt"))
	if err != nil {
		log.Fatal(err)
	}
	cmd := exec.Command(goCmd(), "tool", "api",
		"-c", strings.Join(files, ","),
		"-approval", strings.Join(append(approvalNeeded(files), next...), ","),
		allowNew(),
		"-next", strings.Join(next, ","),
		"-except", "except.txt",
	)
	out, err := cmd.CombinedOutput()
	if err != nil {
		log.Fatalf("Error running API checker: %v\n%s", err, out)
	}
	fmt.Print(string(out))
}

func approvalNeeded(files []string) []string {
	var out []string
	for _, f := range files {
		name := filepath.Base(f)
		if name == "go1.txt" {
			continue
		}
		minor := strings.TrimSuffix(strings.TrimPrefix(name, "go1."), ".txt")
		n, err := strconv.Atoi(minor)
		if err != nil {
			log.Fatalf("unexpected api file: %v", f)
		}
		if n >= 19 { // approvals started being tracked in Go 1.19
			out = append(out, f)
		}
	}
	return out
}

// allowNew returns the -allow_new flag to use for the 'go tool api' invocation.
func allowNew() string {
	// Experiment for Go 1.19: always require api file updates.
	return "-allow_new=false"

	// Verify that the api/go1.n.txt for previous Go version exists.
	// It definitely should, otherwise it's a signal that the logic below may be outdated.
	if _, err := os.Stat(fmt.Sprintf("go1.%d.txt", goversion.Version-1)); err != nil {
		log.Fatalln("Problem with api file for previous release:", err)
	}

	// See whether the api/go1.n.txt for this Go version has been created.
	// (As of April 2021, it gets created during the release of the first Beta.)
	_, err := os.Stat(fmt.Sprintf("go1.%d.txt", goversion.Version))
	if errors.Is(err, fs.ErrNotExist) {
		// It doesn't exist, so we're in development or before Beta 1.
		// At this stage, unmentioned API additions are deemed okay.
		// (They will be quietly shown in API check output, but the test won't fail).
		return "-allow_new=true"
	} else if err == nil {
		// The api/go1.n.txt for this Go version has been created,
		// so we're definitely past Beta 1 in the release cycle.
		//
		// From this point, enforce that api/go1.n.txt is an accurate and complete
		// representation of what's going into the release by failing API check if
		// there are API additions (a month into the freeze, there shouldn't be many).
		//
		// See golang.org/issue/43956.
		return "-allow_new=false"
	} else {
		log.Fatal(err)
	}
	panic("unreachable")
}