Files
coder/coderd/usersecrets_audit_test.go
T
Cian Johnston df1bfe6479 feat: audit user secret create, update, and delete (#24756) (#24849)
Emit user secret audit log entries for create/update/delete operations.
Reads stay un-audited, matching every other resource.

Audit log entries record changes in user secret name, environment
variable name, file path, and value. The secret value column is marked
`ActionSecret` so the diff records the change without showing the
ciphertext or plaintext.

Closes a TOCTOU window on delete to ensure no phantom audit logs for a
delete of a non-existent secret. Secret update accepts a small TOCTOU
window matching the other audited resources (templates, workspaces,
chats). The two-query pattern is wrapped in a transaction so audit state
can't leak from a failed mutation.

(cherry picked from commit 1c30d52b2b)

<!--

If you have used AI to produce some or all of this PR, please ensure you
have read our [AI Contribution
guidelines](https://coder.com/docs/about/contributing/AI_CONTRIBUTING)
before submitting.

-->

Co-authored-by: Zach <3724288+zedkipp@users.noreply.github.com>
2026-04-30 21:01:27 +01:00

179 lines
5.2 KiB
Go

package coderd_test
import (
"net/http"
"strings"
"testing"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"github.com/coder/coder/v2/coderd/audit"
"github.com/coder/coder/v2/coderd/coderdtest"
"github.com/coder/coder/v2/coderd/database"
"github.com/coder/coder/v2/codersdk"
"github.com/coder/coder/v2/testutil"
)
//nolint:paralleltest,tparallel // Subtests share one coderdtest.New server and run sequentially.
func TestUserSecretAudit(t *testing.T) {
t.Parallel()
auditor := audit.NewMock()
client := coderdtest.New(t, &coderdtest.Options{Auditor: auditor})
_ = coderdtest.CreateFirstUser(t, client)
ctx := testutil.Context(t, testutil.WaitMedium)
genSecretName := func(t *testing.T) string {
// Use test name derived secret names so subtests cannot
// collide in the shared user's secret namespace.
return strings.ReplaceAll(t.Name(), "/", "-")
}
t.Run("CreateEmitsLog", func(t *testing.T) {
auditor.ResetLogs()
secret, err := client.CreateUserSecret(ctx, codersdk.Me, codersdk.CreateUserSecretRequest{
Name: genSecretName(t),
Value: "ghp_xxxxxxxxxxxx",
})
require.NoError(t, err)
logs := auditor.AuditLogs()
require.Len(t, logs, 1)
assert.Equal(t, database.AuditActionCreate, logs[0].Action)
assert.Equal(t, secret.ID, logs[0].ResourceID)
assert.Equal(t, secret.Name, logs[0].ResourceTarget)
assert.EqualValues(t, http.StatusCreated, logs[0].StatusCode)
})
t.Run("UpdateEmitsLog", func(t *testing.T) {
auditor.ResetLogs()
secret, err := client.CreateUserSecret(ctx, codersdk.Me, codersdk.CreateUserSecretRequest{
Name: genSecretName(t),
Value: "old",
})
require.NoError(t, err)
newDescription := "rotated"
newValue := "new-value"
_, err = client.UpdateUserSecret(ctx, codersdk.Me, secret.Name, codersdk.UpdateUserSecretRequest{
Description: &newDescription,
Value: &newValue,
})
require.NoError(t, err)
logs := auditor.AuditLogs()
require.Len(t, logs, 2)
assert.Equal(t, database.AuditActionCreate, logs[0].Action)
assert.Equal(t, database.AuditActionWrite, logs[1].Action)
assert.Equal(t, secret.ID, logs[1].ResourceID)
assert.Equal(t, secret.Name, logs[1].ResourceTarget)
assert.EqualValues(t, http.StatusOK, logs[1].StatusCode)
})
t.Run("DeleteEmitsLog", func(t *testing.T) {
auditor.ResetLogs()
secret, err := client.CreateUserSecret(ctx, codersdk.Me, codersdk.CreateUserSecretRequest{
Name: genSecretName(t),
Value: "value",
})
require.NoError(t, err)
require.NoError(t, client.DeleteUserSecret(ctx, codersdk.Me, secret.Name))
logs := auditor.AuditLogs()
require.Len(t, logs, 2)
assert.Equal(t, database.AuditActionCreate, logs[0].Action)
assert.Equal(t, database.AuditActionDelete, logs[1].Action)
assert.Equal(t, secret.ID, logs[1].ResourceID)
assert.Equal(t, secret.Name, logs[1].ResourceTarget)
assert.EqualValues(t, http.StatusNoContent, logs[1].StatusCode)
})
t.Run("DeleteOfMissingWritesNoLog", func(t *testing.T) {
auditor.ResetLogs()
err := client.DeleteUserSecret(ctx, codersdk.Me, "does-not-exist")
var sdkErr *codersdk.Error
require.ErrorAs(t, err, &sdkErr)
assert.Equal(t, http.StatusNotFound, sdkErr.StatusCode())
require.Empty(t, auditor.AuditLogs())
})
t.Run("UpdateOfMissingWritesNoLog", func(t *testing.T) {
auditor.ResetLogs()
desc := "anything"
_, err := client.UpdateUserSecret(ctx, codersdk.Me, "does-not-exist", codersdk.UpdateUserSecretRequest{
Description: &desc,
})
var sdkErr *codersdk.Error
require.ErrorAs(t, err, &sdkErr)
assert.Equal(t, http.StatusNotFound, sdkErr.StatusCode())
require.Empty(t, auditor.AuditLogs())
})
t.Run("ValidationFailureWritesNoLog", func(t *testing.T) {
auditor.ResetLogs()
_, err := client.CreateUserSecret(ctx, codersdk.Me, codersdk.CreateUserSecretRequest{
Name: genSecretName(t),
Value: "value",
EnvName: "1invalid",
})
var sdkErr *codersdk.Error
require.ErrorAs(t, err, &sdkErr)
assert.Equal(t, http.StatusBadRequest, sdkErr.StatusCode())
require.Empty(t, auditor.AuditLogs())
})
t.Run("EmptyUpdateWritesNoLog", func(t *testing.T) {
auditor.ResetLogs()
name := genSecretName(t)
_, err := client.CreateUserSecret(ctx, codersdk.Me, codersdk.CreateUserSecretRequest{
Name: name,
Value: "value",
})
require.NoError(t, err)
// Reset to ignore the created log. We are only testing that the
// no-op update does not add a new log.
auditor.ResetLogs()
_, err = client.UpdateUserSecret(ctx, codersdk.Me, name, codersdk.UpdateUserSecretRequest{})
var sdkErr *codersdk.Error
require.ErrorAs(t, err, &sdkErr)
assert.Equal(t, http.StatusBadRequest, sdkErr.StatusCode())
require.Empty(t, auditor.AuditLogs())
})
t.Run("ReadsDoNotAudit", func(t *testing.T) {
auditor.ResetLogs()
secretName := genSecretName(t)
_, err := client.CreateUserSecret(ctx, codersdk.Me, codersdk.CreateUserSecretRequest{
Name: secretName,
Value: "value",
})
require.NoError(t, err)
// Discard the create log so the assertion below only sees audit entries
// produced by later reads.
auditor.ResetLogs()
_, err = client.UserSecrets(ctx, codersdk.Me)
require.NoError(t, err)
_, err = client.UserSecretByName(ctx, codersdk.Me, secretName)
require.NoError(t, err)
require.Empty(t, auditor.AuditLogs())
})
}