func (c *context) SetChannelOffset(ch *nsqd.Channel, startFrom *ConsumeOffset, force bool) (int64, int64, error) { var l *consistence.CommitLogData var queueOffset int64 cnt := int64(0) var err error if startFrom.OffsetType == offsetTimestampType { if c.nsqdCoord != nil { l, queueOffset, cnt, err = c.nsqdCoord.SearchLogByMsgTimestamp(ch.GetTopicName(), ch.GetTopicPart(), startFrom.OffsetValue) } else { err = errors.New("Not supported while coordinator disabled") } } else if startFrom.OffsetType == offsetSpecialType { if startFrom.OffsetValue == -1 { e := ch.GetChannelEnd() queueOffset = int64(e.Offset()) cnt = e.TotalMsgCnt() } else { nsqd.NsqLogger().Logf("not known special offset :%v", startFrom) err = errors.New("not supported offset type") } } else if startFrom.OffsetType == offsetVirtualQueueType { queueOffset = startFrom.OffsetValue cnt = 0 if c.nsqdCoord != nil { l, queueOffset, cnt, err = c.nsqdCoord.SearchLogByMsgOffset(ch.GetTopicName(), ch.GetTopicPart(), queueOffset) } else { err = errors.New("Not supported while coordinator disabled") } } else if startFrom.OffsetType == offsetMsgCountType { if c.nsqdCoord != nil { l, queueOffset, cnt, err = c.nsqdCoord.SearchLogByMsgCnt(ch.GetTopicName(), ch.GetTopicPart(), startFrom.OffsetValue) } else { err = errors.New("Not supported while coordinator disabled") } } else { nsqd.NsqLogger().Logf("not supported offset type:%v", startFrom) err = errors.New("not supported offset type") } if err != nil { nsqd.NsqLogger().Logf("failed to search the consume offset: %v, err:%v", startFrom, err) return 0, 0, err } nsqd.NsqLogger().Logf("%v searched log : %v, offset: %v:%v", startFrom, l, queueOffset, cnt) if c.nsqdCoord == nil { err = ch.SetConsumeOffset(nsqd.BackendOffset(queueOffset), cnt, force) if err != nil { if err != nsqd.ErrSetConsumeOffsetNotFirstClient { nsqd.NsqLogger().Logf("failed to set the consume offset: %v, err:%v", startFrom, err) return 0, 0, err } nsqd.NsqLogger().Logf("the consume offset: %v can only be set by the first client", startFrom) } } else { err = c.nsqdCoord.SetChannelConsumeOffsetToCluster(ch, queueOffset, cnt, force) if err != nil { if coordErr, ok := err.(*consistence.CommonCoordErr); ok { if coordErr.IsEqual(consistence.ErrLocalSetChannelOffsetNotFirstClient) { nsqd.NsqLogger().Logf("the consume offset: %v can only be set by the first client", startFrom) return queueOffset, cnt, nil } } nsqd.NsqLogger().Logf("failed to set the consume offset: %v (%v:%v), err: %v ", startFrom, queueOffset, cnt, err) return 0, 0, err } } return queueOffset, cnt, nil }
func (self *NsqdCoordinator) updateChannelOffsetOnSlave(tc *coordData, channelName string, offset ChannelConsumerOffset) *CoordErr { topicName := tc.topicInfo.Name partition := tc.topicInfo.Partition if !tc.IsMineISR(self.myNode.GetID()) { return ErrTopicWriteOnNonISR } if coordLog.Level() >= levellogger.LOG_DETAIL { coordLog.Debugf("got update channel(%v) offset on slave : %v", channelName, offset) } coord, coordErr := self.getTopicCoord(topicName, partition) if coordErr != nil { return ErrMissingTopicCoord } topic, localErr := self.localNsqd.GetExistingTopic(topicName, partition) if localErr != nil { coordLog.Warningf("slave missing topic : %v", topicName) // TODO: leave the isr and try re-sync with leader return &CoordErr{localErr.Error(), RpcCommonErr, CoordSlaveErr} } if topic.GetTopicPart() != partition { coordLog.Errorf("topic on slave has different partition : %v vs %v", topic.GetTopicPart(), partition) return ErrLocalMissingTopic } var ch *nsqd.Channel ch, localErr = topic.GetExistingChannel(channelName) // if a new channel on slave, we should set the consume offset by force if localErr != nil { offset.AllowBackward = true ch = topic.GetChannel(channelName) coordLog.Infof("slave init the channel : %v, %v, offset: %v", topic.GetTopicName(), channelName, ch.GetConfirmed()) } if ch.IsEphemeral() { coordLog.Errorf("ephemeral channel %v should not be synced on slave", channelName) } currentEnd := ch.GetChannelEnd() if nsqd.BackendOffset(offset.VOffset) > currentEnd.Offset() { coordLog.Debugf("update channel(%v) consume offset exceed end %v on slave : %v", channelName, offset, currentEnd) // cache the offset (using map?) to reduce the slave channel flush. coord.consumeMgr.Lock() cur, ok := coord.consumeMgr.channelConsumeOffset[channelName] if !ok || cur.VOffset < offset.VOffset { coord.consumeMgr.channelConsumeOffset[channelName] = offset } coord.consumeMgr.Unlock() if offset.Flush { topic.ForceFlush() currentEnd = ch.GetChannelEnd() if nsqd.BackendOffset(offset.VOffset) > currentEnd.Offset() { offset.VOffset = int64(currentEnd.Offset()) offset.VCnt = currentEnd.TotalMsgCnt() } } else { return nil } } err := ch.ConfirmBackendQueueOnSlave(nsqd.BackendOffset(offset.VOffset), offset.VCnt, offset.AllowBackward) if err != nil { coordLog.Warningf("update local channel(%v) offset %v failed: %v, current channel end: %v, topic end: %v", channelName, offset, err, currentEnd, topic.TotalDataSize()) if err == nsqd.ErrExiting { return &CoordErr{err.Error(), RpcNoErr, CoordTmpErr} } return &CoordErr{err.Error(), RpcCommonErr, CoordSlaveErr} } return nil }