Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/t-dedah/gh-cli-extension-demo
https://github.com/t-dedah/gh-cli-extension-demo
gh-extension
Last synced: 1 day ago
JSON representation
- Host: GitHub
- URL: https://github.com/t-dedah/gh-cli-extension-demo
- Owner: t-dedah
- License: mit
- Created: 2022-08-17T16:29:14.000Z (over 2 years ago)
- Default Branch: master
- Last Pushed: 2022-08-18T05:09:02.000Z (over 2 years ago)
- Last Synced: 2024-11-10T11:42:35.943Z (about 2 months ago)
- Topics: gh-extension
- Language: Go
- Homepage:
- Size: 15.6 KB
- Stars: 1
- Watchers: 1
- Forks: 0
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# gh-cli-extension-demo
This repo is part of a demo where we learn how to create an extension for gh-cli in golang.
## Libraries Used
1. [go-gh](https://github.com/cli/go-gh) - A Go module for interacting with gh and the GitHub API from the command line.
2. [Cobra](https://github.com/spf13/cobra) - It is a library used giving structure and managing inputs in our cli
3. [Gock](https://github.com/h2non/gock) - HTTP traffic mocking and testing library## Development Setup
1. Install [golang](https://go.dev/doc/install)
2. Install the `gh` CLI - see the [installation](https://github.com/cli/cli#installation)
_Installation requires a minimum version (2.0.0) of the the GitHub CLI that supports extensions._## Stage 1
In this stage we will:-
1. Create a basic extension
2. properly structure the folder structure
3. run our extension**A. Create extension**
gh extension create --precompiled=go
This command will generate a new precompiled golang extension.**B. Install cobra**
go get -u github.com/spf13/cobra@latest
**C. Update directory structure**
While we are welcome to provide our own organization, typically a Cobra-based application will follow the following organizational structure:▾ appName/
▾ cmd/
root.go
main.go
In a Cobra app, typically the main.go file is very bare. It serves one purpose: initializing Cobra.package main
import (
"{pathToYourApp}/cmd"
)func main() {
cmd.Execute()
}
**D. Final Code**root.go
package cmd
import (
"fmt"
"os""github.com/spf13/cobra"
)var rootCmd = &cobra.Command{
Use: "gh cli-extension-demo",
Short: "This extension prints out all the input user provides",
Run: func(cmd *cobra.Command, args []string) {
fmt.Println("Stage 1 of demo is done")
},
}func Execute() {
if err := rootCmd.Execute(); err != nil {
fmt.Fprintln(os.Stderr, err)
os.Exit(1)
}
}main.go
package main
import "github.com/t-dedah/gh-cli-extension-demo/cmd"
func main() {
cmd.Execute()
}**E. Build and Test**
1. Build your project using `go build`. This will generate a new binary for the cli.
2. Install the binary using `gh extension install .`. This will create a symlink between the binary this repo and go package directory in your machine. Any further builds will not require the installation step.
3. Test the extension using `gh ` This should print `Stage 1 of demo is done` on the terminal
## Stage 2
In this stage we will:-
1. Add a new command
2. Take input for command
3. Make api calls**A. Add new command**
1. Create a new file `list.go` on cmd folder
2. Add this to `list.go````
package cmdimport (
"fmt""github.com/spf13/cobra"
gh "github.com/cli/go-gh"
ghRepo "github.com/cli/go-gh/pkg/repository"
)type ListOptions struct {
Repo string
}func NewCmdList() *cobra.Command {
f := ListOptions{}var listCmd = &cobra.Command{
Use: "list",
Short: "Lists the issues",
RunE: func(cmd *cobra.Command, args []string) error {
fmt.Printf("Input Repo: %s", f.Repo)
return nil
},
}listCmd.Flags().StringVarP(&f.Repo, "repo", "R", "", "Select another repository for listing issues")
return listCmd
}
```This will take one string argument `-R` and print the value for that input.
**B. Add list command to root**
1. Add new function to `root.go````
func addCommandsToRoot() {
rootCmd.AddCommand(NewCmdList())
}
```2. Call this function from `Execute()`
```
func Execute() {
addCommandsToRoot()
if err := rootCmd.Execute(); err != nil {
fmt.Fprintln(os.Stderr, err)
os.Exit(1)
}
}
```3. Build and test the extension
**C. Use go-gh to make api calls**
1. Update the list.go```
package cmdimport (
"fmt""github.com/spf13/cobra"
gh "github.com/cli/go-gh"
ghRepo "github.com/cli/go-gh/pkg/repository"
"github.com/cli/go-gh/pkg/api"
)type ListOptions struct {
Repo string
Limit int
}func NewCmdList() *cobra.Command {
f := ListOptions{}var listCmd = &cobra.Command{
Use: "list",
Short: "Lists the issues",
RunE: func(cmd *cobra.Command, args []string) error {
if len(args) != 0 {
return fmt.Errorf(fmt.Sprintf("Invalid argument(s). Expected 0 received %d", len(args)))
}repo, err := GetRepo(f.Repo)
if err != nil {
return err
}opts := api.ClientOptions{
Host: repo.Host(),
}
restClient, err := gh.RESTClient(&opts)pathComponent := fmt.Sprintf("repos/%s/%s/actions/runs?per_page=%d", repo.Owner(), repo.Name(), f.Limit)
var apiResults ApiResponse
err = restClient.Get(pathComponent, &apiResults)
if err != nil {
return err
}
fmt.Println(apiResults.WorkflowRuns)
return nil
},
}listCmd.Flags().StringVarP(&f.Repo, "repo", "R", "", "Select another repository for listing issues")
listCmd.Flags().IntVarP(&f.Limit, "limit", "L", 5, "Limit of workflow runs to display")return listCmd
}func GetRepo(r string) (ghRepo.Repository, error) {
if r != "" {
return ghRepo.Parse(r)
}return gh.CurrentRepository()
}type ApiResponse struct {
TotalCount int `json:"total_count"`
WorkflowRuns []WorkflowRun `json:"workflow_runs"`
}type WorkflowRun struct {
Id int `json:"id"`
Name string `json:"name"`
Status string `json:"status"`
}
```**D. Add table printer**
1. This function will tabulated data using go-gh table printer
```
func PrettyPrint(workflowRuns []WorkflowRun) {
terminal := ghTerm.FromEnv()
w, _, _ := terminal.Size()
tp := ghTableprinter.New(terminal.Out(), terminal.IsTerminalOutput(), w)for _, workflowRun := range workflowRuns {
tp.AddField(strconv.Itoa(workflowRun.Id))
tp.AddField(workflowRun.Name)
tp.AddField(workflowRun.Status)
tp.EndRow()
}_ = tp.Render()
}
```2. Final list.go
```
package cmdimport (
"fmt"
"strconv""github.com/spf13/cobra"
gh "github.com/cli/go-gh"
ghRepo "github.com/cli/go-gh/pkg/repository"
"github.com/cli/go-gh/pkg/api"
ghTableprinter "github.com/cli/go-gh/pkg/tableprinter"
ghTerm "github.com/cli/go-gh/pkg/term"
)type ListOptions struct {
Repo string
Limit int
}func NewCmdList() *cobra.Command {
f := ListOptions{}var listCmd = &cobra.Command{
Use: "list",
Short: "Lists the issues",
RunE: func(cmd *cobra.Command, args []string) error {
if len(args) != 0 {
return fmt.Errorf(fmt.Sprintf("Invalid argument(s). Expected 0 received %d", len(args)))
}repo, err := GetRepo(f.Repo)
if err != nil {
return err
}opts := api.ClientOptions{
Host: repo.Host(),
}
restClient, err := gh.RESTClient(&opts)pathComponent := fmt.Sprintf("repos/%s/%s/actions/runs?per_page=%d", repo.Owner(), repo.Name(), f.Limit)
var apiResults ApiResponse
err = restClient.Get(pathComponent, &apiResults)
if err != nil {
return err
}
PrettyPrint(apiResults.WorkflowRuns)
return nil
},
}listCmd.Flags().StringVarP(&f.Repo, "repo", "R", "", "Select another repository for listing issues")
listCmd.Flags().IntVarP(&f.Limit, "limit", "L", 5, "Limit of workflow runs to display")return listCmd
}func PrettyPrint(workflowRuns []WorkflowRun) {
terminal := ghTerm.FromEnv()
w, _, _ := terminal.Size()
tp := ghTableprinter.New(terminal.Out(), terminal.IsTerminalOutput(), w)for _, workflowRun := range workflowRuns {
tp.AddField(strconv.Itoa(workflowRun.Id))
tp.AddField(workflowRun.Name)
tp.AddField(workflowRun.Status)
tp.EndRow()
}_ = tp.Render()
}func GetRepo(r string) (ghRepo.Repository, error) {
if r != "" {
return ghRepo.Parse(r)
}return gh.CurrentRepository()
}type ApiResponse struct {
TotalCount int `json:"total_count"`
WorkflowRuns []WorkflowRun `json:"workflow_runs"`
}type WorkflowRun struct {
Id int `json:"id"`
Name string `json:"name"`
Status string `json:"status"`
}
```3. Build and test
## Stage 3
In this stage we learn how to:-
1. Add tests
2. Mock api calls**A. Install packages**
1. go get github.com/stretchr/testify/assert
2. go get gopkg.in/h2non/gock.v1**B. Create test file**
1. Create a new file with name `list_test.go` in cmd folder.
2. Add first basic test```
package cmdimport (
"fmt"
"testing""github.com/stretchr/testify/assert"
"gopkg.in/h2non/gock.v1"
)func TestListWithIncorrectArguments(t *testing.T) {
t.Cleanup(gock.Off)
cmd := NewCmdList()
cmd.SetArgs([]string{"INCORRECT"})
err := cmd.Execute()assert.NotNil(t, err)
assert.Equal(t, err, fmt.Errorf("Invalid argument(s). Expected 0 received 1"))
}
```This test checks if the error being throw is correct or not.
**C. Test API calls**
Now we will mock the api calls using gock package.final list_test.go
```
package cmdimport (
"errors"
"fmt"
"strings"
"testing""github.com/stretchr/testify/assert"
"github.com/cli/go-gh/pkg/api"
"gopkg.in/h2non/gock.v1"
)func TestListWithIncorrectArguments(t *testing.T) {
t.Cleanup(gock.Off)
cmd := NewCmdList()
cmd.SetArgs([]string{"INCORRECT"})
err := cmd.Execute()assert.NotNil(t, err)
assert.Equal(t, err, fmt.Errorf("Invalid argument(s). Expected 0 received 1"))
}func TestListWithSuccess(t *testing.T) {
t.Cleanup(gock.Off)
gock.New("https://api.github.com").
Get("/repos/testOrg/testRepo/actions/runs").
Reply(200).
JSON(`{
"total_count":1,
"workflow_runs":[
{
"id":1,
"name":"Test Workflow",
"status":"queued"
}
]
}`)cmd := NewCmdList()
cmd.SetArgs([]string{"-R", "testOrg/testRepo"})
err := cmd.Execute()assert.Nil(t, err)
assert.True(t, gock.IsDone(), PrintPendingMocks(gock.Pending()))
}func TestListWithInternalServerError(t *testing.T) {
t.Cleanup(gock.Off)
gock.New("https://api.github.com").
Get("/repos/testOrg/testRepo/actions/runs").
Reply(500).
JSON(`{
"message": "Internal Server Error",
"documentation_url": "https://docs.github.com/rest/reference/actions#get-github-actions-cache-list-for-a-repository"
}`)cmd := NewCmdList()
cmd.SetArgs([]string{"-R", "testOrg/testRepo"})
err := cmd.Execute()assert.NotNil(t, err)
var httpError api.HTTPError
errors.As(err, &httpError)
assert.Equal(t, httpError.StatusCode, 500)assert.True(t, gock.IsDone(), PrintPendingMocks(gock.Pending()))
}func TestListWithPendingMocks(t *testing.T) {
t.Cleanup(gock.Off)
gock.New("https://api.github.com").
Get("/repos/testOrg/testRepo/actions/runs/xyz").
Reply(500).
JSON(`{
"message": "Internal Server Error",
"documentation_url": "https://docs.github.com/rest/reference/actions#get-github-actions-cache-list-for-a-repository"
}`)gock.New("https://api.github.com").
Get("/repos/testOrg/testRepo/actions/runs").
Reply(500).
JSON(`{
"message": "Internal Server Error",
"documentation_url": "https://docs.github.com/rest/reference/actions#get-github-actions-cache-list-for-a-repository"
}`)cmd := NewCmdList()
cmd.SetArgs([]string{"-R", "testOrg/testRepo"})
err := cmd.Execute()assert.NotNil(t, err)
assert.False(t, gock.IsDone(), PrintPendingMocks(gock.Pending()))
}func PrintPendingMocks(mocks []gock.Mock) string {
paths := []string{}
for _, mock := range mocks {
paths = append(paths, mock.Request().URLStruct.String())
}
return fmt.Sprintf("%d unmatched mocks: %s", len(paths), strings.Join(paths, ", "))
}```
## Stage 4
Last stage of the demo is to know how to release the newly created extension.1. For releasing we use `cli/gh-extension-precompile@v1` workflow which builds binary for all the major platforms like linux, windows, darwin and many more
Sample workflow to test and release
```
name: releaseon:
push:
tags:
- "v*"permissions:
contents: writejobs:
release:
runs-on: ubuntu-latest
steps:
- name: Set up Go 1.18
uses: actions/setup-go@v3
with:
go-version: 1.18
- name: Check out code
uses: actions/checkout@v3- name: Restore Go modules cache
uses: actions/cache@v3
with:
path: ~/go/pkg/mod
key: go-${{ runner.os }}-${{ hashFiles('go.mod') }}
restore-keys: |
go-${{ runner.os }}-- name: Download dependencies
run: go mod download- name: Run tests
run: go test -v ./...
env:
GH_TOKEN: dummy-token-to-facilitate-rest-client- uses: cli/gh-extension-precompile@v1
with:
go_version: "1.18"
```2. Add `gh-extension` topic to the repo for more visibility