Unblock startup if kubernetes API is unavailable (#2126)
This commit is contained in:
parent
fdb31a27f3
commit
6539a10a62
3 changed files with 67 additions and 8 deletions
|
@ -65,6 +65,10 @@ func (k Kubernetes) ServeDNS(ctx context.Context, w dns.ResponseWriter, r *dns.M
|
||||||
if k.Fall.Through(state.Name()) {
|
if k.Fall.Through(state.Name()) {
|
||||||
return plugin.NextOrFailure(k.Name(), k.Next, ctx, w, r)
|
return plugin.NextOrFailure(k.Name(), k.Next, ctx, w, r)
|
||||||
}
|
}
|
||||||
|
if !k.APIConn.HasSynced() {
|
||||||
|
// If we haven't synchronized with the kubernetes cluster, return server failure
|
||||||
|
return plugin.BackendError(&k, zone, dns.RcodeServerFailure, state, nil /* err */, opt)
|
||||||
|
}
|
||||||
return plugin.BackendError(&k, zone, dns.RcodeNameError, state, nil /* err */, opt)
|
return plugin.BackendError(&k, zone, dns.RcodeNameError, state, nil /* err */, opt)
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -377,9 +377,58 @@ func TestServeDNS(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
type APIConnServeTest struct{}
|
var notSyncedTestCases = []test.Case{
|
||||||
|
{
|
||||||
|
// We should get ServerFailure instead of NameError for missing records when we kubernetes hasn't synced
|
||||||
|
Qname: "svc0.testns.svc.cluster.local.", Qtype: dns.TypeA,
|
||||||
|
Rcode: dns.RcodeServerFailure,
|
||||||
|
Ns: []dns.RR{
|
||||||
|
test.SOA("cluster.local. 303 IN SOA ns.dns.cluster.local. hostmaster.cluster.local. 1499347823 7200 1800 86400 60"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
func (APIConnServeTest) HasSynced() bool { return true }
|
func TestNotSyncedServeDNS(t *testing.T) {
|
||||||
|
|
||||||
|
k := New([]string{"cluster.local."})
|
||||||
|
k.APIConn = &APIConnServeTest{
|
||||||
|
notSynced: true,
|
||||||
|
}
|
||||||
|
k.Next = test.NextHandler(dns.RcodeSuccess, nil)
|
||||||
|
k.Namespaces = map[string]bool{"testns": true}
|
||||||
|
ctx := context.TODO()
|
||||||
|
|
||||||
|
for i, tc := range notSyncedTestCases {
|
||||||
|
r := tc.Msg()
|
||||||
|
|
||||||
|
w := dnstest.NewRecorder(&test.ResponseWriter{})
|
||||||
|
|
||||||
|
_, err := k.ServeDNS(ctx, w, r)
|
||||||
|
if err != tc.Error {
|
||||||
|
t.Errorf("Test %d expected no error, got %v", i, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if tc.Error != nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := w.Msg
|
||||||
|
if resp == nil {
|
||||||
|
t.Fatalf("Test %d, got nil message and no error for %q", i, r.Question[0].Name)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Before sorting, make sure that CNAMES do not appear after their target records
|
||||||
|
test.CNAMEOrder(t, resp)
|
||||||
|
|
||||||
|
test.SortAndCheck(t, resp, tc)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type APIConnServeTest struct {
|
||||||
|
notSynced bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a APIConnServeTest) HasSynced() bool { return !a.notSynced }
|
||||||
func (APIConnServeTest) Run() { return }
|
func (APIConnServeTest) Run() { return }
|
||||||
func (APIConnServeTest) Stop() error { return nil }
|
func (APIConnServeTest) Stop() error { return nil }
|
||||||
func (APIConnServeTest) EpIndexReverse(string) []*object.Endpoints { return nil }
|
func (APIConnServeTest) EpIndexReverse(string) []*object.Endpoints { return nil }
|
||||||
|
|
|
@ -77,13 +77,19 @@ func (k *Kubernetes) RegisterKubeCache(c *caddy.Controller) {
|
||||||
if k.APIProxy != nil {
|
if k.APIProxy != nil {
|
||||||
k.APIProxy.Run()
|
k.APIProxy.Run()
|
||||||
}
|
}
|
||||||
synced := false
|
|
||||||
for synced == false {
|
|
||||||
synced = k.APIConn.HasSynced()
|
|
||||||
time.Sleep(100 * time.Millisecond)
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
timeout := time.After(5 * time.Second)
|
||||||
|
ticker := time.NewTicker(100 * time.Millisecond)
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-ticker.C:
|
||||||
|
if k.APIConn.HasSynced() {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
case <-timeout:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
c.OnShutdown(func() error {
|
c.OnShutdown(func() error {
|
||||||
|
|
Loading…
Add table
Reference in a new issue