mirror of
https://github.com/token2/snapd.git
synced 2026-03-13 11:15:47 -07:00
Now that we have app sets in the interfaces repo, keep a pointer to them in ConnectedPlug/Slot types. Use this to build label expressions in the interfaces. * many: add a pointer to a SnapAppSet into Connected(Plug|Slot) to that interfaces can build a complete label expression, including component hooks * interfaces: update doc comments on ConnectedPlug/Slot.AppSet * interfaces: remove TODO that has been addressed * interfaces: use app set pointer for instance name check * snap: add Runnable type that represents the runnable parts of a snap * interfaces, o/ifacestate: use snap.Runnable rather than interfaces.Runnable * interfaces, i/builtin, o/ifacestate: panic on failed invariant check in NewConnectedPlug/Slot * interfaces: add methods to app set for getting runnables that can connect to plug/slot * interfaces: build label expressions using runnables * interfaces: doc comment for SlotRunnables * interfaces: implement Slot/PlugRunnables with shared helper * interfaces: log and skip security tags that do not match expected pattern * snap, interfaces: move runnable constructors to methods on AppInfo and HookInfo * interfaces: refactor to allow labelExpr to operate directly on a ConnectedPlug/Slot * snap: move around Runnable methods
490 lines
15 KiB
Go
490 lines
15 KiB
Go
// -*- Mode: Go; indent-tabs-mode: t -*-
|
|
|
|
/*
|
|
* Copyright (C) 2017 Canonical Ltd
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 3 as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
package interfaces_test
|
|
|
|
import (
|
|
"errors"
|
|
|
|
. "gopkg.in/check.v1"
|
|
|
|
"github.com/snapcore/snapd/dirs"
|
|
"github.com/snapcore/snapd/interfaces"
|
|
"github.com/snapcore/snapd/interfaces/ifacetest"
|
|
"github.com/snapcore/snapd/interfaces/utils"
|
|
"github.com/snapcore/snapd/snap"
|
|
"github.com/snapcore/snapd/testutil"
|
|
)
|
|
|
|
type connSuite struct {
|
|
testutil.BaseTest
|
|
plug *snap.PlugInfo
|
|
plugAppSet *interfaces.SnapAppSet
|
|
slot *snap.SlotInfo
|
|
slotAppSet *interfaces.SnapAppSet
|
|
}
|
|
|
|
var _ = Suite(&connSuite{})
|
|
|
|
func (s *connSuite) SetUpTest(c *C) {
|
|
s.BaseTest.SetUpTest(c)
|
|
|
|
dirs.SetRootDir(c.MkDir())
|
|
s.AddCleanup(func() { dirs.SetRootDir("") })
|
|
|
|
s.BaseTest.AddCleanup(snap.MockSanitizePlugsSlots(func(snapInfo *snap.Info) {}))
|
|
s.plugAppSet = ifacetest.MockInfoAndAppSet(c, `
|
|
name: consumer
|
|
version: 0
|
|
apps:
|
|
app:
|
|
plugs:
|
|
plug:
|
|
interface: interface
|
|
attr: value
|
|
complex:
|
|
c: d
|
|
`, nil, nil)
|
|
s.plug = s.plugAppSet.Info().Plugs["plug"]
|
|
s.slotAppSet = ifacetest.MockInfoAndAppSet(c, `
|
|
name: producer
|
|
version: 0
|
|
apps:
|
|
app:
|
|
slots:
|
|
slot:
|
|
interface: interface
|
|
attr: value
|
|
number: 100
|
|
complex:
|
|
a: b
|
|
`, nil, nil)
|
|
s.slot = s.slotAppSet.Info().Slots["slot"]
|
|
}
|
|
|
|
func (s *connSuite) TearDownTest(c *C) {
|
|
s.BaseTest.TearDownTest(c)
|
|
}
|
|
|
|
// Make sure ConnectedPlug,ConnectedSlot, PlugInfo, SlotInfo implement Attrer.
|
|
var _ interfaces.Attrer = (*interfaces.ConnectedPlug)(nil)
|
|
var _ interfaces.Attrer = (*interfaces.ConnectedSlot)(nil)
|
|
var _ interfaces.Attrer = (*snap.PlugInfo)(nil)
|
|
var _ interfaces.Attrer = (*snap.SlotInfo)(nil)
|
|
|
|
func (s *connSuite) TestStaticSlotAttrs(c *C) {
|
|
slot := interfaces.NewConnectedSlot(s.slot, s.slotAppSet, nil, nil)
|
|
c.Assert(slot, NotNil)
|
|
|
|
var val string
|
|
var intVal int
|
|
c.Assert(slot.StaticAttr("unknown", &val), ErrorMatches, `snap "producer" does not have attribute "unknown" for interface "interface"`)
|
|
|
|
attrs := slot.StaticAttrs()
|
|
c.Assert(attrs, DeepEquals, map[string]interface{}{
|
|
"attr": "value",
|
|
"number": int64(100),
|
|
"complex": map[string]interface{}{"a": "b"},
|
|
})
|
|
slot.StaticAttr("attr", &val)
|
|
c.Assert(val, Equals, "value")
|
|
|
|
c.Assert(slot.StaticAttr("unknown", &val), ErrorMatches, `snap "producer" does not have attribute "unknown" for interface "interface"`)
|
|
c.Check(slot.StaticAttr("attr", &intVal), ErrorMatches, `snap "producer" has interface "interface" with invalid value type string for "attr" attribute: \*int`)
|
|
c.Check(slot.StaticAttr("attr", val), ErrorMatches, `internal error: cannot get "attr" attribute of interface "interface" with non-pointer value`)
|
|
|
|
// static attributes passed via args take precedence over slot.Attrs
|
|
slot2 := interfaces.NewConnectedSlot(s.slot, s.slotAppSet, map[string]interface{}{"foo": "bar"}, nil)
|
|
slot2.StaticAttr("foo", &val)
|
|
c.Assert(val, Equals, "bar")
|
|
}
|
|
|
|
func (s *connSuite) TestSlotRef(c *C) {
|
|
slot := interfaces.NewConnectedSlot(s.slot, s.slotAppSet, nil, nil)
|
|
c.Assert(slot, NotNil)
|
|
c.Assert(*slot.Ref(), DeepEquals, interfaces.SlotRef{Snap: "producer", Name: "slot"})
|
|
}
|
|
|
|
func (s *connSuite) TestPlugRef(c *C) {
|
|
plug := interfaces.NewConnectedPlug(s.plug, s.plugAppSet, nil, nil)
|
|
c.Assert(plug, NotNil)
|
|
c.Assert(*plug.Ref(), DeepEquals, interfaces.PlugRef{Snap: "consumer", Name: "plug"})
|
|
}
|
|
|
|
func (s *connSuite) TestStaticPlugAttrs(c *C) {
|
|
plug := interfaces.NewConnectedPlug(s.plug, s.plugAppSet, nil, nil)
|
|
c.Assert(plug, NotNil)
|
|
|
|
var val string
|
|
var intVal int
|
|
c.Assert(plug.StaticAttr("unknown", &val), ErrorMatches, `snap "consumer" does not have attribute "unknown" for interface "interface"`)
|
|
|
|
attrs := plug.StaticAttrs()
|
|
c.Assert(attrs, DeepEquals, map[string]interface{}{
|
|
"attr": "value",
|
|
"complex": map[string]interface{}{"c": "d"},
|
|
})
|
|
plug.StaticAttr("attr", &val)
|
|
c.Assert(val, Equals, "value")
|
|
|
|
c.Assert(plug.StaticAttr("unknown", &val), ErrorMatches, `snap "consumer" does not have attribute "unknown" for interface "interface"`)
|
|
c.Check(plug.StaticAttr("attr", &intVal), ErrorMatches, `snap "consumer" has interface "interface" with invalid value type string for "attr" attribute: \*int`)
|
|
c.Check(plug.StaticAttr("attr", val), ErrorMatches, `internal error: cannot get "attr" attribute of interface "interface" with non-pointer value`)
|
|
|
|
// static attributes passed via args take precedence over plug.Attrs
|
|
plug2 := interfaces.NewConnectedPlug(s.plug, s.plugAppSet, map[string]interface{}{"foo": "bar"}, nil)
|
|
plug2.StaticAttr("foo", &val)
|
|
c.Assert(val, Equals, "bar")
|
|
}
|
|
|
|
func (s *connSuite) TestDynamicSlotAttrs(c *C) {
|
|
attrs := map[string]interface{}{
|
|
"foo": "bar",
|
|
"number": int(100),
|
|
}
|
|
slot := interfaces.NewConnectedSlot(s.slot, s.slotAppSet, nil, attrs)
|
|
c.Assert(slot, NotNil)
|
|
|
|
var strVal string
|
|
var intVal int64
|
|
var mapVal map[string]interface{}
|
|
|
|
c.Assert(slot.Attr("foo", &strVal), IsNil)
|
|
c.Assert(strVal, Equals, "bar")
|
|
|
|
c.Assert(slot.Attr("attr", &strVal), IsNil)
|
|
c.Assert(strVal, Equals, "value")
|
|
|
|
c.Assert(slot.Attr("number", &intVal), IsNil)
|
|
c.Assert(intVal, Equals, int64(100))
|
|
|
|
err := slot.SetAttr("other", map[string]interface{}{"number-two": int(222)})
|
|
c.Assert(err, IsNil)
|
|
c.Assert(slot.Attr("other", &mapVal), IsNil)
|
|
num := mapVal["number-two"]
|
|
c.Assert(num, Equals, int64(222))
|
|
|
|
c.Check(slot.Attr("unknown", &strVal), ErrorMatches, `snap "producer" does not have attribute "unknown" for interface "interface"`)
|
|
c.Check(slot.Attr("foo", &intVal), ErrorMatches, `snap "producer" has interface "interface" with invalid value type string for "foo" attribute: \*int64`)
|
|
c.Check(slot.Attr("number", intVal), ErrorMatches, `internal error: cannot get "number" attribute of interface "interface" with non-pointer value`)
|
|
}
|
|
|
|
func (s *connSuite) TestDottedPathSlot(c *C) {
|
|
attrs := map[string]interface{}{
|
|
"nested": map[string]interface{}{
|
|
"foo": "bar",
|
|
},
|
|
}
|
|
var strVal string
|
|
|
|
slot := interfaces.NewConnectedSlot(s.slot, s.slotAppSet, nil, attrs)
|
|
c.Assert(slot, NotNil)
|
|
|
|
// static attribute complex.a
|
|
c.Assert(slot.Attr("complex.a", &strVal), IsNil)
|
|
c.Assert(strVal, Equals, "b")
|
|
|
|
v, ok := slot.Lookup("complex.a")
|
|
c.Assert(ok, Equals, true)
|
|
c.Assert(v, Equals, "b")
|
|
|
|
// dynamic attribute nested.foo
|
|
c.Assert(slot.Attr("nested.foo", &strVal), IsNil)
|
|
c.Assert(strVal, Equals, "bar")
|
|
|
|
v, ok = slot.Lookup("nested.foo")
|
|
c.Assert(ok, Equals, true)
|
|
c.Assert(v, Equals, "bar")
|
|
|
|
_, ok = slot.Lookup("..")
|
|
c.Assert(ok, Equals, false)
|
|
}
|
|
|
|
func (s *connSuite) TestDottedPathPlug(c *C) {
|
|
attrs := map[string]interface{}{
|
|
"a": "b",
|
|
"nested": map[string]interface{}{
|
|
"foo": "bar",
|
|
},
|
|
}
|
|
var strVal string
|
|
|
|
plug := interfaces.NewConnectedPlug(s.plug, s.plugAppSet, nil, attrs)
|
|
c.Assert(plug, NotNil)
|
|
|
|
v, ok := plug.Lookup("a")
|
|
c.Assert(ok, Equals, true)
|
|
c.Assert(v, Equals, "b")
|
|
|
|
// static attribute complex.c
|
|
c.Assert(plug.Attr("complex.c", &strVal), IsNil)
|
|
c.Assert(strVal, Equals, "d")
|
|
|
|
v, ok = plug.Lookup("complex.c")
|
|
c.Assert(ok, Equals, true)
|
|
c.Assert(v, Equals, "d")
|
|
|
|
// dynamic attribute nested.foo
|
|
c.Assert(plug.Attr("nested.foo", &strVal), IsNil)
|
|
c.Assert(strVal, Equals, "bar")
|
|
|
|
v, ok = plug.Lookup("nested.foo")
|
|
c.Assert(ok, Equals, true)
|
|
c.Assert(v, Equals, "bar")
|
|
|
|
_, ok = plug.Lookup("nested.x")
|
|
c.Assert(ok, Equals, false)
|
|
|
|
_, ok = plug.Lookup("nested.foo.y")
|
|
c.Assert(ok, Equals, false)
|
|
|
|
_, ok = plug.Lookup("..")
|
|
c.Assert(ok, Equals, false)
|
|
}
|
|
|
|
func (s *connSuite) TestLookupFailure(c *C) {
|
|
attrs := map[string]interface{}{}
|
|
|
|
slot := interfaces.NewConnectedSlot(s.slot, s.slotAppSet, nil, attrs)
|
|
c.Assert(slot, NotNil)
|
|
plug := interfaces.NewConnectedPlug(s.plug, s.plugAppSet, nil, attrs)
|
|
c.Assert(plug, NotNil)
|
|
|
|
v, ok := slot.Lookup("a")
|
|
c.Assert(ok, Equals, false)
|
|
c.Assert(v, IsNil)
|
|
|
|
v, ok = plug.Lookup("a")
|
|
c.Assert(ok, Equals, false)
|
|
c.Assert(v, IsNil)
|
|
}
|
|
|
|
func (s *connSuite) TestDynamicPlugAttrs(c *C) {
|
|
attrs := map[string]interface{}{
|
|
"foo": "bar",
|
|
"number": int(100),
|
|
}
|
|
plug := interfaces.NewConnectedPlug(s.plug, s.plugAppSet, nil, attrs)
|
|
c.Assert(plug, NotNil)
|
|
|
|
var strVal string
|
|
var intVal int64
|
|
var mapVal map[string]interface{}
|
|
|
|
c.Assert(plug.Attr("foo", &strVal), IsNil)
|
|
c.Assert(strVal, Equals, "bar")
|
|
|
|
c.Assert(plug.Attr("attr", &strVal), IsNil)
|
|
c.Assert(strVal, Equals, "value")
|
|
|
|
c.Assert(plug.Attr("number", &intVal), IsNil)
|
|
c.Assert(intVal, Equals, int64(100))
|
|
|
|
err := plug.SetAttr("other", map[string]interface{}{"number-two": int(222)})
|
|
c.Assert(err, IsNil)
|
|
c.Assert(plug.Attr("other", &mapVal), IsNil)
|
|
num := mapVal["number-two"]
|
|
c.Assert(num, Equals, int64(222))
|
|
|
|
c.Check(plug.Attr("unknown", &strVal), ErrorMatches, `snap "consumer" does not have attribute "unknown" for interface "interface"`)
|
|
c.Check(plug.Attr("foo", &intVal), ErrorMatches, `snap "consumer" has interface "interface" with invalid value type string for "foo" attribute: \*int64`)
|
|
c.Check(plug.Attr("number", intVal), ErrorMatches, `internal error: cannot get "number" attribute of interface "interface" with non-pointer value`)
|
|
}
|
|
|
|
func (s *connSuite) TestOverwriteStaticAttrError(c *C) {
|
|
attrs := map[string]interface{}{}
|
|
|
|
plug := interfaces.NewConnectedPlug(s.plug, s.plugAppSet, nil, attrs)
|
|
c.Assert(plug, NotNil)
|
|
slot := interfaces.NewConnectedSlot(s.slot, s.slotAppSet, nil, attrs)
|
|
c.Assert(slot, NotNil)
|
|
|
|
err := plug.SetAttr("attr", "overwrite")
|
|
c.Assert(err, NotNil)
|
|
c.Assert(err, ErrorMatches, `cannot change attribute "attr" as it was statically specified in the snap details`)
|
|
|
|
err = slot.SetAttr("attr", "overwrite")
|
|
c.Assert(err, NotNil)
|
|
c.Assert(err, ErrorMatches, `cannot change attribute "attr" as it was statically specified in the snap details`)
|
|
}
|
|
|
|
func (s *connSuite) TestCopyAttributes(c *C) {
|
|
orig := map[string]interface{}{
|
|
"a": "A",
|
|
"b": true,
|
|
"c": int(100),
|
|
"d": []interface{}{"x", "y", true},
|
|
"e": map[string]interface{}{"e1": "E1"},
|
|
}
|
|
|
|
cpy := utils.CopyAttributes(orig)
|
|
c.Check(cpy, DeepEquals, orig)
|
|
|
|
cpy["d"].([]interface{})[0] = 999
|
|
c.Check(orig["d"].([]interface{})[0], Equals, "x")
|
|
cpy["e"].(map[string]interface{})["e1"] = "x"
|
|
c.Check(orig["e"].(map[string]interface{})["e1"], Equals, "E1")
|
|
}
|
|
|
|
func (s *connSuite) TestNewConnectedPlugExplicitStaticAttrs(c *C) {
|
|
staticAttrs := map[string]interface{}{
|
|
"baz": "boom",
|
|
}
|
|
dynAttrs := map[string]interface{}{
|
|
"foo": "bar",
|
|
}
|
|
plug := interfaces.NewConnectedPlug(s.plug, s.plugAppSet, staticAttrs, dynAttrs)
|
|
c.Assert(plug, NotNil)
|
|
c.Assert(plug.StaticAttrs(), DeepEquals, map[string]interface{}{"baz": "boom"})
|
|
c.Assert(plug.DynamicAttrs(), DeepEquals, map[string]interface{}{"foo": "bar"})
|
|
}
|
|
|
|
func (s *connSuite) TestNewConnectedSlotExplicitStaticAttrs(c *C) {
|
|
staticAttrs := map[string]interface{}{
|
|
"baz": "boom",
|
|
}
|
|
dynAttrs := map[string]interface{}{
|
|
"foo": "bar",
|
|
}
|
|
slot := interfaces.NewConnectedSlot(s.slot, s.slotAppSet, staticAttrs, dynAttrs)
|
|
c.Assert(slot, NotNil)
|
|
c.Assert(slot.StaticAttrs(), DeepEquals, map[string]interface{}{"baz": "boom"})
|
|
c.Assert(slot.DynamicAttrs(), DeepEquals, map[string]interface{}{"foo": "bar"})
|
|
}
|
|
|
|
func (s *connSuite) TestGetAttributeUnhappy(c *C) {
|
|
attrs := map[string]interface{}{}
|
|
var stringVal string
|
|
err := interfaces.GetAttribute("snap0", "iface0", attrs, attrs, "non-existent", &stringVal)
|
|
c.Check(stringVal, Equals, "")
|
|
c.Check(err, ErrorMatches, `snap "snap0" does not have attribute "non-existent" for interface "iface0"`)
|
|
c.Check(errors.Is(err, snap.AttributeNotFoundError{}), Equals, true)
|
|
}
|
|
|
|
func (s *connSuite) TestGetAttributeHappy(c *C) {
|
|
staticAttrs := map[string]interface{}{
|
|
"attr0": "a string",
|
|
"attr1": 12,
|
|
}
|
|
dynamicAttrs := map[string]interface{}{
|
|
"attr0": "second string",
|
|
"attr1": 42,
|
|
}
|
|
var intVal int
|
|
err := interfaces.GetAttribute("snap0", "iface0", staticAttrs, dynamicAttrs, "attr1", &intVal)
|
|
c.Check(err, IsNil)
|
|
c.Check(intVal, Equals, 42)
|
|
}
|
|
|
|
func (s *connSuite) TestPlugLabelExpression(c *C) {
|
|
const snapYaml = `
|
|
name: consumer
|
|
version: 0
|
|
apps:
|
|
app:
|
|
plugs: [plug, app-plug]
|
|
plugs:
|
|
plug:
|
|
interface: interface
|
|
comp-plug:
|
|
interface: interface
|
|
app-plug:
|
|
interface: interface
|
|
hook-plug:
|
|
interface: interface
|
|
hooks:
|
|
install:
|
|
plugs: [plug, hook-plug]
|
|
components:
|
|
comp:
|
|
type: test
|
|
hooks:
|
|
install:
|
|
plugs: [plug, comp-plug]
|
|
`
|
|
|
|
const componentYaml = `
|
|
component: consumer+comp
|
|
type: test
|
|
version: 1
|
|
`
|
|
|
|
appSet := ifacetest.MockInfoAndAppSet(c, snapYaml, []string{componentYaml}, nil)
|
|
info := appSet.Info()
|
|
|
|
// all runnables have it, should use glob
|
|
connectedPlug := interfaces.NewConnectedPlug(info.Plugs["plug"], appSet, nil, nil)
|
|
c.Assert(connectedPlug.LabelExpression(), Equals, `"snap.consumer.*"`)
|
|
|
|
// make sure component hooks are considered
|
|
connectedPlug = interfaces.NewConnectedPlug(info.Plugs["comp-plug"], appSet, nil, nil)
|
|
c.Assert(connectedPlug.LabelExpression(), Equals, `"snap.consumer+comp.hook.install"`)
|
|
|
|
// make sure apps are considered
|
|
connectedPlug = interfaces.NewConnectedPlug(info.Plugs["app-plug"], appSet, nil, nil)
|
|
c.Assert(connectedPlug.LabelExpression(), Equals, `"snap.consumer.app"`)
|
|
|
|
// make sure hooks are considered
|
|
connectedPlug = interfaces.NewConnectedPlug(info.Plugs["hook-plug"], appSet, nil, nil)
|
|
c.Assert(connectedPlug.LabelExpression(), Equals, `"snap.consumer.hook.install"`)
|
|
}
|
|
|
|
func (s *connSuite) TestSlotLabelExpression(c *C) {
|
|
const snapYaml = `
|
|
name: producer
|
|
version: 0
|
|
apps:
|
|
app:
|
|
slots: [slot, app-slot]
|
|
slots:
|
|
slot:
|
|
app-slot:
|
|
hook-slot:
|
|
hooks:
|
|
install:
|
|
slots: [slot, hook-slot]
|
|
components:
|
|
comp:
|
|
type: test
|
|
hooks:
|
|
install:
|
|
`
|
|
|
|
const componentYaml = `
|
|
component: producer+comp
|
|
type: test
|
|
version: 1
|
|
`
|
|
|
|
appSet := ifacetest.MockInfoAndAppSet(c, snapYaml, []string{componentYaml}, nil)
|
|
info := appSet.Info()
|
|
|
|
// components do not have slots right now, so we should not expect a glob
|
|
connectedPlug := interfaces.NewConnectedSlot(info.Slots["slot"], appSet, nil, nil)
|
|
c.Check(connectedPlug.LabelExpression(), Equals, `"snap.producer{.app,.hook.install}"`)
|
|
|
|
// make sure apps are considered
|
|
connectedPlug = interfaces.NewConnectedSlot(info.Slots["app-slot"], appSet, nil, nil)
|
|
c.Check(connectedPlug.LabelExpression(), Equals, `"snap.producer.app"`)
|
|
|
|
// make sure hooks are considered
|
|
connectedPlug = interfaces.NewConnectedSlot(info.Slots["hook-slot"], appSet, nil, nil)
|
|
c.Check(connectedPlug.LabelExpression(), Equals, `"snap.producer.hook.install"`)
|
|
}
|