Files
snapd/interfaces/policy/basedeclaration.go
Zygmunt Krynicki fbea4c4d1a interfaces,daemon,overlord: rename MetaData to StaticInfo
There is a lot of meta-data considering interfaces but some if it is
static and is baked into the binary. Let's give it an appropriate name.

Signed-off-by: Zygmunt Krynicki <zygmunt.krynicki@canonical.com>
2017-07-26 15:09:29 +02:00

200 lines
6.5 KiB
Go

// -*- Mode: Go; indent-tabs-mode: t -*-
/*
* Copyright (C) 2016 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 policy
import (
"bytes"
"fmt"
"strings"
"github.com/snapcore/snapd/asserts"
"github.com/snapcore/snapd/interfaces"
"github.com/snapcore/snapd/interfaces/builtin"
)
// The headers of the builtin base-declaration describing the default
// interface policies for all snaps. The base declaration focuses on the slot
// side for almost all interfaces. Importantly, items are not merged between
// the slots and plugs or between the base declaration and snap declaration
// for a particular type of rule. This means that if you specify an
// installation rule for both slots and plugs in the base declaration, only
// the plugs side is used (plugs is preferred over slots).
//
// The interfaces listed in the base declaration can be broadly categorized
// into:
//
// - manually connected implicit slots (eg, bluetooth-control)
// - auto-connected implicit slots (eg, network)
// - manually connected app-provided slots (eg, bluez)
// - auto-connected app-provided slots (eg, mir)
//
// such that they will follow this pattern:
//
// slots:
// manual-connected-implicit-slot:
// allow-installation:
// slot-snap-type:
// - core # implicit slot
// deny-auto-connection: true # force manual connect
//
// auto-connected-implicit-slot:
// allow-installation:
// slot-snap-type:
// - core # implicit slot
// allow-auto-connection: true # allow auto-connect
//
// manual-connected-provided-slot:
// allow-installation:
// slot-snap-type:
// - app # app provided slot
// deny-connection: true # require allow-connection in snap decl
// deny-auto-connection: true # force manual connect
//
// auto-connected-provided-slot:
// allow-installation:
// slot-snap-type:
// - app # app provided slot
// deny-connection: true # require allow-connection in snap decl
//
// App-provided slots use 'deny-connection: true' since slot implementations
// require privileged access to the system and the snap must be trusted. In
// this manner a snap declaration is required to override the base declaration
// to allow connections with the app-provided slot.
//
// Slots dealing with hardware typically will specify 'gadget' and 'core' as
// the slot-snap-type (eg, serial-port). Eg:
//
// slots:
// manual-connected-hw-slot:
// allow-installation:
// slot-snap-type:
// - core
// - gadget
// deny-auto-connection: true
//
// So called super-privileged slot implementations should also be disallowed
// installation on a system and a snap declaration is required to override the
// base declaration to allow installation (eg, docker). Eg:
//
// slots:
// manual-connected-super-privileged-slot:
// allow-installation: false
// deny-connection: true
// deny-auto-connection: true
//
// Like super-privileged slot implementation, super-privileged plugs should
// also be disallowed installation on a system and a snap declaration is
// required to override the base declaration to allow installation (eg,
// kernel-module-control). Eg:
//
// plugs:
// manual-connected-super-privileged-plug:
// allow-installation: false
// deny-auto-connection: true
// (remember this overrides slot side rules)
//
// Some interfaces have policy that is meant to be used with slot
// implementations and on classic images. Since the slot implementation is
// privileged, we require a snap declaration to be used for app-provided slot
// implementations on non-classic systems (eg, network-manager). Eg:
//
// slots:
// classic-or-not-slot:
// allow-installation:
// slot-snap-type:
// - app
// - core
// deny-auto-connection: true
// deny-connection:
// on-classic: false
//
// Some interfaces have policy that is only used with implicit slots on
// classic and should be autoconnected only there (eg, home). Eg:
//
// slots:
// implicit-classic-slot:
// allow-installation:
// slot-snap-type:
// - core
// deny-auto-connection:
// on-classic: false
//
const baseDeclarationHeader = `
type: base-declaration
authority-id: canonical
series: 16
revision: 0
`
const baseDeclarationPlugs = `
plugs:
`
const baseDeclarationSlots = `
slots:
`
func trimTrailingNewline(s string) string {
return strings.TrimRight(s, "\n")
}
func composeBaseDeclaration(ifaces []interfaces.Interface) ([]byte, error) {
var buf bytes.Buffer
// Trim newlines at the end of the string. All the elements may have
// spurious trailing newlines. All elements start with a leading newline.
// We don't want any blanks as that would no longer parse.
if _, err := buf.WriteString(trimTrailingNewline(baseDeclarationHeader)); err != nil {
return nil, err
}
if _, err := buf.WriteString(trimTrailingNewline(baseDeclarationPlugs)); err != nil {
return nil, err
}
for _, iface := range ifaces {
plugPolicy := interfaces.StaticInfoOf(iface).BaseDeclarationPlugs
if _, err := buf.WriteString(trimTrailingNewline(plugPolicy)); err != nil {
return nil, err
}
}
if _, err := buf.WriteString(trimTrailingNewline(baseDeclarationSlots)); err != nil {
return nil, err
}
for _, iface := range ifaces {
slotPolicy := interfaces.StaticInfoOf(iface).BaseDeclarationSlots
if _, err := buf.WriteString(trimTrailingNewline(slotPolicy)); err != nil {
return nil, err
}
}
if _, err := buf.WriteRune('\n'); err != nil {
return nil, err
}
return buf.Bytes(), nil
}
func init() {
decl, err := composeBaseDeclaration(builtin.Interfaces())
if err != nil {
panic(fmt.Sprintf("cannot compose base-declaration: %v", err))
}
if err := asserts.InitBuiltinBaseDeclaration(decl); err != nil {
panic(fmt.Sprintf("cannot initialize the builtin base-declaration: %v", err))
}
}