mattermost/tools/sharedchannel-test/test_membership.go
Doug Lauder 85dcb8b9e7
MM-67944: Add shared channel integration test tool (#35639)
* Add shared channel integration test tool (MM-67944)

  Add tools/sharedchannel-test, a standalone Go tool that validates shared
  channel synchronization between two real Mattermost Enterprise instances.
  The tool builds and manages two server processes with separate databases,
  establishes a remote cluster connection, and runs integration tests for
  membership, post, and reaction sync.

  Test coverage:
  - Membership: add, remove, re-add, bulk remove
  - Posts: create, edit, delete
  - Reactions: add, remove

  Uses mlog with dual console targets (stdout for info, stderr for errors)
  and exits non-zero on failure for CI integration.
2026-03-25 11:53:56 -04:00

129 lines
3.7 KiB
Go

package main
import (
"context"
"fmt"
"time"
)
func (tr *TestRunner) runMembershipTests(ctx context.Context) error {
tr.logger.Info("=== Membership Sync Tests ===")
channelA, channelB, err := tr.setupSharedChannel(ctx, "membership-test")
if err != nil {
tr.fail("membership/setup", err.Error())
return err
}
// Create test users
userIDs := make([]string, 3)
for i := range 3 {
name := fmt.Sprintf("memtest%d", i+1)
id, err := tr.createTestUser(ctx, name)
if err != nil {
tr.fail("membership/create-user-"+name, err.Error())
return err
}
userIDs[i] = id
}
// ── Test: Add users and verify sync ─────────────────────
tr.logger.Info("Adding users to shared channel...")
for i, uid := range userIDs {
_, _, err := tr.clientA.AddChannelMember(ctx, channelA, uid)
if err != nil {
tr.fail(fmt.Sprintf("membership/add-user-%d", i+1), err.Error())
return err
}
}
tr.logger.Info("Waiting for membership sync...")
for i := range 3 {
name := fmt.Sprintf("memtest%d", i+1)
testName := "membership/add-sync-" + name
err := tr.waitFor(ctx, 30*time.Second, func() bool {
return tr.verifyMemberOnB(ctx, channelB, name)
})
if err != nil {
tr.fail(testName, fmt.Sprintf("user %s did not sync to Server B: %v", name, err))
} else {
tr.pass(testName)
}
}
// ── Test: Remove one user and verify sync ───────────────
tr.logger.Info("Removing memtest3 from shared channel...")
_, err = tr.clientA.RemoveUserFromChannel(ctx, channelA, userIDs[2])
if err != nil {
tr.fail("membership/remove-memtest3", err.Error())
} else {
testName := "membership/remove-sync-memtest3"
err := tr.waitFor(ctx, 30*time.Second, func() bool {
return !tr.verifyMemberOnB(ctx, channelB, "memtest3")
})
if err != nil {
tr.fail(testName, "memtest3 still present on Server B after removal")
} else {
tr.pass(testName)
}
// Verify others still present
for _, name := range []string{"memtest1", "memtest2"} {
testName := "membership/still-present-" + name
if tr.verifyMemberOnB(ctx, channelB, name) {
tr.pass(testName)
} else {
tr.fail(testName, name+" unexpectedly missing after memtest3 removal")
}
}
}
// ── Test: Re-add removed user ───────────────────────────
tr.logger.Info("Re-adding memtest3...")
_, _, err = tr.clientA.AddChannelMember(ctx, channelA, userIDs[2])
if err != nil {
tr.fail("membership/re-add-memtest3", err.Error())
} else {
testName := "membership/re-add-sync-memtest3"
err := tr.waitFor(ctx, 30*time.Second, func() bool {
return tr.verifyMemberOnB(ctx, channelB, "memtest3")
})
if err != nil {
tr.fail(testName, "memtest3 re-add did not sync to Server B")
} else {
tr.pass(testName)
}
}
// ── Test: Bulk removal ──────────────────────────────────
tr.logger.Info("Bulk removing all 3 users...")
for _, uid := range userIDs {
_, _ = tr.clientA.RemoveUserFromChannel(ctx, channelA, uid)
}
testName := "membership/bulk-remove-sync"
err = tr.waitFor(ctx, 30*time.Second, func() bool {
for i := range 3 {
if tr.verifyMemberOnB(ctx, channelB, fmt.Sprintf("memtest%d", i+1)) {
return false
}
}
return true
})
if err != nil {
tr.fail(testName, "not all users removed from Server B after bulk removal")
} else {
tr.pass(testName)
}
return nil
}
// verifyChannelMemberCount returns the number of members in a channel on Server B.
func (tr *TestRunner) verifyChannelMemberCount(ctx context.Context, channelB string) int {
members, _, err := tr.clientB.GetChannelMembers(ctx, channelB, 0, 200, "")
if err != nil {
return -1
}
return len(members)
}