mirror of
https://github.com/grafana/grafana.git
synced 2025-08-02 03:02:18 +08:00

* Plugins: Unsigned chromium file should not invalidate signature for Renderer plugin * fix test * re-work solution
559 lines
15 KiB
Go
559 lines
15 KiB
Go
package signature
|
|
|
|
import (
|
|
"path/filepath"
|
|
"sort"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/grafana/grafana/pkg/infra/log"
|
|
"github.com/grafana/grafana/pkg/plugins"
|
|
"github.com/grafana/grafana/pkg/setting"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestReadPluginManifest(t *testing.T) {
|
|
txt := `-----BEGIN PGP SIGNED MESSAGE-----
|
|
Hash: SHA512
|
|
|
|
{
|
|
"plugin": "grafana-googlesheets-datasource",
|
|
"version": "1.0.0-dev",
|
|
"files": {
|
|
"LICENSE": "7df059597099bb7dcf25d2a9aedfaf4465f72d8d",
|
|
"README.md": "08ec6d704b6115bef57710f6d7e866c050cb50ee",
|
|
"gfx_sheets_darwin_amd64": "1b8ae92c6e80e502bb0bf2d0ae9d7223805993ab",
|
|
"gfx_sheets_linux_amd64": "f39e0cc7344d3186b1052e6d356eecaf54d75b49",
|
|
"gfx_sheets_windows_amd64.exe": "c8825dfec512c1c235244f7998ee95182f9968de",
|
|
"module.js": "aaec6f51a995b7b843b843cd14041925274d960d",
|
|
"module.js.LICENSE.txt": "7f822fe9341af8f82ad1b0c69aba957822a377cf",
|
|
"module.js.map": "c5a524f5c4237f6ed6a016d43cd46938efeadb45",
|
|
"plugin.json": "55556b845e91935cc48fae3aa67baf0f22694c3f"
|
|
},
|
|
"time": 1586817677115,
|
|
"keyId": "7e4d0c6a708866e7"
|
|
}
|
|
-----BEGIN PGP SIGNATURE-----
|
|
Version: OpenPGP.js v4.10.1
|
|
Comment: https://openpgpjs.org
|
|
|
|
wqEEARMKAAYFAl6U6o0ACgkQfk0ManCIZuevWAIHSvcxOy1SvvL5gC+HpYyG
|
|
VbSsUvF2FsCoXUCTQflK6VdJfSPNzm8YdCdx7gNrBdly6HEs06ZaRp44F/ve
|
|
NR7DnB0CCQHO+4FlSPtXFTzNepoc+CytQyDAeOLMLmf2Tqhk2YShk+G/YlVX
|
|
74uuP5UXZxwK2YKJovdSknDIU7MhfuvvQIP/og==
|
|
=hBea
|
|
-----END PGP SIGNATURE-----`
|
|
|
|
t.Run("valid manifest", func(t *testing.T) {
|
|
manifest, err := readPluginManifest([]byte(txt))
|
|
|
|
require.NoError(t, err)
|
|
require.NotNil(t, manifest)
|
|
assert.Equal(t, "grafana-googlesheets-datasource", manifest.Plugin)
|
|
assert.Equal(t, "1.0.0-dev", manifest.Version)
|
|
assert.Equal(t, int64(1586817677115), manifest.Time)
|
|
assert.Equal(t, "7e4d0c6a708866e7", manifest.KeyID)
|
|
expectedFiles := []string{"LICENSE", "README.md", "gfx_sheets_darwin_amd64", "gfx_sheets_linux_amd64",
|
|
"gfx_sheets_windows_amd64.exe", "module.js", "module.js.LICENSE.txt", "module.js.map", "plugin.json",
|
|
}
|
|
assert.Equal(t, expectedFiles, fileList(manifest))
|
|
})
|
|
|
|
t.Run("invalid manifest", func(t *testing.T) {
|
|
modified := strings.ReplaceAll(txt, "README.md", "xxxxxxxxxx")
|
|
_, err := readPluginManifest([]byte(modified))
|
|
require.Error(t, err)
|
|
})
|
|
}
|
|
|
|
func TestReadPluginManifestV2(t *testing.T) {
|
|
txt := `-----BEGIN PGP SIGNED MESSAGE-----
|
|
Hash: SHA512
|
|
|
|
{
|
|
"manifestVersion": "2.0.0",
|
|
"signatureType": "private",
|
|
"signedByOrg": "willbrowne",
|
|
"signedByOrgName": "Will Browne",
|
|
"rootUrls": [
|
|
"http://localhost:3000/"
|
|
],
|
|
"plugin": "test",
|
|
"version": "1.0.0",
|
|
"time": 1605807018050,
|
|
"keyId": "7e4d0c6a708866e7",
|
|
"files": {
|
|
"plugin.json": "2bb467c0bfd6c454551419efe475b8bf8573734e73c7bab52b14842adb62886f"
|
|
}
|
|
}
|
|
-----BEGIN PGP SIGNATURE-----
|
|
Version: OpenPGP.js v4.10.1
|
|
Comment: https://openpgpjs.org
|
|
|
|
wqIEARMKAAYFAl+2q6oACgkQfk0ManCIZudmzwIJAXWz58cd/91rTXszKPnE
|
|
xbVEvERCbjKTtPBQBNQyqEvV+Ig3MuBSNOVy2SOGrMsdbS6lONgvgt4Cm+iS
|
|
wV+vYifkAgkBJtg/9DMB7/iX5O0h49CtSltcpfBFXlGqIeOwRac/yENzRzAA
|
|
khdr/tZ1PDgRxMqB/u+Vtbpl0xSxgblnrDOYMSI=
|
|
=rLIE
|
|
-----END PGP SIGNATURE-----`
|
|
|
|
t.Run("valid manifest", func(t *testing.T) {
|
|
manifest, err := readPluginManifest([]byte(txt))
|
|
|
|
require.NoError(t, err)
|
|
require.NotNil(t, manifest)
|
|
assert.Equal(t, "test", manifest.Plugin)
|
|
assert.Equal(t, "1.0.0", manifest.Version)
|
|
assert.Equal(t, int64(1605807018050), manifest.Time)
|
|
assert.Equal(t, "7e4d0c6a708866e7", manifest.KeyID)
|
|
assert.Equal(t, "2.0.0", manifest.ManifestVersion)
|
|
assert.Equal(t, plugins.PrivateSignature, manifest.SignatureType)
|
|
assert.Equal(t, "willbrowne", manifest.SignedByOrg)
|
|
assert.Equal(t, "Will Browne", manifest.SignedByOrgName)
|
|
assert.Equal(t, []string{"http://localhost:3000/"}, manifest.RootURLs)
|
|
assert.Equal(t, []string{"plugin.json"}, fileList(manifest))
|
|
})
|
|
}
|
|
|
|
func TestCalculate(t *testing.T) {
|
|
t.Run("Validate root URL against App URL for non-private plugin if is specified in manifest", func(t *testing.T) {
|
|
tcs := []struct {
|
|
appURL string
|
|
expectedSignature plugins.Signature
|
|
}{
|
|
{
|
|
appURL: "https://dev.grafana.com",
|
|
expectedSignature: plugins.Signature{
|
|
Status: plugins.SignatureValid,
|
|
Type: plugins.GrafanaSignature,
|
|
SigningOrg: "Grafana Labs",
|
|
},
|
|
},
|
|
{
|
|
appURL: "https://non.matching.url.com",
|
|
expectedSignature: plugins.Signature{
|
|
Status: plugins.SignatureInvalid,
|
|
},
|
|
},
|
|
}
|
|
|
|
parentDir, err := filepath.Abs("../")
|
|
if err != nil {
|
|
t.Errorf("could not construct absolute path of current dir")
|
|
return
|
|
}
|
|
|
|
for _, tc := range tcs {
|
|
origAppURL := setting.AppUrl
|
|
t.Cleanup(func() {
|
|
setting.AppUrl = origAppURL
|
|
})
|
|
setting.AppUrl = tc.appURL
|
|
|
|
sig, err := Calculate(log.NewNopLogger(), &plugins.Plugin{
|
|
JSONData: plugins.JSONData{
|
|
ID: "test-datasource",
|
|
Info: plugins.Info{
|
|
Version: "1.0.0",
|
|
},
|
|
},
|
|
PluginDir: filepath.Join(parentDir, "testdata/non-pvt-with-root-url/plugin"),
|
|
Class: plugins.External,
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, tc.expectedSignature, sig)
|
|
}
|
|
})
|
|
|
|
t.Run("Unsigned Chromium file should not invalidate signature for Renderer plugin running on Windows", func(t *testing.T) {
|
|
backup := runningWindows
|
|
t.Cleanup(func() {
|
|
runningWindows = backup
|
|
})
|
|
|
|
runningWindows = true
|
|
sig, err := Calculate(log.NewNopLogger(), &plugins.Plugin{
|
|
JSONData: plugins.JSONData{
|
|
ID: "test-renderer",
|
|
Type: plugins.Renderer,
|
|
Info: plugins.Info{
|
|
Version: "1.0.0",
|
|
},
|
|
},
|
|
PluginDir: "../testdata/renderer-added-file/plugin",
|
|
})
|
|
require.NoError(t, err)
|
|
require.Equal(t, plugins.Signature{
|
|
Status: plugins.SignatureValid,
|
|
Type: plugins.GrafanaSignature,
|
|
SigningOrg: "Grafana Labs",
|
|
}, sig)
|
|
})
|
|
}
|
|
|
|
func fileList(manifest *pluginManifest) []string {
|
|
var keys []string
|
|
for k := range manifest.Files {
|
|
keys = append(keys, k)
|
|
}
|
|
sort.Strings(keys)
|
|
return keys
|
|
}
|
|
|
|
func Test_urlMatch_privateGlob(t *testing.T) {
|
|
type args struct {
|
|
specs []string
|
|
target string
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
args args
|
|
shouldMatch bool
|
|
}{
|
|
{
|
|
name: "Support single wildcard matching single subdomain",
|
|
args: args{
|
|
specs: []string{"https://*.example.com"},
|
|
target: "https://test.example.com",
|
|
},
|
|
shouldMatch: true,
|
|
},
|
|
{
|
|
name: "Do not support single wildcard matching multiple subdomains",
|
|
args: args{
|
|
specs: []string{"https://*.example.com"},
|
|
target: "https://more.test.example.com",
|
|
},
|
|
shouldMatch: false,
|
|
},
|
|
{
|
|
name: "Support multiple wildcards matching multiple subdomains",
|
|
args: args{
|
|
specs: []string{"https://**.example.com"},
|
|
target: "https://test.example.com",
|
|
},
|
|
shouldMatch: true,
|
|
},
|
|
{
|
|
name: "Support multiple wildcards matching multiple subdomains",
|
|
args: args{
|
|
specs: []string{"https://**.example.com"},
|
|
target: "https://more.test.example.com",
|
|
},
|
|
shouldMatch: true,
|
|
},
|
|
{
|
|
name: "Support single wildcard matching single paths",
|
|
args: args{
|
|
specs: []string{"https://www.example.com/*"},
|
|
target: "https://www.example.com/grafana1",
|
|
},
|
|
shouldMatch: true,
|
|
},
|
|
{
|
|
name: "Do not support single wildcard matching multiple paths",
|
|
args: args{
|
|
specs: []string{"https://www.example.com/*"},
|
|
target: "https://www.example.com/other/grafana",
|
|
},
|
|
shouldMatch: false,
|
|
},
|
|
{
|
|
name: "Support double wildcard matching multiple paths",
|
|
args: args{
|
|
specs: []string{"https://www.example.com/**"},
|
|
target: "https://www.example.com/other/grafana",
|
|
},
|
|
shouldMatch: true,
|
|
},
|
|
{
|
|
name: "Do not support subdomain mismatch",
|
|
args: args{
|
|
specs: []string{"https://www.test.example.com/grafana/docs"},
|
|
target: "https://www.dev.example.com/grafana/docs",
|
|
},
|
|
shouldMatch: false,
|
|
},
|
|
{
|
|
name: "Support single wildcard matching single path",
|
|
args: args{
|
|
specs: []string{"https://www.example.com/grafana*"},
|
|
target: "https://www.example.com/grafana1",
|
|
},
|
|
shouldMatch: true,
|
|
},
|
|
{
|
|
name: "Do not support single wildcard matching different path prefix",
|
|
args: args{
|
|
specs: []string{"https://www.example.com/grafana*"},
|
|
target: "https://www.example.com/somethingelse",
|
|
},
|
|
shouldMatch: false,
|
|
},
|
|
{
|
|
name: "Do not support path mismatch",
|
|
args: args{
|
|
specs: []string{"https://example.com/grafana"},
|
|
target: "https://example.com/grafana1",
|
|
},
|
|
shouldMatch: false,
|
|
},
|
|
{
|
|
name: "Support both domain and path wildcards",
|
|
args: args{
|
|
specs: []string{"https://*.example.com/*"},
|
|
target: "https://www.example.com/grafana1",
|
|
},
|
|
shouldMatch: true,
|
|
},
|
|
{
|
|
name: "Do not support wildcards without TLDs",
|
|
args: args{
|
|
specs: []string{"https://example.*"},
|
|
target: "https://www.example.com/grafana1",
|
|
},
|
|
shouldMatch: false,
|
|
},
|
|
{
|
|
name: "Support exact match",
|
|
args: args{
|
|
specs: []string{"https://example.com/test"},
|
|
target: "https://example.com/test",
|
|
},
|
|
shouldMatch: true,
|
|
},
|
|
{
|
|
name: "Does not support scheme mismatch",
|
|
args: args{
|
|
specs: []string{"https://test.example.com/grafana"},
|
|
target: "http://test.example.com/grafana",
|
|
},
|
|
shouldMatch: false,
|
|
},
|
|
{
|
|
name: "Support trailing slash in spec",
|
|
args: args{
|
|
specs: []string{"https://example.com/"},
|
|
target: "https://example.com",
|
|
},
|
|
shouldMatch: true,
|
|
},
|
|
{
|
|
name: "Support trailing slash in target",
|
|
args: args{
|
|
specs: []string{"https://example.com"},
|
|
target: "https://example.com/",
|
|
},
|
|
shouldMatch: true,
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
got, err := urlMatch(tt.args.specs, tt.args.target, plugins.PrivateGlobSignature)
|
|
require.NoError(t, err)
|
|
require.Equal(t, tt.shouldMatch, got)
|
|
})
|
|
}
|
|
}
|
|
|
|
func Test_urlMatch_private(t *testing.T) {
|
|
type args struct {
|
|
specs []string
|
|
target string
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
args args
|
|
shouldMatch bool
|
|
}{
|
|
{
|
|
name: "Support exact match",
|
|
args: args{
|
|
specs: []string{"https://example.com/test"},
|
|
target: "https://example.com/test",
|
|
},
|
|
shouldMatch: true,
|
|
},
|
|
{
|
|
name: "Support trailing slash in spec",
|
|
args: args{
|
|
specs: []string{"https://example.com/test/"},
|
|
target: "https://example.com/test",
|
|
},
|
|
shouldMatch: true,
|
|
},
|
|
{
|
|
name: "Support trailing slash in target",
|
|
args: args{
|
|
specs: []string{"https://example.com/test"},
|
|
target: "https://example.com/test/",
|
|
},
|
|
shouldMatch: true,
|
|
},
|
|
{
|
|
name: "Do not support single wildcard matching single subdomain",
|
|
args: args{
|
|
specs: []string{"https://*.example.com"},
|
|
target: "https://test.example.com",
|
|
},
|
|
shouldMatch: false,
|
|
},
|
|
{
|
|
name: "Do not support multiple wildcards matching multiple subdomains",
|
|
args: args{
|
|
specs: []string{"https://**.example.com"},
|
|
target: "https://more.test.example.com",
|
|
},
|
|
shouldMatch: false,
|
|
},
|
|
{
|
|
name: "Do not support single wildcard matching single paths",
|
|
args: args{
|
|
specs: []string{"https://www.example.com/*"},
|
|
target: "https://www.example.com/grafana1",
|
|
},
|
|
shouldMatch: false,
|
|
},
|
|
{
|
|
name: "Do not support double wildcard matching multiple paths",
|
|
args: args{
|
|
specs: []string{"https://www.example.com/**"},
|
|
target: "https://www.example.com/other/grafana",
|
|
},
|
|
shouldMatch: false,
|
|
},
|
|
{
|
|
name: "Do not support subdomain mismatch",
|
|
args: args{
|
|
specs: []string{"https://www.test.example.com/grafana/docs"},
|
|
target: "https://www.dev.example.com/grafana/docs",
|
|
},
|
|
shouldMatch: false,
|
|
},
|
|
{
|
|
name: "Do not support path mismatch",
|
|
args: args{
|
|
specs: []string{"https://example.com/grafana"},
|
|
target: "https://example.com/grafana1",
|
|
},
|
|
shouldMatch: false,
|
|
},
|
|
{
|
|
name: "Do not support both domain and path wildcards",
|
|
args: args{
|
|
specs: []string{"https://*.example.com/*"},
|
|
target: "https://www.example.com/grafana1",
|
|
},
|
|
shouldMatch: false,
|
|
},
|
|
{
|
|
name: "Do not support wildcards without TLDs",
|
|
args: args{
|
|
specs: []string{"https://example.*"},
|
|
target: "https://www.example.com/grafana1",
|
|
},
|
|
shouldMatch: false,
|
|
},
|
|
{
|
|
name: "Do not support scheme mismatch",
|
|
args: args{
|
|
specs: []string{"https://test.example.com/grafana"},
|
|
target: "http://test.example.com/grafana",
|
|
},
|
|
shouldMatch: false,
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
got, err := urlMatch(tt.args.specs, tt.args.target, plugins.PrivateSignature)
|
|
require.NoError(t, err)
|
|
require.Equal(t, tt.shouldMatch, got)
|
|
})
|
|
}
|
|
}
|
|
|
|
func Test_validateManifest(t *testing.T) {
|
|
tcs := []struct {
|
|
name string
|
|
manifest *pluginManifest
|
|
expectedErr string
|
|
}{
|
|
{
|
|
name: "Empty plugin field",
|
|
manifest: createV2Manifest(t, func(m *pluginManifest) { m.Plugin = "" }),
|
|
expectedErr: "valid manifest field plugin is required",
|
|
},
|
|
{
|
|
name: "Empty keyId field",
|
|
manifest: createV2Manifest(t, func(m *pluginManifest) { m.KeyID = "" }),
|
|
expectedErr: "valid manifest field keyId is required",
|
|
},
|
|
{
|
|
name: "Empty signedByOrg field",
|
|
manifest: createV2Manifest(t, func(m *pluginManifest) { m.SignedByOrg = "" }),
|
|
expectedErr: "valid manifest field signedByOrg is required",
|
|
},
|
|
{
|
|
name: "Empty signedByOrgName field",
|
|
manifest: createV2Manifest(t, func(m *pluginManifest) { m.SignedByOrgName = "" }),
|
|
expectedErr: "valid manifest field SignedByOrgName is required",
|
|
},
|
|
{
|
|
name: "Empty signatureType field",
|
|
manifest: createV2Manifest(t, func(m *pluginManifest) { m.SignatureType = "" }),
|
|
expectedErr: "valid manifest field signatureType is required",
|
|
},
|
|
{
|
|
name: "Invalid signatureType field",
|
|
manifest: createV2Manifest(t, func(m *pluginManifest) { m.SignatureType = "invalidSignatureType" }),
|
|
expectedErr: "valid manifest field signatureType is required",
|
|
},
|
|
{
|
|
name: "Empty files field",
|
|
manifest: createV2Manifest(t, func(m *pluginManifest) { m.Files = map[string]string{} }),
|
|
expectedErr: "valid manifest field files is required",
|
|
},
|
|
{
|
|
name: "Empty time field",
|
|
manifest: createV2Manifest(t, func(m *pluginManifest) { m.Time = 0 }),
|
|
expectedErr: "valid manifest field time is required",
|
|
},
|
|
{
|
|
name: "Empty version field",
|
|
manifest: createV2Manifest(t, func(m *pluginManifest) { m.Version = "" }),
|
|
expectedErr: "valid manifest field version is required",
|
|
},
|
|
}
|
|
for _, tc := range tcs {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
err := validateManifest(*tc.manifest, nil)
|
|
require.Errorf(t, err, tc.expectedErr)
|
|
})
|
|
}
|
|
}
|
|
|
|
func createV2Manifest(t *testing.T, cbs ...func(*pluginManifest)) *pluginManifest {
|
|
t.Helper()
|
|
|
|
m := &pluginManifest{
|
|
Plugin: "grafana-test-app",
|
|
Version: "2.5.3",
|
|
KeyID: "7e4d0c6a708866e7",
|
|
Time: 1586817677115,
|
|
Files: map[string]string{
|
|
"plugin.json": "55556b845e91935cc48fae3aa67baf0f22694c3f",
|
|
},
|
|
ManifestVersion: "2.0.0",
|
|
SignatureType: plugins.GrafanaSignature,
|
|
SignedByOrg: "grafana",
|
|
SignedByOrgName: "grafana",
|
|
}
|
|
|
|
for _, cb := range cbs {
|
|
cb(m)
|
|
}
|
|
|
|
return m
|
|
}
|