Skip to content

Commit 67bb208

Browse files
author
OpenShift Bot
committed
Merge pull request #5476 from smarterclayton/display_info_on_new_app
Merged by openshift-bot
2 parents 7d78675 + 93f4af3 commit 67bb208

File tree

16 files changed

+427
-149
lines changed

16 files changed

+427
-149
lines changed

contrib/completions/bash/oc

+2
Original file line numberDiff line numberDiff line change
@@ -305,6 +305,7 @@ _oc_new-app()
305305
flags+=("--code=")
306306
flags+=("--context-dir=")
307307
flags+=("--docker-image=")
308+
flags+=("--dry-run")
308309
flags+=("--env=")
309310
two_word_flags+=("-e")
310311
flags+=("--file=")
@@ -477,6 +478,7 @@ _oc_new-build()
477478
flags+=("--docker-image=")
478479
flags+=("--dockerfile=")
479480
two_word_flags+=("-D")
481+
flags+=("--dry-run")
480482
flags+=("--env=")
481483
two_word_flags+=("-e")
482484
flags+=("--image=")

contrib/completions/bash/openshift

+2
Original file line numberDiff line numberDiff line change
@@ -2113,6 +2113,7 @@ _openshift_cli_new-app()
21132113
flags+=("--code=")
21142114
flags+=("--context-dir=")
21152115
flags+=("--docker-image=")
2116+
flags+=("--dry-run")
21162117
flags+=("--env=")
21172118
two_word_flags+=("-e")
21182119
flags+=("--file=")
@@ -2285,6 +2286,7 @@ _openshift_cli_new-build()
22852286
flags+=("--docker-image=")
22862287
flags+=("--dockerfile=")
22872288
two_word_flags+=("-D")
2289+
flags+=("--dry-run")
22882290
flags+=("--env=")
22892291
two_word_flags+=("-e")
22902292
flags+=("--image=")

hack/util.sh

+1-1
Original file line numberDiff line numberDiff line change
@@ -220,7 +220,7 @@ function tryuntil {
220220
timeout=$(($(date +%s) + 90))
221221
echo "++ Retrying until success or timeout: ${@}"
222222
while [ 1 ]; do
223-
if eval "${@}" 2>&1 >/dev/null; then
223+
if eval "${@}" >/dev/null 2>&1; then
224224
return 0
225225
fi
226226
if [[ $(date +%s) -gt $timeout ]]; then

pkg/cmd/cli/cmd/newapp.go

+66-24
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ package cmd
33
import (
44
"fmt"
55
"io"
6+
"io/ioutil"
67
"os"
78
"sort"
89
"strings"
@@ -17,6 +18,7 @@ import (
1718
kcmd "k8s.io/kubernetes/pkg/kubectl/cmd"
1819
cmdutil "k8s.io/kubernetes/pkg/kubectl/cmd/util"
1920
"k8s.io/kubernetes/pkg/kubectl/resource"
21+
"k8s.io/kubernetes/pkg/labels"
2022
"k8s.io/kubernetes/pkg/runtime"
2123
"k8s.io/kubernetes/pkg/util/errors"
2224
"k8s.io/kubernetes/pkg/util/sets"
@@ -115,6 +117,7 @@ To search templates, image streams, and Docker images that match the arguments p
115117
// NewCmdNewApplication implements the OpenShift cli new-app command
116118
func NewCmdNewApplication(fullName string, f *clientcmd.Factory, out io.Writer) *cobra.Command {
117119
config := newcmd.NewAppConfig()
120+
config.Deploy = true
118121

119122
cmd := &cobra.Command{
120123
Use: "new-app (IMAGE | IMAGESTREAM | TEMPLATE | PATH | URL ...)",
@@ -155,6 +158,7 @@ func NewCmdNewApplication(fullName string, f *clientcmd.Factory, out io.Writer)
155158
cmd.Flags().BoolVar(&config.AllowMissingImages, "allow-missing-images", false, "If true, indicates that referenced Docker images that cannot be found locally or in a registry should still be used.")
156159
cmd.Flags().BoolVar(&config.AllowSecretUse, "grant-install-rights", false, "If true, a component that requires access to your account may use your token to install software into your project. Only grant images you trust the right to run with your token.")
157160
cmd.Flags().BoolVar(&config.SkipGeneration, "no-install", false, "Do not attempt to run images that describe themselves as being installable")
161+
cmd.Flags().BoolVar(&config.DryRun, "dry-run", false, "If true, do not actually create resources.")
158162

159163
// TODO AddPrinterFlags disabled so that it doesn't conflict with our own "template" flag.
160164
// Need a better solution.
@@ -170,10 +174,15 @@ func NewCmdNewApplication(fullName string, f *clientcmd.Factory, out io.Writer)
170174
// RunNewApplication contains all the necessary functionality for the OpenShift cli new-app command
171175
func RunNewApplication(fullName string, f *clientcmd.Factory, out io.Writer, c *cobra.Command, args []string, config *newcmd.AppConfig) error {
172176
output := cmdutil.GetFlagString(c, "output")
177+
shortOutput := output == "name"
173178

174179
if err := setupAppConfig(f, out, c, args, config); err != nil {
175180
return err
176181
}
182+
if shortOutput || len(output) != 0 {
183+
config.Out = ioutil.Discard
184+
}
185+
177186
if config.Querying() {
178187
result, err := config.RunQuery()
179188
if err != nil {
@@ -189,11 +198,15 @@ func RunNewApplication(fullName string, f *clientcmd.Factory, out io.Writer, c *
189198
if err := setAppConfigLabels(c, config); err != nil {
190199
return err
191200
}
192-
result, err := config.RunAll()
201+
result, err := config.Run()
193202
if err := handleRunError(c, err, fullName); err != nil {
194203
return err
195204
}
196205

206+
if len(config.Labels) == 0 && len(result.Name) > 0 {
207+
config.Labels = map[string]string{"app": result.Name}
208+
}
209+
197210
if err := setLabels(config.Labels, result); err != nil {
198211
return err
199212
}
@@ -202,15 +215,41 @@ func RunNewApplication(fullName string, f *clientcmd.Factory, out io.Writer, c *
202215
return err
203216
}
204217

205-
if len(output) != 0 && output != "name" {
218+
indent := " "
219+
switch {
220+
case shortOutput:
221+
indent = ""
222+
case len(output) != 0:
206223
return f.Factory.PrintObject(c, result.List, out)
224+
case !result.GeneratedJobs:
225+
if len(config.Labels) > 0 {
226+
fmt.Fprintf(out, "--> Creating resources with label %s ...\n", labels.SelectorFromSet(config.Labels).String())
227+
} else {
228+
fmt.Fprintf(out, "--> Creating resources ...\n")
229+
}
230+
}
231+
if config.DryRun {
232+
return nil
207233
}
208234

235+
mapper, _ := f.Object()
236+
var afterFn func(*resource.Info, error)
237+
switch {
209238
// only print success if we don't have installables
210-
if err := createObjects(f, out, c.Out(), output == "name", !result.GeneratedJobs, result); err != nil {
239+
case !result.GeneratedJobs:
240+
afterFn = configcmd.NewPrintNameOrErrorAfterIndent(mapper, shortOutput, "created", out, c.Out(), indent)
241+
default:
242+
afterFn = configcmd.NewPrintErrorAfter(mapper, c.Out())
243+
}
244+
245+
if err := createObjects(f, afterFn, result); err != nil {
211246
return err
212247
}
213248

249+
if !shortOutput && !result.GeneratedJobs {
250+
fmt.Fprintf(out, "--> Success\n")
251+
}
252+
214253
hasMissingRepo := false
215254
installing := []*kapi.Pod{}
216255
for _, item := range result.List.Items {
@@ -221,18 +260,23 @@ func RunNewApplication(fullName string, f *clientcmd.Factory, out io.Writer, c *
221260
}
222261
case *buildapi.BuildConfig:
223262
if len(t.Spec.Triggers) > 0 {
224-
fmt.Fprintf(c.Out(), "Build scheduled for %q - use the build-logs command to track its progress.\n", t.Name)
263+
fmt.Fprintf(out, "%sBuild scheduled for %q - use the build-logs command to track its progress.\n", indent, t.Name)
225264
}
226265
case *imageapi.ImageStream:
227266
if len(t.Status.DockerImageRepository) == 0 {
228267
if hasMissingRepo {
229268
continue
230269
}
231270
hasMissingRepo = true
232-
fmt.Fprint(c.Out(), "WARNING: No Docker registry has been configured with the server. Automatic builds and deployments may not function.\n")
271+
fmt.Fprintf(out, "%sWARNING: No Docker registry has been configured with the server. Automatic builds and deployments may not function.\n", indent)
233272
}
234273
}
235274
}
275+
276+
if shortOutput {
277+
return nil
278+
}
279+
236280
switch {
237281
case len(installing) == 1:
238282
// TODO: should get this set on the config or up above
@@ -244,16 +288,16 @@ func RunNewApplication(fullName string, f *clientcmd.Factory, out io.Writer, c *
244288
return followInstallation(f, jobInput, installing[0], kclient, out)
245289
case len(installing) > 1:
246290
for i := range installing {
247-
fmt.Fprintf(c.Out(), "Track installation of %s with '%s logs %s'.\n", installing[i].Name, fullName, installing[i].Name)
291+
fmt.Fprintf(out, "%sTrack installation of %s with '%s logs %s'.\n", indent, installing[i].Name, fullName, installing[i].Name)
248292
}
249293
case len(result.List.Items) > 0:
250-
fmt.Fprintf(c.Out(), "Run '%s %s' to view your app.\n", fullName, StatusRecommendedName)
294+
fmt.Fprintf(out, "%sRun '%s %s' to view your app.\n", indent, fullName, StatusRecommendedName)
251295
}
252296
return nil
253297
}
254298

255299
func followInstallation(f *clientcmd.Factory, input string, pod *kapi.Pod, kclient kclient.Interface, out io.Writer) error {
256-
fmt.Fprintf(out, "Installing %q with pod %q ...\n", input, pod.Name)
300+
fmt.Fprintf(out, "--> Installing ...\n")
257301

258302
// we cannot retrieve logs until the pod is out of pending
259303
// TODO: move this to the server side
@@ -307,7 +351,9 @@ func installationStarted(c kclient.PodInterface, name string, s kclient.SecretsI
307351
if secret, err := s.Get(name); err == nil {
308352
if secret.Annotations[newcmd.GeneratedForJob] == "true" &&
309353
secret.Annotations[newcmd.GeneratedForJobFor] == pod.Annotations[newcmd.GeneratedForJobFor] {
310-
s.Delete(name)
354+
if err := s.Delete(name); err != nil {
355+
glog.V(4).Infof("Failed to delete install secret %s: %v", name, err)
356+
}
311357
}
312358
}
313359
return true, nil
@@ -325,8 +371,10 @@ func installationComplete(c kclient.PodInterface, name string, out io.Writer) wa
325371
}
326372
switch pod.Status.Phase {
327373
case kapi.PodSucceeded:
328-
fmt.Fprintf(out, "Installation complete\n")
329-
c.Delete(name, nil)
374+
fmt.Fprintf(out, "--> Success\n")
375+
if err := c.Delete(name, nil); err != nil {
376+
glog.V(4).Infof("Failed to delete install pod %s: %v", name, err)
377+
}
330378
return true, nil
331379
case kapi.PodFailed:
332380
return true, fmt.Errorf("installation of %q did not complete successfully", name)
@@ -404,11 +452,6 @@ func setAnnotations(annotations map[string]string, result *newcmd.AppResult) err
404452
}
405453

406454
func setLabels(labels map[string]string, result *newcmd.AppResult) error {
407-
if len(labels) == 0 {
408-
if len(result.Name) > 0 {
409-
labels = map[string]string{"app": result.Name}
410-
}
411-
}
412455
for _, object := range result.List.Items {
413456
err := util.AddObjectLabels(object, labels)
414457
if err != nil {
@@ -459,22 +502,18 @@ func retryBuildConfig(info *resource.Info, err error) runtime.Object {
459502
return nil
460503
}
461504

462-
func createObjects(f *clientcmd.Factory, out, errout io.Writer, shortOutput, includeSuccess bool, result *newcmd.AppResult) error {
505+
func createObjects(f *clientcmd.Factory, after func(*resource.Info, error), result *newcmd.AppResult) error {
463506
mapper, typer := f.Factory.Object()
464507
bulk := configcmd.Bulk{
465508
Mapper: mapper,
466509
Typer: typer,
467510
RESTClientFactory: f.Factory.RESTClient,
511+
512+
After: after,
468513
// Retry is used to support previous versions of the API server that will
469514
// consider the presence of an unknown trigger type to be an error.
470515
Retry: retryBuildConfig,
471516
}
472-
switch {
473-
case includeSuccess:
474-
bulk.After = configcmd.NewPrintNameOrErrorAfter(mapper, shortOutput, "created", out, errout)
475-
default:
476-
bulk.After = configcmd.NewPrintErrorAfter(mapper, errout)
477-
}
478517
if errs := bulk.Create(result.List, result.Namespace); len(errs) != 0 {
479518
return errExit
480519
}
@@ -492,7 +531,10 @@ func handleRunError(c *cobra.Command, err error, fullName string) error {
492531
}
493532
switch t := err.(type) {
494533
case newcmd.ErrRequiresExplicitAccess:
495-
return fmt.Errorf("installing %q requires that you grant the image access to run with your credentials; if you trust the provided image, include the flag --grant-install-rights", t.Match.Value)
534+
return fmt.Errorf(`installing %q requires that you grant the image access to run with your credentials
535+
536+
You can see more information about the image by adding the --dry-run flag.
537+
If you trust the provided image, include the flag --grant-install-rights.`, t.Match.Value)
496538
case newapp.ErrNoMatch:
497539
return fmt.Errorf(`%[1]v
498540

pkg/cmd/cli/cmd/newbuild.go

+35-5
Original file line numberDiff line numberDiff line change
@@ -12,8 +12,10 @@ import (
1212

1313
buildapi "github.com/openshift/origin/pkg/build/api"
1414
"github.com/openshift/origin/pkg/cmd/util/clientcmd"
15+
configcmd "github.com/openshift/origin/pkg/config/cmd"
1516
newapp "github.com/openshift/origin/pkg/generate/app"
1617
newcmd "github.com/openshift/origin/pkg/generate/app/cmd"
18+
"k8s.io/kubernetes/pkg/labels"
1719
)
1820

1921
const (
@@ -104,6 +106,7 @@ func NewCmdNewBuild(fullName string, f *clientcmd.Factory, in io.Reader, out io.
104106
cmd.Flags().StringP("labels", "l", "", "Label to set in all generated resources.")
105107
cmd.Flags().BoolVar(&config.AllowMissingImages, "allow-missing-images", false, "If true, indicates that referenced Docker images that cannot be found locally or in a registry should still be used.")
106108
cmd.Flags().StringVar(&config.ContextDir, "context-dir", "", "Context directory to be used for the build.")
109+
cmd.Flags().BoolVar(&config.DryRun, "dry-run", false, "If true, do not actually create resources.")
107110
cmdutil.AddPrinterFlags(cmd)
108111

109112
return cmd
@@ -112,6 +115,7 @@ func NewCmdNewBuild(fullName string, f *clientcmd.Factory, in io.Reader, out io.
112115
// RunNewBuild contains all the necessary functionality for the OpenShift cli new-build command
113116
func RunNewBuild(fullName string, f *clientcmd.Factory, out io.Writer, in io.Reader, c *cobra.Command, args []string, config *newcmd.AppConfig) error {
114117
output := cmdutil.GetFlagString(c, "output")
118+
shortOutput := output == "name"
115119

116120
if config.Dockerfile == "-" {
117121
data, err := ioutil.ReadAll(in)
@@ -128,31 +132,57 @@ func RunNewBuild(fullName string, f *clientcmd.Factory, out io.Writer, in io.Rea
128132
if err := setAppConfigLabels(c, config); err != nil {
129133
return err
130134
}
131-
result, err := config.RunBuilds()
135+
result, err := config.Run()
132136
if err != nil {
133137
return handleBuildError(c, err, fullName)
134138
}
139+
140+
if len(config.Labels) == 0 && len(result.Name) > 0 {
141+
config.Labels = map[string]string{"build": result.Name}
142+
}
143+
135144
if err := setLabels(config.Labels, result); err != nil {
136145
return err
137146
}
138147
if err := setAnnotations(map[string]string{newcmd.GeneratedByNamespace: newcmd.GeneratedByNewBuild}, result); err != nil {
139148
return err
140149
}
141-
if len(output) != 0 && output != "name" {
150+
151+
indent := " "
152+
switch {
153+
case shortOutput:
154+
indent = ""
155+
case len(output) != 0:
142156
return f.Factory.PrintObject(c, result.List, out)
157+
default:
158+
if len(config.Labels) > 0 {
159+
fmt.Fprintf(out, "--> Creating resources with label %s ...\n", labels.SelectorFromSet(config.Labels).String())
160+
} else {
161+
fmt.Fprintf(out, "--> Creating resources ...\n")
162+
}
143163
}
144-
if err := createObjects(f, out, c.Out(), output == "name", true, result); err != nil {
164+
if config.DryRun {
165+
return nil
166+
}
167+
168+
mapper, _ := f.Object()
169+
if err := createObjects(f, configcmd.NewPrintNameOrErrorAfterIndent(mapper, shortOutput, "created", out, c.Out(), indent), result); err != nil {
145170
return err
146171
}
147172

173+
if shortOutput {
174+
return nil
175+
}
176+
177+
fmt.Fprintf(out, "--> Success\n")
148178
for _, item := range result.List.Items {
149179
switch t := item.(type) {
150180
case *buildapi.BuildConfig:
151-
fmt.Fprintf(c.Out(), "Build configuration %q created and build triggered.\n", t.Name)
181+
fmt.Fprintf(out, "%sBuild configuration %q created and build triggered.\n", indent, t.Name)
152182
}
153183
}
154184
if len(result.List.Items) > 0 {
155-
fmt.Fprintf(c.Out(), "Run '%s %s' to check the progress.\n", fullName, StatusRecommendedName)
185+
fmt.Fprintf(out, "%sRun '%s %s' to check the progress.\n", indent, fullName, StatusRecommendedName)
156186
}
157187

158188
return nil

pkg/cmd/cli/describe/helpers.go

+5
Original file line numberDiff line numberDiff line change
@@ -113,6 +113,11 @@ func formatRelativeTime(t time.Time) string {
113113
return units.HumanDuration(timeNowFn().Sub(t))
114114
}
115115

116+
// FormatRelativeTime converts a time field into a human readable age string (hours, minutes, days).
117+
func FormatRelativeTime(t time.Time) string {
118+
return formatRelativeTime(t)
119+
}
120+
116121
func formatMeta(out *tabwriter.Writer, m api.ObjectMeta) {
117122
formatString(out, "Name", m.Name)
118123
if !m.CreationTimestamp.IsZero() {

pkg/config/cmd/cmd.go

+6-2
Original file line numberDiff line numberDiff line change
@@ -21,12 +21,16 @@ type Bulk struct {
2121
}
2222

2323
func NewPrintNameOrErrorAfter(mapper meta.RESTMapper, short bool, operation string, out, errs io.Writer) func(*resource.Info, error) {
24+
return NewPrintNameOrErrorAfterIndent(mapper, short, operation, out, errs, "")
25+
}
26+
27+
func NewPrintNameOrErrorAfterIndent(mapper meta.RESTMapper, short bool, operation string, out, errs io.Writer, indent string) func(*resource.Info, error) {
2428
return func(info *resource.Info, err error) {
2529
if err == nil {
26-
//cmdutil.PrintSuccess(mapper, short, out, info.Mapping.Resource, info.Name, operation)
30+
fmt.Fprintf(out, indent)
2731
cmdutil.PrintSuccess(mapper, short, out, info.Mapping.Kind, info.Name, operation)
2832
} else {
29-
fmt.Fprintf(errs, "error: %v\n", err)
33+
fmt.Fprintf(errs, "%serror: %v\n", indent, err)
3034
}
3135
}
3236
}

0 commit comments

Comments
 (0)