2014-06-06 23:40:48 +00:00
|
|
|
/*
|
2015-05-01 16:19:44 +00:00
|
|
|
Copyright 2014 The Kubernetes Authors All rights reserved.
|
2014-06-06 23:40:48 +00:00
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
2014-06-12 00:01:02 +00:00
|
|
|
|
2014-06-06 23:40:48 +00:00
|
|
|
package util
|
|
|
|
|
|
|
|
import (
|
2015-02-26 20:15:50 +00:00
|
|
|
"bufio"
|
|
|
|
"encoding/hex"
|
2014-06-06 23:40:48 +00:00
|
|
|
"encoding/json"
|
2014-06-14 01:11:32 +00:00
|
|
|
"fmt"
|
2015-02-26 20:15:50 +00:00
|
|
|
"io"
|
2014-12-24 00:03:26 +00:00
|
|
|
"io/ioutil"
|
2015-02-26 20:15:50 +00:00
|
|
|
"net"
|
2015-04-22 14:51:08 +00:00
|
|
|
"net/http"
|
2015-02-26 20:15:50 +00:00
|
|
|
"os"
|
2015-01-17 00:34:47 +00:00
|
|
|
"path"
|
2014-11-24 05:07:36 +00:00
|
|
|
"reflect"
|
2014-09-04 17:55:30 +00:00
|
|
|
"regexp"
|
2014-06-14 01:11:32 +00:00
|
|
|
"runtime"
|
2014-12-24 00:03:26 +00:00
|
|
|
"strconv"
|
2015-01-17 00:34:47 +00:00
|
|
|
"strings"
|
2014-06-06 23:40:48 +00:00
|
|
|
"time"
|
2014-06-25 03:51:57 +00:00
|
|
|
|
|
|
|
"github.com/golang/glog"
|
2015-03-07 08:04:14 +00:00
|
|
|
"github.com/google/gofuzz"
|
2014-06-06 23:40:48 +00:00
|
|
|
)
|
|
|
|
|
2014-06-29 00:52:08 +00:00
|
|
|
// For testing, bypass HandleCrash.
|
|
|
|
var ReallyCrash bool
|
|
|
|
|
2015-01-27 01:23:46 +00:00
|
|
|
// PanicHandlers is a list of functions which will be invoked when a panic happens.
|
|
|
|
var PanicHandlers = []func(interface{}){logPanic}
|
|
|
|
|
2014-07-11 00:32:42 +00:00
|
|
|
// HandleCrash simply catches a crash and logs an error. Meant to be called via defer.
|
2014-06-06 23:40:48 +00:00
|
|
|
func HandleCrash() {
|
2014-06-29 00:52:08 +00:00
|
|
|
if ReallyCrash {
|
|
|
|
return
|
|
|
|
}
|
2015-01-27 01:23:46 +00:00
|
|
|
if r := recover(); r != nil {
|
|
|
|
for _, fn := range PanicHandlers {
|
|
|
|
fn(r)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-06-29 00:52:08 +00:00
|
|
|
|
2015-01-27 01:23:46 +00:00
|
|
|
// logPanic logs the caller tree when a panic occurs.
|
|
|
|
func logPanic(r interface{}) {
|
|
|
|
callers := ""
|
|
|
|
for i := 0; true; i++ {
|
|
|
|
_, file, line, ok := runtime.Caller(i)
|
|
|
|
if !ok {
|
|
|
|
break
|
2014-06-14 01:11:32 +00:00
|
|
|
}
|
2015-01-27 01:23:46 +00:00
|
|
|
callers = callers + fmt.Sprintf("%v:%v\n", file, line)
|
|
|
|
}
|
2015-04-24 13:09:14 +00:00
|
|
|
glog.Errorf("Recovered from panic: %#v (%v)\n%v", r, r, callers)
|
2015-01-27 01:23:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// ErrorHandlers is a list of functions which will be invoked when an unreturnable
|
|
|
|
// error occurs.
|
|
|
|
var ErrorHandlers = []func(error){logError}
|
|
|
|
|
|
|
|
// HandlerError is a method to invoke when a non-user facing piece of code cannot
|
|
|
|
// return an error and needs to indicate it has been ignored. Invoking this method
|
|
|
|
// is preferable to logging the error - the default behavior is to log but the
|
|
|
|
// errors may be sent to a remote server for analysis.
|
|
|
|
func HandleError(err error) {
|
|
|
|
for _, fn := range ErrorHandlers {
|
|
|
|
fn(err)
|
2014-06-06 23:40:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-27 01:23:46 +00:00
|
|
|
// logError prints an error with the call stack of the location it was reported
|
|
|
|
func logError(err error) {
|
|
|
|
glog.ErrorDepth(2, err)
|
|
|
|
}
|
|
|
|
|
2015-01-06 18:05:33 +00:00
|
|
|
// Forever loops forever running f every period. Catches any panics, and keeps going.
|
2015-02-28 00:49:51 +00:00
|
|
|
// Deprecated. Please use Until and pass NeverStop as the stopCh.
|
2014-06-06 23:40:48 +00:00
|
|
|
func Forever(f func(), period time.Duration) {
|
2014-11-28 20:14:49 +00:00
|
|
|
Until(f, period, nil)
|
|
|
|
}
|
|
|
|
|
2015-02-28 00:49:51 +00:00
|
|
|
// NeverStop may be passed to Until to make it never stop.
|
|
|
|
var NeverStop <-chan struct{} = make(chan struct{})
|
|
|
|
|
2015-01-06 18:05:33 +00:00
|
|
|
// Until loops until stop channel is closed, running f every period.
|
2014-11-28 20:14:49 +00:00
|
|
|
// Catches any panics, and keeps going. f may not be invoked if
|
|
|
|
// stop channel is already closed.
|
|
|
|
func Until(f func(), period time.Duration, stopCh <-chan struct{}) {
|
2014-06-06 23:40:48 +00:00
|
|
|
for {
|
2014-11-28 20:14:49 +00:00
|
|
|
select {
|
|
|
|
case <-stopCh:
|
|
|
|
return
|
|
|
|
default:
|
|
|
|
}
|
2014-06-06 23:40:48 +00:00
|
|
|
func() {
|
|
|
|
defer HandleCrash()
|
|
|
|
f()
|
|
|
|
}()
|
|
|
|
time.Sleep(period)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-10 18:09:47 +00:00
|
|
|
// IntOrString is a type that can hold an int or a string. When used in
|
|
|
|
// JSON or YAML marshalling and unmarshalling, it produces or consumes the
|
|
|
|
// inner type. This allows you to have, for example, a JSON field that can
|
|
|
|
// accept a name or number.
|
|
|
|
type IntOrString struct {
|
|
|
|
Kind IntstrKind
|
|
|
|
IntVal int
|
|
|
|
StrVal string
|
|
|
|
}
|
|
|
|
|
2014-07-11 00:32:42 +00:00
|
|
|
// IntstrKind represents the stored type of IntOrString.
|
2014-07-10 18:09:47 +00:00
|
|
|
type IntstrKind int
|
|
|
|
|
|
|
|
const (
|
2014-07-11 00:32:42 +00:00
|
|
|
IntstrInt IntstrKind = iota // The IntOrString holds an int.
|
|
|
|
IntstrString // The IntOrString holds a string.
|
2014-07-10 18:09:47 +00:00
|
|
|
)
|
|
|
|
|
2014-08-21 04:27:19 +00:00
|
|
|
// NewIntOrStringFromInt creates an IntOrString object with an int value.
|
|
|
|
func NewIntOrStringFromInt(val int) IntOrString {
|
2014-08-11 06:26:42 +00:00
|
|
|
return IntOrString{Kind: IntstrInt, IntVal: val}
|
|
|
|
}
|
|
|
|
|
2014-09-02 10:00:28 +00:00
|
|
|
// NewIntOrStringFromString creates an IntOrString object with a string value.
|
2014-08-21 04:27:19 +00:00
|
|
|
func NewIntOrStringFromString(val string) IntOrString {
|
2014-08-11 06:26:42 +00:00
|
|
|
return IntOrString{Kind: IntstrString, StrVal: val}
|
|
|
|
}
|
|
|
|
|
2014-07-11 00:32:42 +00:00
|
|
|
// UnmarshalJSON implements the json.Unmarshaller interface.
|
2014-07-10 18:09:47 +00:00
|
|
|
func (intstr *IntOrString) UnmarshalJSON(value []byte) error {
|
|
|
|
if value[0] == '"' {
|
|
|
|
intstr.Kind = IntstrString
|
|
|
|
return json.Unmarshal(value, &intstr.StrVal)
|
|
|
|
}
|
|
|
|
intstr.Kind = IntstrInt
|
|
|
|
return json.Unmarshal(value, &intstr.IntVal)
|
|
|
|
}
|
|
|
|
|
2015-01-03 03:10:03 +00:00
|
|
|
// String returns the string value, or Itoa's the int value.
|
|
|
|
func (intstr *IntOrString) String() string {
|
|
|
|
if intstr.Kind == IntstrString {
|
|
|
|
return intstr.StrVal
|
|
|
|
}
|
|
|
|
return strconv.Itoa(intstr.IntVal)
|
|
|
|
}
|
|
|
|
|
2014-07-11 00:32:42 +00:00
|
|
|
// MarshalJSON implements the json.Marshaller interface.
|
2014-07-10 18:09:47 +00:00
|
|
|
func (intstr IntOrString) MarshalJSON() ([]byte, error) {
|
|
|
|
switch intstr.Kind {
|
|
|
|
case IntstrInt:
|
|
|
|
return json.Marshal(intstr.IntVal)
|
|
|
|
case IntstrString:
|
|
|
|
return json.Marshal(intstr.StrVal)
|
|
|
|
default:
|
2014-07-11 18:22:22 +00:00
|
|
|
return []byte{}, fmt.Errorf("impossible IntOrString.Kind")
|
2014-07-10 18:09:47 +00:00
|
|
|
}
|
|
|
|
}
|
2014-07-26 00:58:36 +00:00
|
|
|
|
2015-03-07 08:04:14 +00:00
|
|
|
func (intstr *IntOrString) Fuzz(c fuzz.Continue) {
|
|
|
|
if c.RandBool() {
|
|
|
|
intstr.Kind = IntstrInt
|
|
|
|
c.Fuzz(&intstr.IntVal)
|
|
|
|
intstr.StrVal = ""
|
|
|
|
} else {
|
|
|
|
intstr.Kind = IntstrString
|
|
|
|
intstr.IntVal = 0
|
|
|
|
c.Fuzz(&intstr.StrVal)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-04 17:55:30 +00:00
|
|
|
// Takes a list of strings and compiles them into a list of regular expressions
|
2014-10-08 13:23:25 +00:00
|
|
|
func CompileRegexps(regexpStrings []string) ([]*regexp.Regexp, error) {
|
2014-09-04 17:55:30 +00:00
|
|
|
regexps := []*regexp.Regexp{}
|
|
|
|
for _, regexpStr := range regexpStrings {
|
|
|
|
r, err := regexp.Compile(regexpStr)
|
|
|
|
if err != nil {
|
|
|
|
return []*regexp.Regexp{}, err
|
|
|
|
}
|
|
|
|
regexps = append(regexps, r)
|
|
|
|
}
|
|
|
|
return regexps, nil
|
|
|
|
}
|
2014-12-24 00:03:26 +00:00
|
|
|
|
2015-06-24 18:10:10 +00:00
|
|
|
// Detects if using systemd as the init system
|
|
|
|
// Please note that simply reading /proc/1/cmdline can be misleading because
|
|
|
|
// some installation of various init programs can automatically make /sbin/init
|
|
|
|
// a symlink or even a renamed version of their main program.
|
|
|
|
// TODO(dchen1107): realiably detects the init system using on the system:
|
|
|
|
// systemd, upstart, initd, etc.
|
|
|
|
func UsingSystemdInitSystem() bool {
|
|
|
|
if _, err := os.Stat("/run/systemd/system"); err != nil {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2015-02-20 01:16:31 +00:00
|
|
|
// Writes 'value' to /proc/<pid>/oom_score_adj. PID = 0 means self
|
|
|
|
func ApplyOomScoreAdj(pid int, value int) error {
|
2014-12-24 00:03:26 +00:00
|
|
|
if value < -1000 || value > 1000 {
|
2015-01-07 08:40:16 +00:00
|
|
|
return fmt.Errorf("invalid value(%d) specified for oom_score_adj. Values must be within the range [-1000, 1000]", value)
|
2014-12-24 00:03:26 +00:00
|
|
|
}
|
2015-02-20 01:16:31 +00:00
|
|
|
if pid < 0 {
|
|
|
|
return fmt.Errorf("invalid PID %d specified for oom_score_adj", pid)
|
|
|
|
}
|
|
|
|
|
|
|
|
var pidStr string
|
|
|
|
if pid == 0 {
|
|
|
|
pidStr = "self"
|
|
|
|
} else {
|
|
|
|
pidStr = strconv.Itoa(pid)
|
|
|
|
}
|
2014-12-24 00:03:26 +00:00
|
|
|
|
2015-05-14 21:40:20 +00:00
|
|
|
oom_value, err := ioutil.ReadFile(path.Join("/proc", pidStr, "oom_score_adj"))
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("failed to read oom_score_adj: %v", err)
|
|
|
|
} else if string(oom_value) != strconv.Itoa(value) {
|
|
|
|
if err := ioutil.WriteFile(path.Join("/proc", pidStr, "oom_score_adj"), []byte(strconv.Itoa(value)), 0700); err != nil {
|
|
|
|
return fmt.Errorf("failed to set oom_score_adj to %d: %v", value, err)
|
|
|
|
}
|
2014-12-24 00:03:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2014-11-24 05:07:36 +00:00
|
|
|
|
|
|
|
// Tests whether all pointer fields in a struct are nil. This is useful when,
|
|
|
|
// for example, an API struct is handled by plugins which need to distinguish
|
|
|
|
// "no plugin accepted this spec" from "this spec is empty".
|
|
|
|
//
|
|
|
|
// This function is only valid for structs and pointers to structs. Any other
|
|
|
|
// type will cause a panic. Passing a typed nil pointer will return true.
|
|
|
|
func AllPtrFieldsNil(obj interface{}) bool {
|
|
|
|
v := reflect.ValueOf(obj)
|
|
|
|
if !v.IsValid() {
|
|
|
|
panic(fmt.Sprintf("reflect.ValueOf() produced a non-valid Value for %#v", obj))
|
|
|
|
}
|
|
|
|
if v.Kind() == reflect.Ptr {
|
|
|
|
if v.IsNil() {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
v = v.Elem()
|
|
|
|
}
|
|
|
|
for i := 0; i < v.NumField(); i++ {
|
|
|
|
if v.Field(i).Kind() == reflect.Ptr && !v.Field(i).IsNil() {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
2015-01-17 00:34:47 +00:00
|
|
|
|
|
|
|
// Splits a fully qualified name and returns its namespace and name.
|
|
|
|
// Assumes that the input 'str' has been validated.
|
|
|
|
func SplitQualifiedName(str string) (string, string) {
|
|
|
|
parts := strings.Split(str, "/")
|
|
|
|
if len(parts) < 2 {
|
|
|
|
return "", str
|
|
|
|
}
|
|
|
|
|
|
|
|
return parts[0], parts[1]
|
|
|
|
}
|
|
|
|
|
|
|
|
// Joins 'namespace' and 'name' and returns a fully qualified name
|
|
|
|
// Assumes that the input is valid.
|
|
|
|
func JoinQualifiedName(namespace, name string) string {
|
|
|
|
return path.Join(namespace, name)
|
|
|
|
}
|
2015-02-26 20:15:50 +00:00
|
|
|
|
|
|
|
type Route struct {
|
|
|
|
Interface string
|
|
|
|
Destination net.IP
|
|
|
|
Gateway net.IP
|
|
|
|
// TODO: add more fields here if needed
|
|
|
|
}
|
|
|
|
|
|
|
|
func getRoutes(input io.Reader) ([]Route, error) {
|
|
|
|
routes := []Route{}
|
|
|
|
if input == nil {
|
|
|
|
return nil, fmt.Errorf("input is nil")
|
|
|
|
}
|
|
|
|
scanner := bufio.NewReader(input)
|
|
|
|
for {
|
|
|
|
line, err := scanner.ReadString('\n')
|
|
|
|
if err == io.EOF {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
//ignore the headers in the route info
|
|
|
|
if strings.HasPrefix(line, "Iface") {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
fields := strings.Fields(line)
|
|
|
|
routes = append(routes, Route{})
|
|
|
|
route := &routes[len(routes)-1]
|
|
|
|
route.Interface = fields[0]
|
|
|
|
ip, err := parseIP(fields[1])
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
route.Destination = ip
|
|
|
|
ip, err = parseIP(fields[2])
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
route.Gateway = ip
|
|
|
|
}
|
|
|
|
return routes, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseIP(str string) (net.IP, error) {
|
|
|
|
if str == "" {
|
|
|
|
return nil, fmt.Errorf("input is nil")
|
|
|
|
}
|
|
|
|
bytes, err := hex.DecodeString(str)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
//TODO add ipv6 support
|
|
|
|
if len(bytes) != net.IPv4len {
|
|
|
|
return nil, fmt.Errorf("only IPv4 is supported")
|
|
|
|
}
|
|
|
|
bytes[0], bytes[1], bytes[2], bytes[3] = bytes[3], bytes[2], bytes[1], bytes[0]
|
|
|
|
return net.IP(bytes), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func isInterfaceUp(intf *net.Interface) bool {
|
|
|
|
if intf == nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if intf.Flags&net.FlagUp != 0 {
|
|
|
|
glog.V(4).Infof("Interface %v is up", intf.Name)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
//getFinalIP method receives all the IP addrs of a Interface
|
|
|
|
//and returns a nil if the address is Loopback , Ipv6 or nil.
|
|
|
|
//It returns a valid IPv4 if an Ipv4 address is found in the array.
|
|
|
|
func getFinalIP(addrs []net.Addr) (net.IP, error) {
|
|
|
|
if len(addrs) > 0 {
|
|
|
|
for i := range addrs {
|
|
|
|
glog.V(4).Infof("Checking addr %s.", addrs[i].String())
|
|
|
|
ip, _, err := net.ParseCIDR(addrs[i].String())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
//Only IPv4
|
|
|
|
//TODO : add IPv6 support
|
|
|
|
if ip.To4() != nil {
|
|
|
|
if !ip.IsLoopback() {
|
|
|
|
glog.V(4).Infof("IP found %v", ip)
|
|
|
|
return ip, nil
|
|
|
|
} else {
|
|
|
|
glog.V(4).Infof("Loopback found %v", ip)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
glog.V(4).Infof("%v is not a valid IPv4 address", ip)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getIPFromInterface(intfName string, nw networkInterfacer) (net.IP, error) {
|
|
|
|
intf, err := nw.InterfaceByName(intfName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if isInterfaceUp(intf) {
|
|
|
|
addrs, err := nw.Addrs(intf)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
glog.V(4).Infof("Interface %q has %d addresses :%v.", intfName, len(addrs), addrs)
|
|
|
|
finalIP, err := getFinalIP(addrs)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if finalIP != nil {
|
|
|
|
glog.V(4).Infof("valid IPv4 address for interface %q found as %v.", intfName, finalIP)
|
|
|
|
return finalIP, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2015-03-12 19:33:06 +00:00
|
|
|
func flagsSet(flags net.Flags, test net.Flags) bool {
|
|
|
|
return flags&test != 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func flagsClear(flags net.Flags, test net.Flags) bool {
|
|
|
|
return flags&test == 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func chooseHostInterfaceNativeGo() (net.IP, error) {
|
|
|
|
intfs, err := net.Interfaces()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
i := 0
|
2015-05-04 18:44:58 +00:00
|
|
|
var ip net.IP
|
2015-03-12 19:33:06 +00:00
|
|
|
for i = range intfs {
|
|
|
|
if flagsSet(intfs[i].Flags, net.FlagUp) && flagsClear(intfs[i].Flags, net.FlagLoopback|net.FlagPointToPoint) {
|
|
|
|
addrs, err := intfs[i].Addrs()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if len(addrs) > 0 {
|
2015-05-04 18:44:58 +00:00
|
|
|
for _, addr := range addrs {
|
|
|
|
if addrIP, _, err := net.ParseCIDR(addr.String()); err == nil {
|
|
|
|
if addrIP.To4() != nil {
|
|
|
|
ip = addrIP.To4()
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ip != nil {
|
|
|
|
// This interface should suffice.
|
|
|
|
break
|
|
|
|
}
|
2015-03-12 19:33:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-05-04 18:44:58 +00:00
|
|
|
if ip == nil {
|
|
|
|
return nil, fmt.Errorf("no acceptable interface from host")
|
2015-03-12 19:33:06 +00:00
|
|
|
}
|
2015-05-23 20:41:11 +00:00
|
|
|
glog.V(4).Infof("Choosing interface %s (IP %v) as default", intfs[i].Name, ip)
|
2015-03-12 19:33:06 +00:00
|
|
|
return ip, nil
|
|
|
|
}
|
|
|
|
|
2015-02-26 20:15:50 +00:00
|
|
|
//ChooseHostInterface is a method used fetch an IP for a daemon.
|
|
|
|
//It uses data from /proc/net/route file.
|
|
|
|
//For a node with no internet connection ,it returns error
|
|
|
|
//For a multi n/w interface node it returns the IP of the interface with gateway on it.
|
|
|
|
func ChooseHostInterface() (net.IP, error) {
|
|
|
|
inFile, err := os.Open("/proc/net/route")
|
|
|
|
if err != nil {
|
2015-03-12 19:33:06 +00:00
|
|
|
if os.IsNotExist(err) {
|
|
|
|
return chooseHostInterfaceNativeGo()
|
|
|
|
}
|
2015-02-26 20:15:50 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer inFile.Close()
|
|
|
|
var nw networkInterfacer = networkInterface{}
|
|
|
|
return chooseHostInterfaceFromRoute(inFile, nw)
|
|
|
|
}
|
|
|
|
|
|
|
|
type networkInterfacer interface {
|
|
|
|
InterfaceByName(intfName string) (*net.Interface, error)
|
|
|
|
Addrs(intf *net.Interface) ([]net.Addr, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
type networkInterface struct{}
|
|
|
|
|
|
|
|
func (_ networkInterface) InterfaceByName(intfName string) (*net.Interface, error) {
|
|
|
|
intf, err := net.InterfaceByName(intfName)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return intf, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (_ networkInterface) Addrs(intf *net.Interface) ([]net.Addr, error) {
|
|
|
|
addrs, err := intf.Addrs()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return addrs, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func chooseHostInterfaceFromRoute(inFile io.Reader, nw networkInterfacer) (net.IP, error) {
|
|
|
|
routes, err := getRoutes(inFile)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
zero := net.IP{0, 0, 0, 0}
|
|
|
|
var finalIP net.IP
|
|
|
|
for i := range routes {
|
|
|
|
//find interface with gateway
|
|
|
|
if routes[i].Destination.Equal(zero) {
|
|
|
|
glog.V(4).Infof("Default route transits interface %q", routes[i].Interface)
|
|
|
|
finalIP, err := getIPFromInterface(routes[i].Interface, nw)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if finalIP != nil {
|
|
|
|
glog.V(4).Infof("Choosing IP %v ", finalIP)
|
|
|
|
return finalIP, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
glog.V(4).Infof("No valid IP found")
|
|
|
|
if finalIP == nil {
|
|
|
|
return nil, fmt.Errorf("Unable to select an IP.")
|
|
|
|
}
|
|
|
|
return nil, nil
|
|
|
|
}
|
2015-04-22 14:51:08 +00:00
|
|
|
|
|
|
|
func GetClient(req *http.Request) string {
|
|
|
|
if userAgent, ok := req.Header["User-Agent"]; ok {
|
|
|
|
if len(userAgent) > 0 {
|
|
|
|
return userAgent[0]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return "unknown"
|
|
|
|
}
|
2015-05-29 20:34:02 +00:00
|
|
|
|
|
|
|
func ShortenString(str string, n int) string {
|
|
|
|
if len(str) <= n {
|
|
|
|
return str
|
|
|
|
} else {
|
|
|
|
return str[:n]
|
|
|
|
}
|
2015-06-05 21:49:26 +00:00
|
|
|
}
|
2015-05-28 18:45:08 +00:00
|
|
|
|
|
|
|
func FileExists(filename string) (bool, error) {
|
2015-06-05 21:49:26 +00:00
|
|
|
if _, err := os.Stat(filename); os.IsNotExist(err) {
|
|
|
|
return false, nil
|
|
|
|
} else if err != nil {
|
|
|
|
return false, err
|
2015-05-28 18:45:08 +00:00
|
|
|
}
|
|
|
|
return true, nil
|
2015-05-29 20:34:02 +00:00
|
|
|
}
|