eng/tools/generator/cmd/v2/release/releaseCmd.go (346 lines of code) (raw):

// Copyright (c) Microsoft Corporation. All rights reserved. // Licensed under the MIT License. See License.txt in the project root for license information. package release import ( "errors" "fmt" "log" "os" "path/filepath" "strings" "time" "github.com/Azure/azure-sdk-for-go/eng/tools/generator/cmd/issue/link" "github.com/Azure/azure-sdk-for-go/eng/tools/generator/cmd/v2/common" "github.com/Azure/azure-sdk-for-go/eng/tools/generator/config" "github.com/Azure/azure-sdk-for-go/eng/tools/generator/flags" "github.com/Azure/azure-sdk-for-go/eng/tools/generator/repo" "github.com/Azure/azure-sdk-for-go/eng/tools/generator/typespec" "github.com/go-git/go-git/v5/plumbing" "github.com/spf13/cobra" "github.com/spf13/pflag" ) var ( releaseBranchNamePattern = "release-%s-%s-%s-%v" confirmComment = "Release [PR](%s) is ready. Please check whether the package works well ([doc for testing with the generated Go SDK](https://github.com/Azure/azure-rest-api-specs/blob/main/documentation/code-gen/configure-go-sdk.md#test-with-the-generated-go-sdk)). If you are not a Golang user, you can mainly check whether the changelog meets your requirements." ) // Release command func Command() *cobra.Command { releaseCmd := &cobra.Command{ Use: "release-v2 <azure-sdk-for-go directory/commitid> <azure-rest-api-specs directory/commitid> <rp-name/release-request-json-config> [namespaceName]", Short: "Generate a v2 release of azure-sdk-for-go", Long: `This command will generate a new v2 release for azure-sdk-for-go with given rp name and namespace name. azure-sdk-for-go directory/commitid: the directory path of the azure-sdk-for-go with git control or just a commitid for remote repo azure-rest-api-specs directory: the directory path of the azure-rest-api-specs with git control or just a commitid for remote repo rp-name/release-request-json-config: name of resource provider to be released (same for the swagger folder name) or release request json config file path generated by 'generator issue' namespaceName: name of namespace to be released, default value is arm+rp-name `, Args: cobra.RangeArgs(3, 4), RunE: func(cmd *cobra.Command, args []string) error { rpName := args[2] namespaceName := "arm" + rpName if len(args) == 4 { namespaceName = args[3] } ctx := commandContext{ rpName: rpName, namespaceName: namespaceName, flags: ParseFlags(cmd.Flags()), } return ctx.execute(args[0], args[1]) }, } BindFlags(releaseCmd.Flags()) return releaseCmd } type Flags struct { VersionNumber string SwaggerRepo string PackageTitle string SDKRepo string SpecRPName string ReleaseDate string SkipCreateBranch bool SkipGenerateExample bool PackageConfig string GoVersion string Token string UpdateSpecVersion bool ForceStableVersion bool TypeSpecConfig string TypeSpecGoOption string TspClientOption []string } func BindFlags(flagSet *pflag.FlagSet) { flagSet.String("version-number", "", "Specify the version number of this release") flagSet.String("package-title", "", "Specifies the title of this package") flagSet.String("sdk-repo", "https://github.com/Azure/azure-sdk-for-go", "Specifies the sdk repo URL for generation") flagSet.String("spec-repo", "https://github.com/Azure/azure-rest-api-specs", "Specifies the swagger repo URL for generation") flagSet.String("spec-rp-name", "", "Specifies the swagger spec RP name, default is RP name") flagSet.String("release-date", "", "Specifies the release date in changelog") flagSet.Bool("skip-create-branch", false, "Skip create release branch after generation") flagSet.Bool("skip-generate-example", false, "Skip generate example for SDK in the same time") flagSet.String("package-config", "", "Additional config for package") flagSet.String("go-version", "1.18", "Go version") flagSet.StringP("token", "t", "", "Specify the personal access token of Github") flagSet.Bool("update-spec-version", true, "Whether to update the commit id, the default is true") flagSet.Bool("force-stable-version", false, "Even if input-files contains preview files, they are forced to be generated as stable versions. At the same time, the tag must not contain preview.") flagSet.String("tsp-config", "", "The path of the typespec tspconfig.yaml") flagSet.String("tsp-option", "", "Emit typespec-go options, only valid when tsp-config is configured. e: option1=value1;option2=value2") flagSet.StringSlice("tsp-client-option", nil, "The tsp-client(@azure-tools/typespec-client-generator-cli) init options. e: --save-inputs,--debug") } func ParseFlags(flagSet *pflag.FlagSet) Flags { return Flags{ VersionNumber: flags.GetString(flagSet, "version-number"), PackageTitle: flags.GetString(flagSet, "package-title"), SDKRepo: flags.GetString(flagSet, "sdk-repo"), SwaggerRepo: flags.GetString(flagSet, "spec-repo"), SpecRPName: flags.GetString(flagSet, "spec-rp-name"), ReleaseDate: flags.GetString(flagSet, "release-date"), SkipCreateBranch: flags.GetBool(flagSet, "skip-create-branch"), SkipGenerateExample: flags.GetBool(flagSet, "skip-generate-example"), PackageConfig: flags.GetString(flagSet, "package-config"), GoVersion: flags.GetString(flagSet, "go-version"), Token: flags.GetString(flagSet, "token"), UpdateSpecVersion: flags.GetBool(flagSet, "update-spec-version"), ForceStableVersion: flags.GetBool(flagSet, "force-stable-version"), TypeSpecConfig: flags.GetString(flagSet, "tsp-config"), TypeSpecGoOption: flags.GetString(flagSet, "tsp-option"), TspClientOption: flags.GetStringSlice(flagSet, "tsp-client-option"), } } type commandContext struct { rpName string namespaceName string flags Flags pullRequestLabels string } func (c *commandContext) execute(sdkRepoParam, specRepoParam string) error { var err error if _, err := os.Stat(sdkRepoParam); os.IsNotExist(err) { return err } if _, err := os.Stat(specRepoParam); os.IsNotExist(err) { return err } sdkRepo, err := common.GetSDKRepo(sdkRepoParam, c.flags.SDKRepo) if err != nil { return err } specCommitHash, err := common.GetSpecCommit(specRepoParam) if err != nil { return err } if filepath.Ext(c.rpName) == ".json" { return c.generateFromRequest(sdkRepo, specRepoParam, specCommitHash) } else { return c.generate(sdkRepo, specCommitHash) } } func (c *commandContext) generate(sdkRepo repo.SDKRepository, specCommitHash string) error { log.Printf("Release generation for rp: %s, namespace: %s", c.rpName, c.namespaceName) generateCtx := common.GenerateContext{ SDKPath: sdkRepo.Root(), SDKRepo: &sdkRepo, SpecCommitHash: specCommitHash, SpecRepoURL: c.flags.SwaggerRepo, UpdateSpecVersion: c.flags.UpdateSpecVersion, } if c.flags.SpecRPName == "" { c.flags.SpecRPName = c.rpName } var err error var result *common.GenerateResult var existTypeSpec bool if c.flags.TypeSpecConfig != "" { tsc, err := typespec.ParseTypeSpecConfig(c.flags.TypeSpecConfig) if err != nil { return err } existTypeSpec = tsc.ExistEmitOption(string(typespec.TypeSpec_GO)) generateCtx.TypeSpecConfig = tsc } if existTypeSpec { log.Printf("Generate SDK through TypeSpec...") result, err = generateCtx.GenerateForTypeSpec(&common.GenerateParam{ RPName: c.rpName, NamespaceName: c.namespaceName, SpecificPackageTitle: c.flags.PackageTitle, SpecificVersion: c.flags.VersionNumber, SpecRPName: c.flags.SpecRPName, ReleaseDate: c.flags.ReleaseDate, SkipGenerateExample: c.flags.SkipGenerateExample, GoVersion: c.flags.GoVersion, TypeSpecEmitOption: c.flags.TypeSpecGoOption, TspClientOptions: c.flags.TspClientOption, }) } else { log.Printf("Generate SDK through AutoRest...") result, err = generateCtx.GenerateForSingleRPNamespace(&common.GenerateParam{ RPName: c.rpName, NamespaceName: c.namespaceName, NamespaceConfig: c.flags.PackageConfig, SpecificPackageTitle: c.flags.PackageTitle, SpecificVersion: c.flags.VersionNumber, SpecRPName: c.flags.SpecRPName, ReleaseDate: c.flags.ReleaseDate, SkipGenerateExample: c.flags.SkipGenerateExample, GoVersion: c.flags.GoVersion, ForceStableVersion: c.flags.ForceStableVersion, }) } if err != nil { return fmt.Errorf("failed to finish release generation process: %+v", err) } // print generation result log.Printf("Generation result: %v", result) c.pullRequestLabels = result.PullRequestLabels if !c.flags.SkipCreateBranch { log.Printf("Create new branch for release") releaseBranchName := fmt.Sprintf(releaseBranchNamePattern, c.rpName, c.namespaceName, result.Version, time.Now().Unix()) if err := sdkRepo.CreateReleaseBranch(releaseBranchName); err != nil { return fmt.Errorf("failed to create release branch: %+v", err) } log.Printf("Include the packages that is about to release in this release and do release commit...") // append a time in long to avoid collision of branch names if err := sdkRepo.AddReleaseCommit(c.rpName, c.namespaceName, generateCtx.SpecCommitHash, result.Version); err != nil { return fmt.Errorf("failed to add release package or do release commit: %+v", err) } } return nil } func (c *commandContext) generateFromRequest(sdkRepo repo.SDKRepository, specRepoParam, specCommitHash string) error { var generateErr []error var pullRequestUrls = make(map[string]string) var pushBranch = make(map[string]struct { requestLink string pullRequestLabel string }) forkRemote, err := repo.GetForkRemote(sdkRepo) if err != nil { return err } log.Printf("track2 parsing the config...") cfg, err := config.ParseConfig(c.rpName) if err != nil { return fmt.Errorf("parse config err: %v", err) } log.Printf("Configuration: %s", cfg.String()) armServices, err := config.ParseTrack2(cfg, specRepoParam) if err != nil { generateErr = append(generateErr, err) } for arm, packageInfos := range armServices { for _, info := range packageInfos { originalHead, err := sdkRepo.Head() if err != nil { return err } // run generator c.rpName = arm c.namespaceName = info.Name c.flags.SpecRPName = info.SpecName c.flags.PackageConfig = info.Tag if info.ReleaseDate != nil { c.flags.ReleaseDate = info.ReleaseDate.Format("2006-01-02") } err = c.generate(sdkRepo, specCommitHash) if err != nil { generateErr = append(generateErr, err) continue } // get current branch name generateHead, err := sdkRepo.Head() if err != nil { return err } pushBranch[generateHead.Name().Short()] = struct { requestLink string pullRequestLabel string }{requestLink: info.RequestLink, pullRequestLabel: c.pullRequestLabels} log.Printf("git checkout %v", originalHead.Name().Short()) if err := sdkRepo.Checkout(&repo.CheckoutOptions{ Branch: plumbing.ReferenceName(originalHead.Name().Short()), Force: true, }); err != nil { return err } } } tspProjects, err := config.GetTypeSpecProjectsFromConfig(cfg, specRepoParam) if err != nil { generateErr = append(generateErr, err) } for rpName, packageInfos := range tspProjects { for _, packageInfo := range packageInfos { originalHead, err := sdkRepo.Head() if err != nil { return err } // run tsp generator c.rpName = rpName c.namespaceName = packageInfo.Name c.flags.TypeSpecConfig = packageInfo.TspConfigPath if packageInfo.ReleaseDate != nil { c.flags.ReleaseDate = packageInfo.ReleaseDate.Format("2006-01-02") } err = c.generate(sdkRepo, specCommitHash) if err != nil { generateErr = append(generateErr, err) continue } // get current branch name generateHead, err := sdkRepo.Head() if err != nil { return err } pushBranch[generateHead.Name().Short()] = struct { requestLink string pullRequestLabel string }{requestLink: packageInfo.RequestLink, pullRequestLabel: c.pullRequestLabels} log.Printf("git checkout %v", originalHead.Name().Short()) if err := sdkRepo.Checkout(&repo.CheckoutOptions{ Branch: plumbing.ReferenceName(originalHead.Name().Short()), Force: true, }); err != nil { return err } } } for branch := range pushBranch { log.Printf("Fixes: %s\n", branch) } if c.flags.Token != "" { for branchName, issue := range pushBranch { log.Printf("git push fork %s\n", branchName) msg, err := common.ExecuteGitPush(sdkRepo.Root(), forkRemote.Config().Name, branchName) if err != nil { return fmt.Errorf("git push fork error:%v,msg:%s", err, msg) } githubUserName := repo.GetRemoteUserName(forkRemote) if githubUserName == "" { return errors.New("github user name not exist") } log.Printf("%s: create pull request...\n", branchName) pullRequestUrl, err := common.ExecuteCreatePullRequest(sdkRepo.Root(), link.SpecOwner, link.SDKRepo, githubUserName, branchName, repo.ReleaseTitle(branchName), issue.requestLink, c.flags.Token, issue.pullRequestLabel) if err != nil { return err } pullRequestUrls[branchName] = pullRequestUrl log.Printf("Leave a comment in %s...\n", issue) issueNumber := strings.Split(issue.requestLink, "/") err = common.ExecuteAddIssueComment(sdkRepo.Root(), link.SpecOwner, link.ReleaseIssueRepo, issueNumber[len(issueNumber)-1], fmt.Sprintf(confirmComment, pullRequestUrl), c.flags.Token) if err != nil { return err } log.Printf("Add Labels...\n") err = common.ExecuteAddIssueLabels(sdkRepo.Root(), link.SpecOwner, link.ReleaseIssueRepo, issueNumber[len(issueNumber)-1], c.flags.Token, []string{"PRready", issue.pullRequestLabel}) if err != nil { return err } } } if len(pullRequestUrls) != 0 { log.Println("Fixes:") for branch, url := range pullRequestUrls { log.Printf("%s : %s", branch, url) } } if len(generateErr) != 0 { fmt.Println("generator error:") for _, e := range generateErr { fmt.Println(e) } } return nil }