scorecard/checks/permissions.go

282 lines
8.5 KiB
Go
Raw Normal View History

// Copyright 2021 Security Scorecard Authors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package checks
import (
"fmt"
"strings"
"gopkg.in/yaml.v2"
"github.com/ossf/scorecard/v2/checker"
sce "github.com/ossf/scorecard/v2/errors"
)
// CheckTokenPermissions is the exported name for Token-Permissions check.
const CheckTokenPermissions = "Token-Permissions"
//nolint:gochecknoinits
func init() {
registerCheck(CheckTokenPermissions, TokenPermissions)
}
// Holds stateful data to pass thru callbacks.
// Each field correpsonds to a GitHub permission type, and
// will hold true if declared non-write, false otherwise.
type permissionCbData struct {
writePermissions map[string]bool
}
// TokenPermissions runs Token-Permissions check.
func TokenPermissions(c *checker.CheckRequest) checker.CheckResult {
data := permissionCbData{writePermissions: make(map[string]bool)}
err := CheckFilesContent(".github/workflows/*", false,
c, validateGitHubActionTokenPermissions, &data)
return createResultForLeastPrivilegeTokens(data, err)
}
func validatePermission(key string, value interface{}, path string,
dl checker.DetailLogger, pdata *permissionCbData) error {
val, ok := value.(string)
if !ok {
//nolint
return sce.Create(sce.ErrScorecardInternal, errInvalidGitHubWorkflowFile.Error())
}
if strings.EqualFold(val, "write") {
if isPermissionOfInterest(key) {
dl.Warn("'%v' permission set to '%v' in %v", key, val, path)
recordPermissionWrite(key, pdata)
} else {
// Only log for debugging, otherwise
// it may confuse users.
dl.Debug("'%v' permission set to '%v' in %v", key, val, path)
}
return nil
}
dl.Info("'%v' permission set to '%v' in %v", key, val, path)
return nil
}
func validateMapPermissions(values map[interface{}]interface{}, path string,
dl checker.DetailLogger, pdata *permissionCbData) error {
// Iterate over the permission, verify keys and values are strings.
for k, v := range values {
key, ok := k.(string)
if !ok {
//nolint
return sce.Create(sce.ErrScorecardInternal, errInvalidGitHubWorkflowFile.Error())
}
if err := validatePermission(key, v, path, dl, pdata); err != nil {
return err
}
}
return nil
}
func recordPermissionWrite(name string, pdata *permissionCbData) {
pdata.writePermissions[name] = true
}
func recordAllPermissionsWrite(pdata *permissionCbData) {
// Special case: `all` does not correspond
// to a GitHub permission.
pdata.writePermissions["all"] = true
}
func validateReadPermissions(config map[interface{}]interface{}, path string,
dl checker.DetailLogger, pdata *permissionCbData) error {
var permissions interface{}
// Check if permissions are set explicitly.
permissions, ok := config["permissions"]
if !ok {
dl.Warn("no permission defined in %v", path)
recordAllPermissionsWrite(pdata)
return nil
}
// Check the type of our values.
switch val := permissions.(type) {
// Empty string is nil type.
// It defaults to 'none'
case nil:
dl.Info("permissions set to 'none' in %v", path)
// String type.
case string:
if !strings.EqualFold(val, "read-all") && val != "" {
dl.Warn("permissions set to '%v' in %v", val, path)
recordAllPermissionsWrite(pdata)
return nil
}
dl.Info("permission set to '%v' in %v", val, path)
// Map type.
case map[interface{}]interface{}:
if err := validateMapPermissions(val, path, dl, pdata); err != nil {
return err
}
// Invalid type.
default:
//nolint
return sce.Create(sce.ErrScorecardInternal, errInvalidGitHubWorkflowFile.Error())
}
return nil
}
func isPermissionOfInterest(name string) bool {
return strings.EqualFold(name, "statuses") ||
strings.EqualFold(name, "checks") ||
strings.EqualFold(name, "security-events") ||
strings.EqualFold(name, "deployments") ||
strings.EqualFold(name, "contents") ||
strings.EqualFold(name, "packages") ||
strings.EqualFold(name, "options")
}
// Calculate the score.
func calculateScore(result permissionCbData) int {
// See list https://github.blog/changelog/2021-04-20-github-actions-control-permissions-for-github_token/.
// Note: there are legitimate reasons to use some of the permissions like checks, deployments, etc.
// in CI/CD systems https://docs.travis-ci.com/user/github-oauth-scopes/.
if _, ok := result.writePermissions["all"]; ok {
return checker.MinResultScore
}
score := float32(checker.MaxResultScore)
// status: https://docs.github.com/en/rest/reference/repos#statuses.
// May allow an attacker to change the result of pre-submit and get a PR merged.
// Low risk: -0.5.
if _, ok := result.writePermissions["statuses"]; ok {
score -= 0.5
}
// checks.
// May allow an attacker to edit checks to remove pre-submit and introduce a bug.
// Low risk: -0.5.
if _, ok := result.writePermissions["checks"]; ok {
score -= 0.5
}
// secEvents.
// May allow attacker to read vuln reports before patch available.
// Low risk: -1
if _, ok := result.writePermissions["security-events"]; ok {
score--
}
// deployments: https://docs.github.com/en/rest/reference/repos#deployments.
// May allow attacker to charge repo owner by triggering VM runs,
// and tiny chance an attacker can trigger a remote
// service with code they own if server accepts code/location var unsanitized.
// Low risk: -1
if _, ok := result.writePermissions["deployments"]; ok {
score--
}
// contents.
// Allows attacker to commit unreviewed code.
// High risk: -10
if _, ok := result.writePermissions["contents"]; ok {
score -= checker.MaxResultScore
}
// packages.
// Allows attacker to publish packages.
// High risk: -10
if _, ok := result.writePermissions["packages"]; ok {
score -= checker.MaxResultScore
}
// actions.
// May allow an attacker to steal GitHub secrets by adding a malicious workflow/action.
// High risk: -10
if _, ok := result.writePermissions["actions"]; ok {
score -= checker.MaxResultScore
}
if score < checker.MinResultScore {
return checker.MinResultScore
}
return int(score)
}
// Create the result.
func createResultForLeastPrivilegeTokens(result permissionCbData, err error) checker.CheckResult {
if err != nil {
return checker.CreateRuntimeErrorResult(CheckTokenPermissions, err)
}
score := calculateScore(result)
if score != checker.MaxResultScore {
return checker.CreateResultWithScore(CheckTokenPermissions,
"non read-only tokens detected in GitHub workflows", score)
}
return checker.CreateMaxScoreResult(CheckTokenPermissions,
"tokens are read-only in GitHub workflows")
}
func testValidateGitHubActionTokenPermissions(pathfn string,
content []byte, dl checker.DetailLogger) checker.CheckResult {
data := permissionCbData{writePermissions: make(map[string]bool)}
_, err := validateGitHubActionTokenPermissions(pathfn, content, dl, &data)
return createResultForLeastPrivilegeTokens(data, err)
}
// Check file content.
func validateGitHubActionTokenPermissions(path string, content []byte,
dl checker.DetailLogger, data FileCbData) (bool, error) {
// Verify the type of the data.
pdata, ok := data.(*permissionCbData)
if !ok {
// This never happens.
panic("invalid type")
}
if !CheckFileContainsCommands(content, "#") {
return true, nil
}
var workflow map[interface{}]interface{}
err := yaml.Unmarshal(content, &workflow)
if err != nil {
//nolint
return false,
sce.Create(sce.ErrScorecardInternal, fmt.Sprintf("yaml.Unmarshal: %v", err))
}
// 1. Check that each file uses 'content: read' only or 'none'.
//nolint
// https://docs.github.com/en/actions/reference/authentication-in-a-workflow#example-1-passing-the-github_token-as-an-input,
// https://github.blog/changelog/2021-04-20-github-actions-control-permissions-for-github_token/,
// https://docs.github.com/en/actions/reference/authentication-in-a-workflow#modifying-the-permissions-for-the-github_token.
if err := validateReadPermissions(workflow, path, dl, pdata); err != nil {
return false, err
}
// TODO(laurent): 2. Identify github actions that require write and add checks.
// TODO(laurent): 3. Read a few runs and ensures they have the same permissions.
return true, nil
}