Skip to content

Commit

Permalink
test: add test for checking logs in fixSemiSync
Browse files Browse the repository at this point in the history
Signed-off-by: Manan Gupta <[email protected]>
  • Loading branch information
GuptaManan100 committed Jan 22, 2022
1 parent b19bdda commit ddcaa78
Showing 1 changed file with 119 additions and 0 deletions.
119 changes: 119 additions & 0 deletions go/vt/vttablet/tabletmanager/rpc_replication_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,11 +17,16 @@ limitations under the License.
package tabletmanager

import (
"bytes"
"context"
"io"
"os"
"testing"

"github.com/stretchr/testify/require"

"vitess.io/vitess/go/vt/mysqlctl/fakemysqldaemon"
"vitess.io/vitess/go/vt/proto/topodata"
"vitess.io/vitess/go/vt/topo/memorytopo"
)

Expand All @@ -38,3 +43,117 @@ func TestPromoteReplicaHealthTicksStopped(t *testing.T) {
require.NoError(t, err)
require.False(t, tm.replManager.ticks.Running())
}

func captureStderr(f func()) (string, error) {
old := os.Stderr // keep backup of the real stderr
r, w, err := os.Pipe()
if err != nil {
return "", err
}
os.Stderr = w

outC := make(chan string)
// copy the output in a separate goroutine so printing can't block indefinitely
go func() {
var buf bytes.Buffer
io.Copy(&buf, r)
outC <- buf.String()
}()

// calling function which stderr we are going to capture:
f()

// back to normal state
w.Close()
os.Stderr = old // restoring the real stderr
return <-outC, nil
}

func TestTabletManager_fixSemiSync(t *testing.T) {
tests := []struct {
name string
tabletType topodata.TabletType
semiSync SemiSyncAction
logOutput string
shouldEnableSemiSync bool
}{
{
name: "enableSemiSync=true(primary eligible),durabilitySemiSync=true",
tabletType: topodata.TabletType_REPLICA,
semiSync: SemiSyncActionTrue,
logOutput: "",
shouldEnableSemiSync: true,
}, {
name: "enableSemiSync=true(primary eligible),durabilitySemiSync=false",
tabletType: topodata.TabletType_REPLICA,
semiSync: SemiSyncActionFalse,
logOutput: "invalid configuration - enabling semi sync even though not specified by durability policies.",
shouldEnableSemiSync: true,
}, {
name: "enableSemiSync=true(primary eligible),durabilitySemiSync=none",
tabletType: topodata.TabletType_REPLICA,
semiSync: SemiSyncActionNone,
logOutput: "",
shouldEnableSemiSync: true,
}, {
name: "enableSemiSync=true(primary not-eligible),durabilitySemiSync=true",
tabletType: topodata.TabletType_DRAINED,
semiSync: SemiSyncActionTrue,
logOutput: "invalid configuration - semi-sync should be setup according to durability policies, but the tablet is not primaryEligible",
shouldEnableSemiSync: true,
}, {
name: "enableSemiSync=true(primary not-eligible),durabilitySemiSync=false",
tabletType: topodata.TabletType_DRAINED,
semiSync: SemiSyncActionFalse,
logOutput: "",
shouldEnableSemiSync: true,
}, {
name: "enableSemiSync=true(primary not-eligible),durabilitySemiSync=none",
tabletType: topodata.TabletType_DRAINED,
semiSync: SemiSyncActionNone,
logOutput: "",
shouldEnableSemiSync: true,
}, {
name: "enableSemiSync=false,durabilitySemiSync=true",
tabletType: topodata.TabletType_REPLICA,
semiSync: SemiSyncActionTrue,
logOutput: "invalid configuration - semi-sync should be setup according to durability policies, but enable_semi_sync is not set",
shouldEnableSemiSync: false,
}, {
name: "enableSemiSync=false,durabilitySemiSync=false",
tabletType: topodata.TabletType_REPLICA,
semiSync: SemiSyncActionFalse,
logOutput: "",
shouldEnableSemiSync: false,
}, {
name: "enableSemiSync=false,durabilitySemiSync=none",
tabletType: topodata.TabletType_REPLICA,
semiSync: SemiSyncActionNone,
logOutput: "",
shouldEnableSemiSync: false,
},
}
oldEnableSemiSync := *enableSemiSync
defer func() {
*enableSemiSync = oldEnableSemiSync
}()
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
*enableSemiSync = tt.shouldEnableSemiSync
fakeMysql := fakemysqldaemon.NewFakeMysqlDaemon(nil)
tm := &TabletManager{
MysqlDaemon: fakeMysql,
}
logOutput, err := captureStderr(func() {
err := tm.fixSemiSync(tt.tabletType, tt.semiSync)
require.NoError(t, err)
})
require.NoError(t, err)
if tt.logOutput != "" {
require.Contains(t, logOutput, tt.logOutput)
} else {
require.Equal(t, "", logOutput)
}
})
}
}

0 comments on commit ddcaa78

Please sign in to comment.