Skip to content

Commit

Permalink
Added response header for traffic splitting feature (#401)
Browse files Browse the repository at this point in the history
* Added response header for traffic splitting feature
  • Loading branch information
nastassia-dailidava authored Dec 12, 2023
1 parent c4a0609 commit 977567b
Show file tree
Hide file tree
Showing 5 changed files with 108 additions and 8 deletions.
4 changes: 4 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,10 @@

Lists all changes with user impact.
The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/).
## [0.20.4]

### Changed
- Added possibility to add response header for weighted secondary cluster

## [0.20.3]

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -157,6 +157,7 @@ class CanaryProperties {

class TrafficSplittingProperties {
var zoneName = ""
var headerName = ""
var serviceByWeightsProperties: Map<String, ZoneWeights> = mapOf()
var secondaryClusterSuffix = "secondary"
var aggregateClusterSuffix = "aggregate"
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ import pl.allegro.tech.servicemesh.envoycontrol.snapshot.StandardRouteSpecificat
import pl.allegro.tech.servicemesh.envoycontrol.snapshot.WeightRouteSpecification
import pl.allegro.tech.servicemesh.envoycontrol.snapshot.resource.clusters.EnvoyClustersFactory.Companion.getAggregateClusterName
import pl.allegro.tech.servicemesh.envoycontrol.snapshot.resource.listeners.filters.ServiceTagFilterFactory
import java.lang.Boolean.TRUE
import pl.allegro.tech.servicemesh.envoycontrol.groups.RetryPolicy as EnvoyControlRetryPolicy

class EnvoyEgressRoutesFactory(
Expand Down Expand Up @@ -359,7 +360,8 @@ class EnvoyEgressRoutesFactory(
.withClusterWeight(routeSpec.clusterName, routeSpec.clusterWeights.main)
.withClusterWeight(
getAggregateClusterName(routeSpec.clusterName, properties),
routeSpec.clusterWeights.secondary
routeSpec.clusterWeights.secondary,
true
)
)
}
Expand All @@ -369,15 +371,40 @@ class EnvoyEgressRoutesFactory(
}
}

private fun WeightedCluster.Builder.withClusterWeight(clusterName: String, weight: Int): WeightedCluster.Builder {
this.addClusters(
WeightedCluster.ClusterWeight.newBuilder()
.setName(clusterName)
.setWeight(UInt32Value.of(weight))
.build()
)
private fun WeightedCluster.Builder.withClusterWeight(
clusterName: String,
weight: Int,
withHeader: Boolean = false
): WeightedCluster.Builder {
val clusters = WeightedCluster.ClusterWeight.newBuilder()
.setName(clusterName)
.setWeight(UInt32Value.of(weight))
.also {
if (withHeader) {
it.withHeader(properties.loadBalancing.trafficSplitting.headerName)
}
}
return this.addClusters(clusters)
}

private fun WeightedCluster.ClusterWeight.Builder.withHeader(key: String?): WeightedCluster.ClusterWeight.Builder {
key?.takeIf { it.isNotBlank() }
?.let {
this.addResponseHeadersToAdd(buildHeader(key))
}
return this
}

private fun buildHeader(key: String): HeaderValueOption.Builder {
return HeaderValueOption.newBuilder()
.setHeader(
HeaderValue.newBuilder()
.setKey(key)
.setValue(TRUE.toString())
)
.setAppendAction(HeaderValueOption.HeaderAppendAction.OVERWRITE_IF_EXISTS_OR_ADD)
.setKeepEmptyValue(false)
}
}

class RequestPolicyMapper private constructor() {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import io.envoyproxy.envoy.config.route.v3.RouteAction
import io.envoyproxy.envoy.config.route.v3.RouteConfiguration
import io.envoyproxy.envoy.config.route.v3.VirtualCluster
import io.envoyproxy.envoy.config.route.v3.VirtualHost
import io.envoyproxy.envoy.config.route.v3.WeightedCluster
import io.envoyproxy.envoy.type.matcher.v3.RegexMatcher
import org.assertj.core.api.Assertions.assertThat
import pl.allegro.tech.servicemesh.envoycontrol.snapshot.LocalRetryPolicyProperties
Expand All @@ -27,6 +28,16 @@ fun RouteConfiguration.hasVirtualHostThat(name: String, condition: VirtualHost.(
return this
}

fun RouteConfiguration.hasClusterThat(name: String, condition: WeightedCluster.ClusterWeight?.() -> Unit): RouteConfiguration {
condition(this.virtualHostsList
.flatMap { it.routesList }
.map { it.route }
.flatMap { route -> route.weightedClusters.clustersList }
.find { it.name == name }
)
return this
}

fun RouteConfiguration.hasRequestHeaderToAdd(key: String, value: String): RouteConfiguration {
assertThat(this.requestHeadersToAddList).anySatisfy {
assertThat(it.header.key).isEqualTo(key)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,13 @@ package pl.allegro.tech.servicemesh.envoycontrol.snapshot.resource.routes

import com.google.protobuf.util.Durations
import org.junit.jupiter.api.Assertions.assertEquals
import org.junit.jupiter.api.Assertions.assertNotNull
import org.junit.jupiter.api.Assertions.assertTrue
import org.junit.jupiter.api.Test
import pl.allegro.tech.servicemesh.envoycontrol.groups.DependencySettings
import pl.allegro.tech.servicemesh.envoycontrol.groups.Outgoing
import pl.allegro.tech.servicemesh.envoycontrol.groups.RetryPolicy
import pl.allegro.tech.servicemesh.envoycontrol.groups.hasClusterThat
import pl.allegro.tech.servicemesh.envoycontrol.groups.hasCustomIdleTimeout
import pl.allegro.tech.servicemesh.envoycontrol.groups.hasCustomRequestTimeout
import pl.allegro.tech.servicemesh.envoycontrol.groups.hasHostRewriteHeader
Expand All @@ -23,6 +26,8 @@ import pl.allegro.tech.servicemesh.envoycontrol.groups.matchingOnMethod
import pl.allegro.tech.servicemesh.envoycontrol.groups.matchingOnPrefix
import pl.allegro.tech.servicemesh.envoycontrol.snapshot.SnapshotProperties
import pl.allegro.tech.servicemesh.envoycontrol.snapshot.StandardRouteSpecification
import pl.allegro.tech.servicemesh.envoycontrol.snapshot.WeightRouteSpecification
import pl.allegro.tech.servicemesh.envoycontrol.snapshot.ZoneWeights

internal class EnvoyEgressRoutesFactoryTest {

Expand All @@ -42,6 +47,15 @@ internal class EnvoyEgressRoutesFactoryTest {
)
)

val weightedClusters = listOf(
WeightRouteSpecification(
clusterName = "srv1",
routeDomains = listOf("srv1"),
settings = DependencySettings(),
ZoneWeights()
)
)

@Test
fun `should add client identity header if incoming permissions are enabled`() {
// given
Expand Down Expand Up @@ -278,4 +292,47 @@ internal class EnvoyEgressRoutesFactoryTest {
defaultRoute.matchingOnPrefix("/")
}
}

@Test
fun `should add traffic splitting header for secondary weighted cluster`() {

val expectedHeaderKey = "test-header"
val routesFactory = EnvoyEgressRoutesFactory(SnapshotProperties().apply {
loadBalancing.trafficSplitting.headerName = expectedHeaderKey
})

val routeConfig = routesFactory.createEgressRouteConfig(
"client1",
weightedClusters,
false
)

routeConfig
.hasClusterThat("srv1-aggregate") {
assertNotNull(this)
assertNotNull(this!!.responseHeadersToAddList.find { it.header.key == expectedHeaderKey })
}.hasClusterThat("srv1") {
assertNotNull(this)
assertTrue(this!!.responseHeadersToAddList.isEmpty())
}
}

@Test
fun `should not add traffic splitting header if header key is not set`() {
val routesFactory = EnvoyEgressRoutesFactory(SnapshotProperties())
val routeConfig = routesFactory.createEgressRouteConfig(
"client1",
weightedClusters,
false
)

routeConfig
.hasClusterThat("srv1-aggregate") {
assertNotNull(this)
assertTrue(this!!.responseHeadersToAddList.isEmpty())
}.hasClusterThat("srv1") {
assertNotNull(this)
assertNotNull(this!!.responseHeadersToAddList.isEmpty())
}
}
}

0 comments on commit 977567b

Please sign in to comment.