2014-08-08 21:36:09 +00:00
|
|
|
package acl
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/md5"
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"github.com/hashicorp/golang-lru"
|
|
|
|
)
|
|
|
|
|
|
|
|
// FaultFunc is a function used to fault in the rules for an
|
|
|
|
// ACL given it's ID
|
|
|
|
type FaultFunc func(id string) (string, error)
|
|
|
|
|
2014-08-08 22:25:11 +00:00
|
|
|
// aclEntry allows us to store the ACL with it's policy ID
|
|
|
|
type aclEntry struct {
|
|
|
|
ACL ACL
|
|
|
|
PolicyID string
|
|
|
|
}
|
|
|
|
|
2014-08-08 21:36:09 +00:00
|
|
|
// Cache is used to implement policy and ACL caching
|
|
|
|
type Cache struct {
|
2014-08-09 00:37:13 +00:00
|
|
|
aclCache *lru.Cache // Cache id -> acl
|
2014-08-08 21:36:09 +00:00
|
|
|
faultfn FaultFunc
|
|
|
|
parent ACL
|
2014-08-09 00:37:13 +00:00
|
|
|
policyCache *lru.Cache // Cache policy -> acl
|
|
|
|
ruleCache *lru.Cache // Cache rules -> policy
|
2014-08-08 21:36:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewCache contructs a new policy and ACL cache of a given size
|
|
|
|
func NewCache(size int, parent ACL, faultfn FaultFunc) (*Cache, error) {
|
|
|
|
if size <= 0 {
|
|
|
|
return nil, fmt.Errorf("Must provide positive cache size")
|
|
|
|
}
|
2014-08-09 00:37:13 +00:00
|
|
|
rc, _ := lru.New(size)
|
2014-08-08 21:36:09 +00:00
|
|
|
pc, _ := lru.New(size)
|
|
|
|
ac, _ := lru.New(size)
|
|
|
|
c := &Cache{
|
|
|
|
aclCache: ac,
|
|
|
|
faultfn: faultfn,
|
|
|
|
parent: parent,
|
|
|
|
policyCache: pc,
|
2014-08-09 00:37:13 +00:00
|
|
|
ruleCache: rc,
|
2014-08-08 21:36:09 +00:00
|
|
|
}
|
|
|
|
return c, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetPolicy is used to get a potentially cached policy set.
|
|
|
|
// If not cached, it will be parsed, and then cached.
|
|
|
|
func (c *Cache) GetPolicy(rules string) (*Policy, error) {
|
2014-08-08 22:25:11 +00:00
|
|
|
return c.getPolicy(c.ruleID(rules), rules)
|
|
|
|
}
|
|
|
|
|
|
|
|
// getPolicy is an internal method to get a cached policy,
|
|
|
|
// but it assumes a pre-computed ID
|
|
|
|
func (c *Cache) getPolicy(id, rules string) (*Policy, error) {
|
2014-08-09 00:37:13 +00:00
|
|
|
raw, ok := c.ruleCache.Get(id)
|
2014-08-08 21:36:09 +00:00
|
|
|
if ok {
|
|
|
|
return raw.(*Policy), nil
|
|
|
|
}
|
|
|
|
policy, err := Parse(rules)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2014-08-08 23:51:19 +00:00
|
|
|
policy.ID = id
|
2014-08-09 00:37:13 +00:00
|
|
|
c.ruleCache.Add(id, policy)
|
2014-08-08 21:36:09 +00:00
|
|
|
return policy, nil
|
2014-08-08 22:25:11 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// ruleID is used to generate an ID for a rule
|
|
|
|
func (c *Cache) ruleID(rules string) string {
|
|
|
|
return fmt.Sprintf("%x", md5.Sum([]byte(rules)))
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetACLPolicy is used to get the potentially cached ACL
|
|
|
|
// policy. If not cached, it will be generated and then cached.
|
|
|
|
func (c *Cache) GetACLPolicy(id string) (*Policy, error) {
|
|
|
|
// Check for a cached acl
|
|
|
|
if raw, ok := c.aclCache.Get(id); ok {
|
|
|
|
cached := raw.(aclEntry)
|
2014-08-09 00:37:13 +00:00
|
|
|
if raw, ok := c.ruleCache.Get(cached.PolicyID); ok {
|
2014-08-08 22:25:11 +00:00
|
|
|
return raw.(*Policy), nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fault in the rules
|
|
|
|
rules, err := c.faultfn(id)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get cached
|
|
|
|
return c.GetPolicy(rules)
|
2014-08-08 21:36:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetACL is used to get a potentially cached ACL policy.
|
|
|
|
// If not cached, it will be generated and then cached.
|
|
|
|
func (c *Cache) GetACL(id string) (ACL, error) {
|
|
|
|
// Look for the ACL directly
|
|
|
|
raw, ok := c.aclCache.Get(id)
|
|
|
|
if ok {
|
2014-08-08 22:25:11 +00:00
|
|
|
return raw.(aclEntry).ACL, nil
|
2014-08-08 21:36:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get the rules
|
|
|
|
rules, err := c.faultfn(id)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2014-08-08 22:25:11 +00:00
|
|
|
ruleID := c.ruleID(rules)
|
2014-08-08 21:36:09 +00:00
|
|
|
|
2014-08-09 00:37:13 +00:00
|
|
|
// Check for a compiled ACL
|
|
|
|
var compiled ACL
|
|
|
|
if raw, ok := c.policyCache.Get(ruleID); ok {
|
|
|
|
compiled = raw.(ACL)
|
|
|
|
} else {
|
|
|
|
// Get the policy
|
|
|
|
policy, err := c.getPolicy(ruleID, rules)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2014-08-08 21:36:09 +00:00
|
|
|
|
2014-08-09 00:37:13 +00:00
|
|
|
// Compile the ACL
|
|
|
|
acl, err := New(c.parent, policy)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Cache the compiled ACL
|
|
|
|
c.policyCache.Add(ruleID, acl)
|
|
|
|
compiled = acl
|
2014-08-08 21:36:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Cache and return the ACL
|
2014-08-09 00:37:13 +00:00
|
|
|
c.aclCache.Add(id, aclEntry{compiled, ruleID})
|
|
|
|
return compiled, nil
|
2014-08-08 21:36:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// ClearACL is used to clear the ACL cache if any
|
|
|
|
func (c *Cache) ClearACL(id string) {
|
|
|
|
c.aclCache.Remove(id)
|
|
|
|
}
|