github.com/keybase/client/go@v0.0.0-20241007131713-f10651d043c8/chat/flip/dealer.go (about)

     1  package flip
     2  
     3  import (
     4  	"context"
     5  	"encoding/base64"
     6  	"encoding/hex"
     7  	"fmt"
     8  	"io"
     9  	"math"
    10  	"math/big"
    11  	"strings"
    12  	"time"
    13  
    14  	chat1 "github.com/keybase/client/go/protocol/chat1"
    15  	clockwork "github.com/keybase/clockwork"
    16  )
    17  
    18  // Excludes `Params` from being logged.
    19  func (s Start) String() string {
    20  	return fmt.Sprintf("{StartTime:%v CommitmentWindowMsec:%v RevealWindowMsec:%v SlackMsec:%v CommitmentCompleteWindowMsec:%v}",
    21  		s.StartTime, s.CommitmentWindowMsec, s.RevealWindowMsec, s.SlackMsec, s.CommitmentWindowMsec)
    22  }
    23  
    24  type GameMessageWrapped struct {
    25  	Sender              UserDevice
    26  	Msg                 GameMessageV1
    27  	Me                  *playerControl
    28  	Forward             bool
    29  	FirstInConversation bool
    30  }
    31  
    32  func (m GameMessageWrapped) isForwardable() bool {
    33  	t, _ := m.Msg.Body.T()
    34  	return t != MessageType_END
    35  }
    36  
    37  func (m GameMessageWrapped) GameMetadata() GameMetadata {
    38  	return m.Msg.Md
    39  }
    40  
    41  func (g GameMetadata) ToKey() GameKey {
    42  	return GameKey(strings.Join([]string{g.Initiator.U.String(), g.Initiator.D.String(), g.ConversationID.String(), g.GameID.String()}, ","))
    43  }
    44  
    45  func (g GameMetadata) String() string {
    46  	return string(g.ToKey())
    47  }
    48  
    49  func (g GameMetadata) check() bool {
    50  	return g.Initiator.check() && !g.ConversationID.IsNil() && g.GameID.Check()
    51  }
    52  
    53  type GameKey string
    54  type GameIDKey string
    55  type UserDeviceKey string
    56  
    57  func (u UserDevice) ToKey() UserDeviceKey {
    58  	return UserDeviceKey(strings.Join([]string{u.U.String(), u.D.String()}, ","))
    59  }
    60  
    61  func (u UserDevice) check() bool {
    62  	return u.U.Bytes() != nil && u.D.Bytes() != nil
    63  }
    64  
    65  func GameIDToKey(g chat1.FlipGameID) GameIDKey {
    66  	return GameIDKey(g.String())
    67  }
    68  
    69  type Result struct {
    70  	Shuffle []int
    71  	Bool    *bool
    72  	Int     *int64
    73  	Big     *big.Int
    74  }
    75  
    76  type Game struct {
    77  	md                     GameMetadata
    78  	msgID                  int
    79  	clockSkew              time.Duration
    80  	start                  time.Time
    81  	isLeader               bool
    82  	params                 Start
    83  	key                    GameKey
    84  	msgCh                  <-chan *GameMessageWrapped
    85  	stage                  Stage
    86  	stageForTimeout        Stage
    87  	players                map[UserDeviceKey]*GamePlayerState
    88  	commitments            map[string]bool
    89  	gameUpdateCh           chan GameStateUpdateMessage
    90  	nPlayers               int
    91  	dealer                 *Dealer
    92  	me                     *playerControl
    93  	commitmentCompleteHash Hash
    94  	clock                  func() clockwork.Clock
    95  	clogf                  func(ctx context.Context, fmt string, args ...interface{})
    96  
    97  	// To handle reorderings between CommitmentComplete and commitements,
    98  	// wee need some extra bookkeeping.
    99  	iWasIncluded          bool
   100  	iOptedOutOfCommit     bool
   101  	gotCommitmentComplete bool
   102  	latecomers            map[UserDeviceKey]bool
   103  }
   104  
   105  type GamePlayerState struct {
   106  	ud               UserDevice
   107  	commitment       *Commitment
   108  	commitmentTime   time.Time
   109  	leaderCommitment *Commitment
   110  	included         bool
   111  	secret           *Secret
   112  }
   113  
   114  func (g *Game) GameMetadata() GameMetadata {
   115  	return g.md
   116  }
   117  
   118  func MakeGameMessageEncoded(s string) GameMessageEncoded {
   119  	return GameMessageEncoded(s)
   120  }
   121  
   122  func (e GameMessageEncoded) String() string {
   123  	return string(e)
   124  }
   125  
   126  func (e GameMessageEncoded) Decode() (*GameMessageV1, error) {
   127  	raw, err := base64.StdEncoding.DecodeString(string(e))
   128  	if err != nil {
   129  		return nil, err
   130  	}
   131  	var msg GameMessage
   132  	err = msgpackDecode(&msg, raw)
   133  	if err != nil {
   134  		return nil, err
   135  	}
   136  	v, err := msg.V()
   137  	if err != nil {
   138  		return nil, err
   139  	}
   140  	if v != Version_V1 {
   141  		return nil, BadVersionError(v)
   142  	}
   143  	tmp := msg.V1()
   144  	if !tmp.Md.check() {
   145  		return nil, ErrBadData
   146  	}
   147  	return &tmp, nil
   148  }
   149  
   150  func (e *GameMessageWrappedEncoded) Decode() (*GameMessageWrapped, error) {
   151  	v1, err := e.Body.Decode()
   152  	if err != nil {
   153  		return nil, err
   154  	}
   155  	ret := GameMessageWrapped{Sender: e.Sender, Msg: *v1, FirstInConversation: e.FirstInConversation}
   156  	if !e.GameID.Eq(ret.Msg.Md.GameID) {
   157  		return nil, BadGameIDError{G: ret.Msg.Md, I: e.GameID}
   158  	}
   159  	return &ret, nil
   160  }
   161  
   162  func (m GameMessageWrapped) Encode() (GameMessageEncoded, error) {
   163  	return m.Msg.Encode()
   164  }
   165  
   166  func (b GameMessageBody) Encode(md GameMetadata) (GameMessageEncoded, error) {
   167  	v1 := GameMessageV1{Md: md, Body: b}
   168  	return v1.Encode()
   169  }
   170  
   171  func (v GameMessageV1) Encode() (GameMessageEncoded, error) {
   172  	msg := NewGameMessageWithV1(v)
   173  	raw, err := msgpackEncode(msg)
   174  	if err != nil {
   175  		return GameMessageEncoded(""), err
   176  	}
   177  	return GameMessageEncoded(base64.StdEncoding.EncodeToString(raw)), nil
   178  }
   179  
   180  func (d *Dealer) run(ctx context.Context, game *Game) {
   181  	doneCh := make(chan error)
   182  	key := game.key
   183  	go game.run(ctx, doneCh)
   184  	err := <-doneCh
   185  
   186  	if err != nil {
   187  		d.dh.CLogf(ctx, "[%s] Error running game %s: %s", d.dh.Me(), key, err.Error())
   188  
   189  	} else {
   190  		d.dh.CLogf(ctx, "Game %s ended cleanly", key)
   191  	}
   192  
   193  	// If the game was shutdown via the Dealer#Stop call, then
   194  	// don't close the channel (again) or remove the channel from the
   195  	// map, it's already dead.
   196  	if _, ok := err.(GameShutdownError); ok {
   197  		return
   198  	}
   199  
   200  	d.Lock()
   201  	if ch := d.games[key]; ch != nil {
   202  		close(ch)
   203  		delete(d.games, key)
   204  		delete(d.gameIDs, GameIDToKey(game.md.GameID))
   205  	}
   206  	d.Unlock()
   207  }
   208  
   209  func (g *Game) getNextTimer() <-chan time.Time {
   210  	dl := g.nextDeadline()
   211  	return g.clock().AfterTime(dl)
   212  }
   213  
   214  func (g *Game) CommitmentEndTime() time.Time {
   215  	// If we're the leader, then let's cut off when we say we're going to cut off
   216  	// If we're not, then let's give extra time (a multiple of 2) to the leader.
   217  	return g.start.Add(g.params.CommitmentWindowWithSlack(g.isLeader))
   218  }
   219  
   220  func (g *Game) RevealEndTime() time.Time {
   221  	return g.start.Add(g.params.RevealWindowWithSlack())
   222  }
   223  
   224  func (g *Game) nextDeadline() time.Time {
   225  	switch g.stageForTimeout {
   226  	case Stage_ROUND1:
   227  		return g.CommitmentEndTime()
   228  	case Stage_ROUND2:
   229  		return g.RevealEndTime()
   230  	default:
   231  		return time.Time{}
   232  	}
   233  }
   234  
   235  func (g Game) commitmentPayload() CommitmentPayload {
   236  	return CommitmentPayload{
   237  		V: Version_V1,
   238  		U: g.md.Initiator.U,
   239  		D: g.md.Initiator.D,
   240  		C: g.md.ConversationID,
   241  		G: g.md.GameID,
   242  		S: g.params.StartTime,
   243  	}
   244  }
   245  
   246  func (g *Game) setSecret(ctx context.Context, ps *GamePlayerState, secret Secret) error {
   247  	expected, err := secret.computeCommitment(g.commitmentPayload())
   248  	if err != nil {
   249  		return err
   250  	}
   251  	if ps.secret != nil {
   252  		return DuplicateRevealError{G: g.md, U: ps.ud}
   253  	}
   254  	if !expected.Eq(*ps.commitment) {
   255  		return BadRevealError{G: g.md, U: ps.ud}
   256  	}
   257  	ps.secret = &secret
   258  	return nil
   259  }
   260  
   261  func (g *Game) finishGame(ctx context.Context) error {
   262  	var xor Secret
   263  	for _, ps := range g.players {
   264  		if !ps.included {
   265  			continue
   266  		}
   267  		if ps.secret == nil {
   268  			return NoRevealError{G: g.md, U: ps.ud}
   269  		}
   270  		xor.XOR(*ps.secret)
   271  	}
   272  	prng := NewPRNG(xor)
   273  	err := g.doFlip(ctx, prng)
   274  	g.sendOutgoingChat(ctx, NewGameMessageBodyWithEnd())
   275  	return err
   276  }
   277  
   278  func (g *Game) doFlip(ctx context.Context, prng *PRNG) error {
   279  	params := g.params.Params
   280  	t, err := params.T()
   281  	if err != nil {
   282  		return err
   283  	}
   284  	var res Result
   285  	switch t {
   286  	case FlipType_BOOL:
   287  		tmp := prng.Bool()
   288  		res.Bool = &tmp
   289  	case FlipType_INT:
   290  		tmp := prng.Int(params.Int())
   291  		res.Int = &tmp
   292  	case FlipType_BIG:
   293  		var modulus big.Int
   294  		modulus.SetBytes(params.Big())
   295  		res.Big = prng.Big(&modulus)
   296  	case FlipType_SHUFFLE:
   297  		res.Shuffle = prng.Permutation(int(params.Shuffle()))
   298  	default:
   299  		return BadFlipTypeError{G: g.GameMetadata(), T: t}
   300  	}
   301  
   302  	g.gameUpdateCh <- GameStateUpdateMessage{
   303  		Metadata: g.GameMetadata(),
   304  		Result:   &res,
   305  	}
   306  	return nil
   307  }
   308  
   309  func (g *Game) playerCommitedInTime(ps *GamePlayerState, now time.Time) bool {
   310  	diff := ps.commitmentTime.Sub(g.start)
   311  	return diff < g.params.CommitmentWindowWithSlack(true)
   312  }
   313  
   314  func (g *Game) getPlayerState(ud UserDevice) *GamePlayerState {
   315  	key := ud.ToKey()
   316  	ret := g.players[key]
   317  	if ret != nil {
   318  		return ret
   319  	}
   320  	ret = &GamePlayerState{ud: ud}
   321  	g.players[key] = ret
   322  	return ret
   323  }
   324  
   325  func (g *Game) handleCommitment(ctx context.Context, sender UserDevice, now time.Time, com Commitment) (err error) {
   326  	ps := g.getPlayerState(sender)
   327  	if ps.commitment != nil {
   328  		return DuplicateRegistrationError{g.md, sender}
   329  	}
   330  	if ps.leaderCommitment != nil && !ps.leaderCommitment.Eq(com) {
   331  		return CommitmentMismatchError{G: g.GameMetadata(), U: sender}
   332  	}
   333  	ps.commitment = &com
   334  	ps.commitmentTime = now
   335  
   336  	comHex := hex.EncodeToString(com[:])
   337  	if g.commitments[comHex] {
   338  		return DuplicateCommitmentError{}
   339  	}
   340  	g.commitments[comHex] = true
   341  
   342  	// If this user was a latecomer (we got the commitment after we got the CommitmentComplete),
   343  	// then we mark them as being accounted for.
   344  	delete(g.latecomers, sender.ToKey())
   345  
   346  	g.gameUpdateCh <- GameStateUpdateMessage{
   347  		Metadata: g.GameMetadata(),
   348  		Commitment: &CommitmentUpdate{
   349  			User:       sender,
   350  			Commitment: com,
   351  		},
   352  	}
   353  	return g.maybeReveal(ctx)
   354  }
   355  
   356  func (g *Game) maybeReveal(ctx context.Context) (err error) {
   357  
   358  	if !g.gotCommitmentComplete {
   359  		return nil
   360  	}
   361  	if len(g.latecomers) > 0 {
   362  		return nil
   363  	}
   364  
   365  	g.stage = Stage_ROUND2
   366  	g.stageForTimeout = Stage_ROUND2
   367  
   368  	if g.me == nil {
   369  		return nil
   370  	}
   371  
   372  	if !g.iWasIncluded && !g.iOptedOutOfCommit {
   373  		g.clogf(ctx, "The leader didn't include me (%s) so not sending a reveal (%s)", g.me.me, g.md)
   374  		return nil
   375  	}
   376  
   377  	reveal := Reveal{
   378  		Secret: g.me.secret,
   379  		Cch:    g.commitmentCompleteHash,
   380  	}
   381  	g.sendOutgoingChat(ctx, NewGameMessageBodyWithReveal(reveal))
   382  	return nil
   383  }
   384  
   385  func (g *Game) handleCommitmentCompletePlayer(ctx context.Context, u UserDeviceCommitment) (err error) {
   386  
   387  	ps := g.getPlayerState(u.Ud)
   388  	if ps.leaderCommitment != nil {
   389  		return DuplicateCommitmentCompleteError{G: g.md, U: u.Ud}
   390  	}
   391  	if ps.commitment != nil && !ps.commitment.Eq(u.C) {
   392  		return CommitmentMismatchError{G: g.md, U: u.Ud}
   393  	}
   394  	if ps.commitment == nil {
   395  		g.latecomers[u.Ud.ToKey()] = true
   396  	}
   397  	ps.leaderCommitment = &u.C
   398  	ps.included = true
   399  	g.nPlayers++
   400  
   401  	if g.me != nil && g.me.me.Eq(u.Ud) {
   402  		g.iWasIncluded = true
   403  	}
   404  	return nil
   405  }
   406  
   407  func (g *Game) handleCommitmentComplete(ctx context.Context, sender UserDevice, now time.Time, cc CommitmentComplete) (err error) {
   408  
   409  	if !sender.Eq(g.md.Initiator) {
   410  		return WrongSenderError{G: g.md, Expected: g.md.Initiator, Actual: sender}
   411  	}
   412  
   413  	if !checkUserDeviceCommitments(cc.Players) {
   414  		return CommitmentCompleteSortError{G: g.md}
   415  	}
   416  
   417  	for _, u := range cc.Players {
   418  		err = g.handleCommitmentCompletePlayer(ctx, u)
   419  		if err != nil {
   420  			return err
   421  		}
   422  	}
   423  
   424  	cch, err := hashUserDeviceCommitments(cc.Players)
   425  	if err != nil {
   426  		return err
   427  	}
   428  
   429  	g.commitmentCompleteHash = cch
   430  	g.gotCommitmentComplete = true
   431  
   432  	// for now, just warn if users who made it in on time weren't included.
   433  	for _, ps := range g.players {
   434  		if !ps.included && g.playerCommitedInTime(ps, now) && g.clogf != nil {
   435  			g.clogf(ctx, "User %s wasn't included, but they should have been", ps.ud)
   436  		}
   437  	}
   438  
   439  	g.gameUpdateCh <- GameStateUpdateMessage{
   440  		Metadata:           g.GameMetadata(),
   441  		CommitmentComplete: &cc,
   442  	}
   443  
   444  	return g.maybeReveal(ctx)
   445  }
   446  
   447  func errToOk(err error) string {
   448  	if err == nil {
   449  		return "ok"
   450  	}
   451  	return "ERROR: " + err.Error()
   452  }
   453  
   454  func (g *Game) handleMessage(ctx context.Context, msg *GameMessageWrapped, now time.Time) (err error) {
   455  
   456  	msgID := g.msgID
   457  	g.msgID++
   458  
   459  	g.clogf(ctx, "+ Game#handleMessage: %s@%d <- %+v", g.GameMetadata(), msgID, *msg)
   460  	defer func() { g.clogf(ctx, "- Game#handleMessage: %s@%d -> %s", g.GameMetadata(), msgID, errToOk(err)) }()
   461  
   462  	t, err := msg.Msg.Body.T()
   463  	if err != nil {
   464  		return err
   465  	}
   466  	badStage := func() error {
   467  		return BadMessageForStageError{G: g.GameMetadata(), MessageType: t, Stage: g.stage}
   468  	}
   469  	switch t {
   470  
   471  	case MessageType_START:
   472  		return badStage()
   473  
   474  	case MessageType_END:
   475  		return io.EOF
   476  
   477  	case MessageType_COMMITMENT:
   478  		if g.stage != Stage_ROUND1 {
   479  			g.clogf(ctx, "User %s sent a commitment too late, not included in game %s", msg.Sender, g.md)
   480  			return nil
   481  		}
   482  
   483  		err = g.handleCommitment(ctx, msg.Sender, now, msg.Msg.Body.Commitment())
   484  		if err != nil {
   485  			return err
   486  		}
   487  
   488  	case MessageType_COMMITMENT_COMPLETE:
   489  		if g.stage != Stage_ROUND1 {
   490  			return badStage()
   491  		}
   492  
   493  		err = g.handleCommitmentComplete(ctx, msg.Sender, now, msg.Msg.Body.CommitmentComplete())
   494  		if err != nil {
   495  			return err
   496  		}
   497  
   498  	case MessageType_REVEAL:
   499  		if g.stage != Stage_ROUND2 {
   500  			return badStage()
   501  		}
   502  
   503  		key := msg.Sender.ToKey()
   504  		ps := g.players[key]
   505  
   506  		if ps == nil {
   507  			g.clogf(ctx, "Skipping unregistered revealer %s for game %s", msg.Sender, g.md)
   508  			return nil
   509  		}
   510  		if !ps.included {
   511  			g.clogf(ctx, "Skipping unincluded revealer %s for game %s", msg.Sender, g.md)
   512  			return nil
   513  		}
   514  		if now.After(g.RevealEndTime()) {
   515  			return RevealTooLateError{G: g.md, U: msg.Sender}
   516  		}
   517  
   518  		reveal := msg.Msg.Body.Reveal()
   519  		if !g.commitmentCompleteHash.Eq(reveal.Cch) {
   520  			return BadCommitmentCompleteHashError{G: g.GameMetadata(), U: msg.Sender}
   521  		}
   522  		err := g.setSecret(ctx, ps, reveal.Secret)
   523  		if err != nil {
   524  			return err
   525  		}
   526  		g.gameUpdateCh <- GameStateUpdateMessage{
   527  			Metadata: g.GameMetadata(),
   528  			Reveal: &RevealUpdate{
   529  				User:   msg.Sender,
   530  				Reveal: reveal.Secret,
   531  			},
   532  		}
   533  
   534  		g.nPlayers--
   535  		if g.nPlayers == 0 {
   536  			return g.finishGame(ctx)
   537  		}
   538  
   539  	default:
   540  		return BadMessageError{G: g.GameMetadata()}
   541  	}
   542  
   543  	return nil
   544  }
   545  
   546  func (g *Game) userDeviceCommitmentList() []UserDeviceCommitment {
   547  	var ret []UserDeviceCommitment
   548  	for _, p := range g.players {
   549  		if p.commitment != nil {
   550  			ret = append(ret, UserDeviceCommitment{Ud: p.ud, C: *p.commitment})
   551  		}
   552  	}
   553  	sortUserDeviceCommitments(ret)
   554  	return ret
   555  }
   556  
   557  func (g *Game) completeCommitments(ctx context.Context) error {
   558  	cc := CommitmentComplete{
   559  		Players: g.userDeviceCommitmentList(),
   560  	}
   561  	body := NewGameMessageBodyWithCommitmentComplete(cc)
   562  	g.stageForTimeout = Stage_ROUND2
   563  	g.sendOutgoingChat(ctx, body)
   564  	return nil
   565  }
   566  
   567  func (g *Game) absentees() []UserDevice {
   568  	var bad []UserDevice
   569  	for _, p := range g.players {
   570  		if p.included && p.secret == nil {
   571  			bad = append(bad, p.ud)
   572  		}
   573  	}
   574  	return bad
   575  }
   576  
   577  func (g *Game) sendOutgoingChat(ctx context.Context, body GameMessageBody) {
   578  	// Call back into the dealer, to reroute a message back into our
   579  	// game, but do so in a Go routine so we don't deadlock. There could be
   580  	// 100 incoming messages in front of us, all coming off the chat channel,
   581  	// so we're ok to send when we can. If use the game in the context of
   582  	// replay, the dealer will be nil, so no need to send.
   583  	if g.dealer != nil {
   584  		go func() {
   585  			err := g.dealer.sendOutgoingChat(ctx, g.GameMetadata(), nil, body)
   586  			if err != nil {
   587  				g.clogf(ctx, "Error sending outgoing chat %+v", err)
   588  			}
   589  		}()
   590  	}
   591  }
   592  
   593  func (g *Game) handleTimerEvent(ctx context.Context) error {
   594  	if g.isLeader && g.stageForTimeout == Stage_ROUND1 {
   595  		return g.completeCommitments(ctx)
   596  	}
   597  
   598  	absentees := g.absentees()
   599  
   600  	if g.stageForTimeout == Stage_ROUND2 && len(absentees) > 0 {
   601  		return AbsenteesError{Absentees: absentees}
   602  	}
   603  
   604  	return TimeoutError{G: g.md, Stage: g.stageForTimeout}
   605  }
   606  
   607  func (g *Game) runMain(ctx context.Context) error {
   608  	for {
   609  		timer := g.getNextTimer()
   610  		var err error
   611  		select {
   612  		case <-timer:
   613  			err = g.handleTimerEvent(ctx)
   614  		case msg, ok := <-g.msgCh:
   615  			if !ok {
   616  				return GameShutdownError{G: g.GameMetadata()}
   617  			}
   618  			err = g.handleMessage(ctx, msg, g.clock().Now())
   619  		case <-ctx.Done():
   620  			return ctx.Err()
   621  		}
   622  		if err == io.EOF {
   623  			return nil
   624  		}
   625  		if err != nil {
   626  			g.gameUpdateCh <- GameStateUpdateMessage{
   627  				Metadata: g.GameMetadata(),
   628  				Err:      err,
   629  			}
   630  			return err
   631  		}
   632  	}
   633  }
   634  
   635  func (g *Game) runDrain(ctx context.Context) {
   636  	i := 0
   637  	for range g.msgCh {
   638  		i++
   639  	}
   640  	if i > 0 {
   641  		g.clogf(ctx, "drained %d messages on shutdown in game %s", i, g.md)
   642  	}
   643  }
   644  
   645  func (g *Game) run(ctx context.Context, doneCh chan error) {
   646  	doneCh <- g.runMain(ctx)
   647  	g.runDrain(ctx)
   648  }
   649  
   650  func absDuration(d time.Duration) time.Duration {
   651  	if d < time.Duration(0) {
   652  		return time.Duration(-1) * d
   653  	}
   654  	return d
   655  }
   656  
   657  func (d *Dealer) computeClockSkew(ctx context.Context, md GameMetadata, leaderTime time.Time, myNow time.Time) (skew time.Duration, err error) {
   658  	serverTime, err := d.dh.ServerTime(ctx)
   659  	if err != nil {
   660  		return skew, err
   661  	}
   662  	return computeClockSkew(md, serverTime, leaderTime, myNow)
   663  }
   664  
   665  func computeClockSkew(md GameMetadata, serverTime time.Time, leaderTime time.Time, myNow time.Time) (skew time.Duration, err error) {
   666  	localTime := myNow
   667  	leaderSkew := leaderTime.Sub(serverTime)
   668  	localSkew := localTime.Sub(serverTime)
   669  
   670  	if absDuration(localSkew) > MaxClockSkew {
   671  		return time.Duration(0), BadLocalClockError{G: md}
   672  	}
   673  	if absDuration(leaderSkew) > MaxClockSkew {
   674  		return time.Duration(0), BadLeaderClockError{G: md}
   675  	}
   676  	totalSkew := localTime.Sub(leaderTime)
   677  
   678  	return totalSkew, nil
   679  }
   680  
   681  func (d *Dealer) handleMessageStart(ctx context.Context, msg *GameMessageWrapped, start Start) error {
   682  	d.Lock()
   683  	defer d.Unlock()
   684  	md := msg.GameMetadata()
   685  	key := md.ToKey()
   686  	gameIDKey := GameIDToKey(md.GameID)
   687  	if d.games[key] != nil {
   688  		return GameAlreadyStartedError{G: md}
   689  	}
   690  	if _, found := d.gameIDs[gameIDKey]; found {
   691  		return GameReplayError{G: md.GameID}
   692  	}
   693  	if !msg.Sender.Eq(md.Initiator) {
   694  		return WrongSenderError{G: md, Expected: msg.Sender, Actual: md.Initiator}
   695  	}
   696  	cs, err := d.computeClockSkew(ctx, md, start.StartTime.Time(), d.dh.Clock().Now())
   697  	if err != nil {
   698  		return err
   699  	}
   700  
   701  	if !msg.FirstInConversation {
   702  		return GameReplayError{md.GameID}
   703  	}
   704  
   705  	isLeader := true
   706  	me := msg.Me
   707  	// Make a new follower player controller if one didn't already exist (since we were
   708  	// the Leader)
   709  	if me == nil {
   710  		me, err = d.newPlayerControl(d.dh.Me(), md, start)
   711  		if err != nil {
   712  			return err
   713  		}
   714  		isLeader = false
   715  	}
   716  
   717  	optedOutOfCommit := false
   718  	if !isLeader {
   719  		optedOutOfCommit = !d.dh.ShouldCommit(ctx)
   720  	}
   721  
   722  	msgCh := make(chan *GameMessageWrapped)
   723  	game := &Game{
   724  		md:                msg.GameMetadata(),
   725  		isLeader:          isLeader,
   726  		clockSkew:         cs,
   727  		start:             d.dh.Clock().Now(),
   728  		key:               key,
   729  		params:            start,
   730  		msgCh:             msgCh,
   731  		stage:             Stage_ROUND1,
   732  		stageForTimeout:   Stage_ROUND1,
   733  		gameUpdateCh:      d.gameUpdateCh,
   734  		players:           make(map[UserDeviceKey]*GamePlayerState),
   735  		commitments:       make(map[string]bool),
   736  		dealer:            d,
   737  		me:                me,
   738  		clock:             d.dh.Clock,
   739  		clogf:             d.dh.CLogf,
   740  		latecomers:        make(map[UserDeviceKey]bool),
   741  		iOptedOutOfCommit: optedOutOfCommit,
   742  	}
   743  	d.games[key] = msgCh
   744  	d.gameIDs[gameIDKey] = md
   745  	d.previousGames[GameIDToKey(md.GameID)] = true
   746  
   747  	go d.run(ctx, game)
   748  
   749  	// Once the game has started, we are free to send a message into the channel
   750  	// with our commitment. We are now in the inner loop of the Dealer, so we
   751  	// have to do this send in a Go-routine, so as not to deadlock the Dealer.
   752  	if !isLeader && !optedOutOfCommit {
   753  		go func() {
   754  			err := d.sendCommitment(ctx, md, me)
   755  			if err != nil {
   756  				game.clogf(ctx, "Error sending commitment: %+v", err)
   757  			}
   758  		}()
   759  	}
   760  	return nil
   761  }
   762  
   763  func (d *Dealer) handleMessageOthers(c context.Context, msg *GameMessageWrapped) error {
   764  	d.Lock()
   765  	defer d.Unlock()
   766  	md := msg.GameMetadata()
   767  	key := md.ToKey()
   768  	game := d.games[key]
   769  	if game == nil {
   770  		return GameFinishedError{G: md}
   771  	}
   772  	game <- msg
   773  	return nil
   774  }
   775  
   776  func (d *Dealer) handleMessage(ctx context.Context, msg *GameMessageWrapped) error {
   777  	d.dh.CLogf(ctx, "flip.Dealer: Incoming: %+v", msg)
   778  
   779  	t, err := msg.Msg.Body.T()
   780  	if err != nil {
   781  		return err
   782  	}
   783  	switch t {
   784  	case MessageType_START:
   785  		err = d.handleMessageStart(ctx, msg, msg.Msg.Body.Start())
   786  		if err != nil {
   787  			d.gameUpdateCh <- GameStateUpdateMessage{
   788  				Metadata: msg.Msg.Md,
   789  				Err:      err,
   790  			}
   791  		}
   792  	default:
   793  		err = d.handleMessageOthers(ctx, msg)
   794  	}
   795  	if err != nil {
   796  		return err
   797  	}
   798  	if !(msg.Forward && msg.isForwardable()) {
   799  		return nil
   800  	}
   801  	// Encode and send the message through the external server-routed chat channel
   802  	emsg, err := msg.Encode()
   803  	if err != nil {
   804  		return err
   805  	}
   806  	err = d.dh.SendChat(ctx, msg.Msg.Md.Initiator.U, msg.Msg.Md.ConversationID, msg.Msg.Md.GameID, emsg)
   807  	if err != nil {
   808  		return err
   809  	}
   810  	return nil
   811  }
   812  
   813  func (d *Dealer) stopGames() {
   814  	d.Lock()
   815  	defer d.Unlock()
   816  	for k, ch := range d.games {
   817  		delete(d.games, k)
   818  		close(ch)
   819  	}
   820  }
   821  
   822  type playerControl struct {
   823  	me         UserDevice
   824  	md         GameMetadata
   825  	secret     Secret
   826  	commitment Commitment
   827  	start      Start
   828  	dealer     *Dealer
   829  }
   830  
   831  func (d *Dealer) newPlayerControl(me UserDevice, md GameMetadata, start Start) (*playerControl, error) {
   832  	secret := GenerateSecret()
   833  	cp := CommitmentPayload{
   834  		V: Version_V1,
   835  		U: md.Initiator.U,
   836  		D: md.Initiator.D,
   837  		C: md.ConversationID,
   838  		G: md.GameID,
   839  		S: start.StartTime,
   840  	}
   841  	commitment, err := secret.computeCommitment(cp)
   842  	if err != nil {
   843  		return nil, err
   844  	}
   845  	return &playerControl{
   846  		me:         me,
   847  		md:         md,
   848  		secret:     secret,
   849  		commitment: commitment,
   850  		start:      start,
   851  		dealer:     d,
   852  	}, nil
   853  }
   854  
   855  func (p *playerControl) GameMetadata() GameMetadata {
   856  	return p.md
   857  }
   858  
   859  func (d *Dealer) startFlip(ctx context.Context, start Start, conversationID chat1.ConversationID) (pc *playerControl, err error) {
   860  	return d.startFlipWithGameID(ctx, start, conversationID, GenerateGameID())
   861  }
   862  
   863  func (d *Dealer) startFlipWithGameID(ctx context.Context, start Start, conversationID chat1.ConversationID,
   864  	gameID chat1.FlipGameID) (pc *playerControl, err error) {
   865  	md := GameMetadata{
   866  		Initiator:      d.dh.Me(),
   867  		ConversationID: conversationID,
   868  		GameID:         gameID,
   869  	}
   870  	pc, err = d.newPlayerControl(d.dh.Me(), md, start)
   871  	if err != nil {
   872  		return nil, err
   873  	}
   874  	err = d.sendOutgoingChatWithFirst(ctx, md, pc, NewGameMessageBodyWithStart(start), true)
   875  	if err != nil {
   876  		return nil, err
   877  	}
   878  	err = d.sendCommitment(ctx, md, pc)
   879  	if err != nil {
   880  		return nil, err
   881  	}
   882  	return pc, nil
   883  }
   884  
   885  func (d *Dealer) sendCommitment(ctx context.Context, md GameMetadata, pc *playerControl) error {
   886  	return d.sendOutgoingChat(ctx, md, nil, NewGameMessageBodyWithCommitment(pc.commitment))
   887  }
   888  
   889  func (d *Dealer) sendOutgoingChat(ctx context.Context, md GameMetadata, me *playerControl, body GameMessageBody) error {
   890  	return d.sendOutgoingChatWithFirst(ctx, md, me, body, false)
   891  }
   892  
   893  func (d *Dealer) sendOutgoingChatWithFirst(ctx context.Context, md GameMetadata, me *playerControl, body GameMessageBody, firstInConversation bool) error {
   894  
   895  	gmw := GameMessageWrapped{
   896  		Sender:              d.dh.Me(),
   897  		Me:                  me,
   898  		FirstInConversation: firstInConversation,
   899  		Msg: GameMessageV1{
   900  			Md:   md,
   901  			Body: body,
   902  		},
   903  	}
   904  
   905  	// Only mark the forward bit to be true on messages that we can forward.
   906  	gmw.Forward = gmw.isForwardable()
   907  
   908  	// Reinject the message into the state machine.
   909  	d.chatInputCh <- &gmw
   910  
   911  	return nil
   912  }
   913  
   914  var DefaultCommitmentWindowMsec int64 = 3 * 1000
   915  var DefaultRevealWindowMsec int64 = 30 * 1000
   916  var DefaultCommitmentCompleteWindowMsec int64 = 15 * 1000
   917  var DefaultSlackMsec int64 = 1 * 1000
   918  
   919  // For bigger groups, everything is slower, like the time to digest all required messages. So we're
   920  // going to inflate our timeouts.
   921  func inflateTimeout(timeout int64, nPlayers int) int64 {
   922  	if nPlayers <= 5 {
   923  		return timeout
   924  	}
   925  	return int64(math.Ceil(math.Log(float64(nPlayers)) * float64(timeout) / math.Log(5.0)))
   926  }
   927  
   928  func newStart(now time.Time, nPlayers int) Start {
   929  	return Start{
   930  		StartTime:                    ToTime(now),
   931  		CommitmentWindowMsec:         inflateTimeout(DefaultCommitmentWindowMsec, nPlayers),
   932  		RevealWindowMsec:             inflateTimeout(DefaultRevealWindowMsec, nPlayers),
   933  		CommitmentCompleteWindowMsec: inflateTimeout(DefaultCommitmentCompleteWindowMsec, nPlayers),
   934  		SlackMsec:                    inflateTimeout(DefaultSlackMsec, nPlayers),
   935  	}
   936  }