2014-12-21 02:49:10 +00:00
|
|
|
/*
|
|
|
|
Copyright 2014 Google Inc. All rights reserved.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package master
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/cloudprovider"
|
2014-12-22 19:54:32 +00:00
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/util"
|
2014-12-21 02:49:10 +00:00
|
|
|
|
|
|
|
"github.com/golang/glog"
|
|
|
|
)
|
|
|
|
|
|
|
|
type ipCacheEntry struct {
|
|
|
|
ip string
|
|
|
|
lastUpdate time.Time
|
|
|
|
}
|
|
|
|
|
|
|
|
type ipCache struct {
|
2014-12-22 19:54:32 +00:00
|
|
|
clock util.Clock
|
2014-12-21 02:49:10 +00:00
|
|
|
cloudProvider cloudprovider.Interface
|
|
|
|
cache map[string]ipCacheEntry
|
|
|
|
lock sync.Mutex
|
2014-12-22 20:00:44 +00:00
|
|
|
ttl time.Duration
|
2014-12-21 02:49:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewIPCache makes a new ip caching layer, which will get IP addresses from cp,
|
|
|
|
// and use clock for deciding when to re-get an IP address.
|
|
|
|
// Thread-safe.
|
2014-12-22 20:00:44 +00:00
|
|
|
//
|
|
|
|
// TODO: when we switch to go1.4, this class would be a good candidate for something
|
|
|
|
// that could be produced from a template and a type via `go generate`.
|
|
|
|
func NewIPCache(cp cloudprovider.Interface, clock util.Clock, ttl time.Duration) *ipCache {
|
2014-12-21 02:49:10 +00:00
|
|
|
return &ipCache{
|
|
|
|
clock: clock,
|
|
|
|
cloudProvider: cp,
|
|
|
|
cache: map[string]ipCacheEntry{},
|
2014-12-22 20:00:44 +00:00
|
|
|
ttl: ttl,
|
2014-12-21 02:49:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetInstanceIP returns the IP address of host, from the cache
|
|
|
|
// if possible, otherwise it asks the cloud provider.
|
|
|
|
func (c *ipCache) GetInstanceIP(host string) string {
|
|
|
|
c.lock.Lock()
|
|
|
|
defer c.lock.Unlock()
|
|
|
|
data, ok := c.cache[host]
|
|
|
|
now := c.clock.Now()
|
|
|
|
|
2014-12-22 20:00:44 +00:00
|
|
|
if !ok || now.Sub(data.lastUpdate) > c.ttl {
|
2014-12-21 02:49:10 +00:00
|
|
|
ip := getInstanceIPFromCloud(c.cloudProvider, host)
|
|
|
|
data = ipCacheEntry{
|
|
|
|
ip: ip,
|
|
|
|
lastUpdate: now,
|
|
|
|
}
|
|
|
|
c.cache[host] = data
|
|
|
|
}
|
|
|
|
return data.ip
|
|
|
|
}
|
|
|
|
|
|
|
|
func getInstanceIPFromCloud(cloud cloudprovider.Interface, host string) string {
|
|
|
|
if cloud == nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
instances, ok := cloud.Instances()
|
|
|
|
if instances == nil || !ok {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
addr, err := instances.IPAddress(host)
|
|
|
|
if err != nil {
|
|
|
|
glog.Errorf("Error getting instance IP for %q: %v", host, err)
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return addr.String()
|
|
|
|
}
|