@@ -3,6 +3,8 @@ package clientcmd
3
3
import (
4
4
"errors"
5
5
"fmt"
6
+ "os"
7
+ "reflect"
6
8
"sort"
7
9
"strconv"
8
10
"time"
@@ -13,6 +15,7 @@ import (
13
15
"k8s.io/kubernetes/pkg/api/meta"
14
16
kclient "k8s.io/kubernetes/pkg/client/unversioned"
15
17
kclientcmd "k8s.io/kubernetes/pkg/client/unversioned/clientcmd"
18
+ kclientcmdapi "k8s.io/kubernetes/pkg/client/unversioned/clientcmd/api"
16
19
"k8s.io/kubernetes/pkg/fields"
17
20
"k8s.io/kubernetes/pkg/kubectl"
18
21
cmdutil "k8s.io/kubernetes/pkg/kubectl/cmd/util"
@@ -33,24 +36,73 @@ import (
33
36
routegen "github.com/openshift/origin/pkg/route/generator"
34
37
)
35
38
39
+ // defaultClusterConfigURL is a local name that is used to identify when the client config
40
+ // is unspecified.
41
+ const defaultClusterConfigURL = "https://localhost:8443"
42
+
36
43
// New creates a default Factory for commands that should share identical server
37
44
// connection behavior. Most commands should use this method to get a factory.
38
45
func New (flags * pflag.FlagSet ) * Factory {
39
- // Override global default to "" so we force the client to ask for user input
40
46
// TODO refactor this upstream:
41
47
// DefaultCluster should not be a global
42
48
// A call to ClientConfig() should always return the best clientCfg possible
43
49
// even if an error was returned, and let the caller decide what to do
44
- kclientcmd .DefaultCluster .Server = ""
50
+ kclientcmd .DefaultCluster .Server = defaultClusterConfigURL
45
51
46
52
// TODO: there should be two client configs, one for OpenShift, and one for Kubernetes
47
53
clientConfig := DefaultClientConfig (flags )
54
+ clientConfig = defaultingClientConfig {clientConfig }
48
55
f := NewFactory (clientConfig )
49
56
f .BindFlags (flags )
50
57
51
58
return f
52
59
}
53
60
61
+ // defaultingClientConfig detects whether the provided config is the default, and if
62
+ // so returns an error that indicates the user should set up their config.
63
+ type defaultingClientConfig struct {
64
+ nested kclientcmd.ClientConfig
65
+ }
66
+
67
+ // RawConfig calls the nested method
68
+ func (c defaultingClientConfig ) RawConfig () (kclientcmdapi.Config , error ) {
69
+ return c .nested .RawConfig ()
70
+ }
71
+
72
+ // Namespace calls the nested method
73
+ func (c defaultingClientConfig ) Namespace () (string , bool , error ) {
74
+ return c .nested .Namespace ()
75
+ }
76
+
77
+ // ClientConfig returns a complete client config
78
+ func (c defaultingClientConfig ) ClientConfig () (* kclient.Config , error ) {
79
+ cfg , err := c .nested .ClientConfig ()
80
+ if err != nil {
81
+ return nil , err
82
+ }
83
+ if isDefaultConfig (cfg ) {
84
+ return nil , fmt .Errorf (`No configuration file found, please login or point to an existing file:
85
+
86
+ 1. Via the command-line flag --config
87
+ 2. Via the KUBECONFIG environment variable
88
+ 3. In your home directory as ~/.kube/config
89
+
90
+ To view or setup config directly use the 'config' command.` )
91
+ }
92
+ return cfg , nil
93
+ }
94
+
95
+ func isDefaultConfig (cfg * kclient.Config ) bool {
96
+ if len (os .Getenv ("KUBERNETES_MASTER" )) > 0 {
97
+ return false
98
+ }
99
+ defaultConfig , err := kclientcmd .DefaultClientConfig .ClientConfig ()
100
+ if err != nil {
101
+ return false
102
+ }
103
+ return reflect .DeepEqual (cfg , defaultConfig )
104
+ }
105
+
54
106
// Factory provides common options for OpenShift commands
55
107
type Factory struct {
56
108
* cmdutil.Factory
0 commit comments