fleet/server/platform/mysql/errors_test.go
Victor Lyuboslavsky 42d5f1fda6
Improve error handling on AWS DB failover (#39841)
<!-- Add the related story/sub-task/bug number, like Resolves #123, or
remove if NA -->
**Related issue:** Resolves #39228 

Manually tested by triggering a failover on loadtest.

# Checklist for submitter

If some of the following don't apply, delete the relevant line.

- [x] Changes file added for user-visible changes in `changes/`,
`orbit/changes/` or `ee/fleetd-chrome/changes`.

## Testing

- [x] Added/updated automated tests
- [x] QA'd all new/changed functionality manually


<!-- This is an auto-generated comment: release notes by coderabbit.ai
-->
## Summary by CodeRabbit

* **Bug Fixes**
* Health checks now detect a primary DB becoming read-only and report
failure so the service restarts and reconnects to a writable primary.
* Write failures due to DB read-only state now trigger immediate fatal
handling to prompt graceful shutdown and recovery.
* Improved detection and handling of read-only DB conditions to increase
stability during failovers.
<!-- end of auto-generated comment: release notes by coderabbit.ai -->
2026-02-17 07:10:52 -06:00

52 lines
1.1 KiB
Go

package mysql
import (
"fmt"
"testing"
"github.com/go-sql-driver/mysql"
"github.com/stretchr/testify/assert"
)
func TestIsReadOnlyError(t *testing.T) {
t.Parallel()
cases := []struct {
name string
err error
want bool
}{
{
name: "unrelated MySQL error",
err: &mysql.MySQLError{Number: 1045, Message: "Access denied"},
want: false,
},
{
name: "error 1792 read-only transaction",
err: &mysql.MySQLError{Number: 1792, Message: "Cannot execute statement in a READ ONLY transaction."},
want: true,
},
{
name: "error 1290 option prevents statement",
err: &mysql.MySQLError{Number: 1290, Message: "The MySQL server is running with the --read-only option"},
want: true,
},
{
name: "error 1836 read-only mode",
err: &mysql.MySQLError{Number: 1836, Message: "Running in read-only mode"},
want: true,
},
{
name: "wrapped read-only error",
err: fmt.Errorf("transaction failed: %w", &mysql.MySQLError{Number: 1792, Message: "read only"}),
want: true,
},
}
for _, tc := range cases {
t.Run(tc.name, func(t *testing.T) {
t.Parallel()
assert.Equal(t, tc.want, IsReadOnlyError(tc.err))
})
}
}