From 3d4cbdc975d341c1765050a9a59882ef89dda006 Mon Sep 17 00:00:00 2001 From: Tigran Najaryan <4194920+tigrannajaryan@users.noreply.github.com> Date: Thu, 28 Jul 2022 14:56:29 -0400 Subject: [PATCH] Correctly handle status compression on the server side (#105) We check the omitted field and the corresponding capability bit to understand if the compression was used. Related to spec issue https://github.com/open-telemetry/opamp-spec/issues/109 The issue was discarded in favour of the implementation that does not change the spec, but instead uses already existing information. --- client/internal/clientcommon.go | 7 +----- internal/examples/server/data/agent.go | 30 ++++++++++++++++++++------ 2 files changed, 25 insertions(+), 12 deletions(-) diff --git a/client/internal/clientcommon.go b/client/internal/clientcommon.go index 1a28c65f..bc8211f5 100644 --- a/client/internal/clientcommon.go +++ b/client/internal/clientcommon.go @@ -213,12 +213,7 @@ func (c *ClientCommon) PrepareFirstMessage(ctx context.Context) error { msg.EffectiveConfig = cfg msg.RemoteConfigStatus = c.ClientSyncedState.RemoteConfigStatus() msg.PackageStatuses = c.ClientSyncedState.PackageStatuses() - - if c.PackagesStateProvider != nil { - // We have a state provider, so package related capabilities can work. - msg.Capabilities |= protobufs.AgentCapabilities_AcceptsPackages - msg.Capabilities |= protobufs.AgentCapabilities_ReportsPackageStatuses - } + msg.Capabilities = c.Capabilities }, ) return nil diff --git a/internal/examples/server/data/agent.go b/internal/examples/server/data/agent.go index 65a86dfa..f684632f 100644 --- a/internal/examples/server/data/agent.go +++ b/internal/examples/server/data/agent.go @@ -194,17 +194,35 @@ func (agent *Agent) updateEffectiveConfig( } } +func (agent *Agent) hasCapability(capability protobufs.AgentCapabilities) bool { + return agent.Status.Capabilities&capability != 0 +} + func (agent *Agent) processStatusUpdate( newStatus *protobufs.AgentToServer, response *protobufs.ServerToAgent, ) { - if agent.Status != nil && agent.Status.SequenceNum+1 != newStatus.SequenceNum { - // We lost the previous status update. Request full status update from the agent. - response.Flags |= protobufs.ServerToAgent_ReportFullState - } + // We don't have any status for this Agent, or we lost the previous status update from the Agent, so our + // current status is not up-to-date. + lostPreviousUpdate := (agent.Status == nil) || (agent.Status != nil && agent.Status.SequenceNum+1 != newStatus.SequenceNum) agentDescrChanged := agent.updateStatusField(newStatus) + // Check if any fields were omitted in the status report. + effectiveConfigOmitted := agent.hasCapability(protobufs.AgentCapabilities_ReportsEffectiveConfig) && newStatus.EffectiveConfig == nil + packageStatusesOmitted := agent.hasCapability(protobufs.AgentCapabilities_ReportsPackageStatuses) && newStatus.PackageStatuses == nil + remoteConfigStatusOmitted := agent.hasCapability(protobufs.AgentCapabilities_AcceptsRemoteConfig) && newStatus.RemoteConfigStatus == nil + healthOmitted := agent.hasCapability(protobufs.AgentCapabilities_ReportsHealth) && newStatus.Health == nil + + // True if the status was not fully reported. + statusIsCompressed := effectiveConfigOmitted || packageStatusesOmitted || remoteConfigStatusOmitted || healthOmitted + + if statusIsCompressed && lostPreviousUpdate { + // The status message is not fully set in the message that we received, but we lost the previous + // status update. Request full status update from the agent. + response.Flags |= protobufs.ServerToAgent_ReportFullState + } + configChanged := false if agentDescrChanged { // Agent description is changed. @@ -219,8 +237,8 @@ func (agent *Agent) processStatusUpdate( // If remote config is changed and different from what the Agent has then // send the new remote config to the Agent. if configChanged || - (newStatus.RemoteConfigStatus != nil && - bytes.Compare(newStatus.RemoteConfigStatus.LastRemoteConfigHash, agent.remoteConfig.ConfigHash) != 0) { + (agent.Status.RemoteConfigStatus != nil && + bytes.Compare(agent.Status.RemoteConfigStatus.LastRemoteConfigHash, agent.remoteConfig.ConfigHash) != 0) { // The new status resulted in a change in the config of the Agent or the Agent // does not have this config (hash is different). Send the new config the Agent. response.RemoteConfig = agent.remoteConfig