22b955cca5
Reviewed-on: https://go-review.googlesource.com/25150 From-SVN: r238662
170 lines
4.8 KiB
Go
170 lines
4.8 KiB
Go
// Copyright 2015 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
// Package socktest provides utilities for socket testing.
|
|
package socktest
|
|
|
|
import (
|
|
"fmt"
|
|
"sync"
|
|
)
|
|
|
|
// A Switch represents a callpath point switch for socket system
|
|
// calls.
|
|
type Switch struct {
|
|
once sync.Once
|
|
|
|
fmu sync.RWMutex
|
|
fltab map[FilterType]Filter
|
|
|
|
smu sync.RWMutex
|
|
sotab Sockets
|
|
stats stats
|
|
}
|
|
|
|
func (sw *Switch) init() {
|
|
sw.fltab = make(map[FilterType]Filter)
|
|
sw.sotab = make(Sockets)
|
|
sw.stats = make(stats)
|
|
}
|
|
|
|
// Stats returns a list of per-cookie socket statistics.
|
|
func (sw *Switch) Stats() []Stat {
|
|
var st []Stat
|
|
sw.smu.RLock()
|
|
for _, s := range sw.stats {
|
|
ns := *s
|
|
st = append(st, ns)
|
|
}
|
|
sw.smu.RUnlock()
|
|
return st
|
|
}
|
|
|
|
// Sockets returns mappings of socket descriptor to socket status.
|
|
func (sw *Switch) Sockets() Sockets {
|
|
sw.smu.RLock()
|
|
tab := make(Sockets, len(sw.sotab))
|
|
for i, s := range sw.sotab {
|
|
tab[i] = s
|
|
}
|
|
sw.smu.RUnlock()
|
|
return tab
|
|
}
|
|
|
|
// A Cookie represents a 3-tuple of a socket; address family, socket
|
|
// type and protocol number.
|
|
type Cookie uint64
|
|
|
|
// Family returns an address family.
|
|
func (c Cookie) Family() int { return int(c >> 48) }
|
|
|
|
// Type returns a socket type.
|
|
func (c Cookie) Type() int { return int(c << 16 >> 32) }
|
|
|
|
// Protocol returns a protocol number.
|
|
func (c Cookie) Protocol() int { return int(c & 0xff) }
|
|
|
|
func cookie(family, sotype, proto int) Cookie {
|
|
return Cookie(family)<<48 | Cookie(sotype)&0xffffffff<<16 | Cookie(proto)&0xff
|
|
}
|
|
|
|
// A Status represents the status of a socket.
|
|
type Status struct {
|
|
Cookie Cookie
|
|
Err error // error status of socket system call
|
|
SocketErr error // error status of socket by SO_ERROR
|
|
}
|
|
|
|
func (so Status) String() string {
|
|
return fmt.Sprintf("(%s, %s, %s): syscallerr=%v socketerr=%v", familyString(so.Cookie.Family()), typeString(so.Cookie.Type()), protocolString(so.Cookie.Protocol()), so.Err, so.SocketErr)
|
|
}
|
|
|
|
// A Stat represents a per-cookie socket statistics.
|
|
type Stat struct {
|
|
Family int // address family
|
|
Type int // socket type
|
|
Protocol int // protocol number
|
|
|
|
Opened uint64 // number of sockets opened
|
|
Connected uint64 // number of sockets connected
|
|
Listened uint64 // number of sockets listened
|
|
Accepted uint64 // number of sockets accepted
|
|
Closed uint64 // number of sockets closed
|
|
|
|
OpenFailed uint64 // number of sockets open failed
|
|
ConnectFailed uint64 // number of sockets connect failed
|
|
ListenFailed uint64 // number of sockets listen failed
|
|
AcceptFailed uint64 // number of sockets accept failed
|
|
CloseFailed uint64 // number of sockets close failed
|
|
}
|
|
|
|
func (st Stat) String() string {
|
|
return fmt.Sprintf("(%s, %s, %s): opened=%d connected=%d listened=%d accepted=%d closed=%d openfailed=%d connectfailed=%d listenfailed=%d acceptfailed=%d closefailed=%d", familyString(st.Family), typeString(st.Type), protocolString(st.Protocol), st.Opened, st.Connected, st.Listened, st.Accepted, st.Closed, st.OpenFailed, st.ConnectFailed, st.ListenFailed, st.AcceptFailed, st.CloseFailed)
|
|
}
|
|
|
|
type stats map[Cookie]*Stat
|
|
|
|
func (st stats) getLocked(c Cookie) *Stat {
|
|
s, ok := st[c]
|
|
if !ok {
|
|
s = &Stat{Family: c.Family(), Type: c.Type(), Protocol: c.Protocol()}
|
|
st[c] = s
|
|
}
|
|
return s
|
|
}
|
|
|
|
// A FilterType represents a filter type.
|
|
type FilterType int
|
|
|
|
const (
|
|
FilterSocket FilterType = iota // for Socket
|
|
FilterConnect // for Connect or ConnectEx
|
|
FilterListen // for Listen
|
|
FilterAccept // for Accept, Accept4 or AcceptEx
|
|
FilterGetsockoptInt // for GetsockoptInt
|
|
FilterClose // for Close or Closesocket
|
|
)
|
|
|
|
// A Filter represents a socket system call filter.
|
|
//
|
|
// It will only be executed before a system call for a socket that has
|
|
// an entry in internal table.
|
|
// If the filter returns a non-nil error, the execution of system call
|
|
// will be canceled and the system call function returns the non-nil
|
|
// error.
|
|
// It can return a non-nil AfterFilter for filtering after the
|
|
// execution of the system call.
|
|
type Filter func(*Status) (AfterFilter, error)
|
|
|
|
func (f Filter) apply(st *Status) (AfterFilter, error) {
|
|
if f == nil {
|
|
return nil, nil
|
|
}
|
|
return f(st)
|
|
}
|
|
|
|
// An AfterFilter represents a socket system call filter after an
|
|
// execution of a system call.
|
|
//
|
|
// It will only be executed after a system call for a socket that has
|
|
// an entry in internal table.
|
|
// If the filter returns a non-nil error, the system call function
|
|
// returns the non-nil error.
|
|
type AfterFilter func(*Status) error
|
|
|
|
func (f AfterFilter) apply(st *Status) error {
|
|
if f == nil {
|
|
return nil
|
|
}
|
|
return f(st)
|
|
}
|
|
|
|
// Set deploys the socket system call filter f for the filter type t.
|
|
func (sw *Switch) Set(t FilterType, f Filter) {
|
|
sw.once.Do(sw.init)
|
|
sw.fmu.Lock()
|
|
sw.fltab[t] = f
|
|
sw.fmu.Unlock()
|
|
}
|