Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[EP] CLI #432

Merged
merged 2 commits into from
Dec 7, 2020
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
195 changes: 195 additions & 0 deletions docs/proposals/cli.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,195 @@
<!--
Copyright The Shipwright Contributors

SPDX-License-Identifier: Apache-2.0
-->

---
title: CLI
authors:
- "@otaviof"

reviewers:
- "@gabemontero"
- "@HeavyWombat"
- "@coreydaley"

approvers:
- TBD
creation-date: 2020-10-07
last-updated: 2020-11-30
status: provisional
---

CLI
---

# Summary

This enhancement proposal describes a command-line client for Shipwright Build operator, covering
from the command line structure, the usage as a kubectl plugin, and as a standalone binary, as
well as testing this new project.

# Motivation

Command-line interface is a fundamental component for developer experience and modern automation.
This enhancement proposal describes a command-line interface, i.e. "CLI", to enhance the developer
experience and provide easier means to interact and automate Shipwright's Builds.

## Goals

- Improve the user experience by providing a specialized client
- Make possible extended automation with shell scripting

## Non-Goals

- Re-implement operator logic
- Become the mandatory API client

# Proposal

Create a command-line client for Shipwright Build to improve the user experience and also improve
the ability to automate workflows.

This new project will concentrate our effects on the API client side of the Shipwright project,
and by maintaining the expected separation of `cmd` and `pkg`, developers can embed and extend the
possibilities of consuming Shipwright Operator API.

A [proof-of-concept command-line interface project](https://github.com/otaviof/shp) has been
written to scaffold the features, and evaluate the changes proposed in this document. The project
documentation (`README.md`) is focused on developers and contributors, while the features it will
provide are contained in this document.

## Usage and Installation

This command-line application will be consumed as a standalone binary, or as a plugin in `kubectl`.
This implies a few more extra settings, and making sure the application binary complies with the
[expected convention](https://krew.sigs.k8s.io/docs/developer-guide/develop/best-practices/).

### Standalone

The standalone binary will be available via the usual means, final users will be able to simply
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Some elaboration here would be good... minimally, point to the Releasing section for both short term and long term goals.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sure, I'm adding a pointer to "releasing".

install with `go get…`, or pick a pre-compiled binary from GitHub release pages. Consider the
[release section](#Releasing) for more.

### Plugin

As a [krew plugin](https://krew.sigs.k8s.io/docs/developer-guide/), the expected usage is:

```sh
$ kubectl shp ...
```

And the following global flags are expected, to keep consistency with `kubectl`:

- `-h/--help`
- `-n/--namespace`
- `-A/--all-namespaces`

Those three flags are mandatory to keep compatibility. However, the
[upstream kubectl package](https://pkg.go.dev/k8s.io/[email protected]/pkg/cmd/util) provides
[more options](https://github.com/otaviof/shp/blob/55ce2e8d58435b0264e3db0bef5cf439abfeca18/vendor/k8s.io/cli-runtime/pkg/genericclioptions/config_flags.go#L317)
out of the box.

## Command

The project will use [cobra](https://github.com/spf13/cobra)/[viper](https://github.com/spf13/viper)
for scaffolding the command-line structure, which in terms of organization and workflow, follows
`kubectl` convention. Thus, the following pattern:

```sh
$ shp <verb> <resource> <name> [options]
```

For instance:

```sh
$ shp create build nodejs-ex
$ shp run build nodejs-ex
```

Alternatively, we may have a more conventional pattern, as in:

```sh
$ shp <resource> [verb] <name> [options]
```

For example:
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The examples below do not follow the above mentioned shp <resource> <verb> syntax.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've updated them.


```sh
$ shp build create nodejs-ex
$ shp build run nodejs-ex
$ shp logs nodejs-ex
```

During the review process of this enhancement proposal, we must decide which pattern suits better,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

+1 for the shp <verb> <resource> pattern

which one we prefer to go ahead with.

## Features

For the initial release, the aim is to be able to define a `Build`, run it and display logs.

### Defining a Build

```sh
$ shp <verb> build <name> [options]
```

Will be responsible for managing Build objects, using the verbs: `create`, `update`, list and
`delete`. Later on, end users should have the ability to bootstrap a local repository clone using a
single command.

### Running a Build

```sh
$ shp run build <name> [options]
```

This sub-command will instantiate a new BuildRun resource for a Build, effectively starting the
build process. And, will print follow up commands that may be issued to see the BuildRun status
and inspect logs.

### Logs

```sh
$ shp logs <name> [options]
```

Retrieve all logs related to the informed `BuildRun` name. It will also be able to follow
(`--follow`) container log output as they are executed. The log lines displayed are organised by
the sequence of steps, and easy to read the whole build process output in a single go.

Additionally, it must provide easy syntax to reach a specific `BuildRun` instance, or the most
recent `BuildRun` for a `Build`.

## More Features

The command-line interface is planned to host more features, like for instance managing local and
remove artifacts, as well as help end-users to upload data into the cluster.

Thus, the initial design of the CLI must allow more subcommands to be added, accommodating more
features which will depend in a client.

## Testing

The strategy to test the command-line application will be based on `go test` for the unit testing,
and for end-to-end testing we should adopt [bats](https://github.com/sstephenson/bats). Bats
framework gives us a structured way to run shell script commands and what we expect them to
return, the command-line client shp is no more than a shell command.

The testing structure will be composed of:

- **Unit**: written in Golang and using [Gomega](https://onsi.github.io/gomega/) for assertion;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not sure if this belongs here, but I would prefer to forgo using Gomega and stick with pure go tests.

- **End-to-End**: written in Golang and located at the traditional `test/e2e` directory;
- **System**: written using Bats framework (Bash), or similar approach;

Bats will also be helpful for a future `shp` container-image, we are able to mount the Bats files
in the container-image produced, and run our system testing against it. Therefore, Bats covers
testing of a local command-line binary, as well as it does a container-image. Another tool may take
Bats' place, covering the same test-use cases accordingly.

## Releasing

The command-line release will initially be available on GitHub Pages process, on which we can also
index on Shipwright website. Later on we can evaluate the need for a container-image carrying on
the binary, or an RPM.