From af5d1680a29968162f38845de82b5fee6b0d312d Mon Sep 17 00:00:00 2001 From: gran Date: Mon, 13 Jul 2020 16:56:46 +0800 Subject: [PATCH] Support service destination in traceflow --- build/yamls/antrea-aks.yml | 5 ++ build/yamls/antrea-eks.yml | 5 ++ build/yamls/antrea-gke.yml | 5 ++ build/yamls/antrea-ipsec.yml | 5 ++ build/yamls/antrea.yml | 5 ++ build/yamls/base/crds.yml | 3 + cmd/antrea-agent/agent.go | 1 + cmd/antrea-controller/controller.go | 2 +- pkg/agent/controller/traceflow/packetin.go | 34 ++++++++ .../traceflow/traceflow_controller.go | 51 +++++++++-- pkg/agent/openflow/pipeline.go | 15 +++- pkg/controller/traceflow/controller.go | 42 +++++++++- test/e2e/traceflow_test.go | 84 ++++++++++++++++++- 13 files changed, 239 insertions(+), 18 deletions(-) diff --git a/build/yamls/antrea-aks.yml b/build/yamls/antrea-aks.yml index 644f103a84f..8003384edbb 100644 --- a/build/yamls/antrea-aks.yml +++ b/build/yamls/antrea-aks.yml @@ -223,6 +223,9 @@ spec: - required: - pod - namespace + - required: + - service + - namespace - required: - ip properties: @@ -233,6 +236,8 @@ spec: type: string pod: type: string + service: + type: string type: object packet: properties: diff --git a/build/yamls/antrea-eks.yml b/build/yamls/antrea-eks.yml index 5095efd120f..d76f16489b6 100644 --- a/build/yamls/antrea-eks.yml +++ b/build/yamls/antrea-eks.yml @@ -223,6 +223,9 @@ spec: - required: - pod - namespace + - required: + - service + - namespace - required: - ip properties: @@ -233,6 +236,8 @@ spec: type: string pod: type: string + service: + type: string type: object packet: properties: diff --git a/build/yamls/antrea-gke.yml b/build/yamls/antrea-gke.yml index 79e318fbddd..0bd23abe120 100644 --- a/build/yamls/antrea-gke.yml +++ b/build/yamls/antrea-gke.yml @@ -223,6 +223,9 @@ spec: - required: - pod - namespace + - required: + - service + - namespace - required: - ip properties: @@ -233,6 +236,8 @@ spec: type: string pod: type: string + service: + type: string type: object packet: properties: diff --git a/build/yamls/antrea-ipsec.yml b/build/yamls/antrea-ipsec.yml index 340a47db0a5..8fb02cfa9f1 100644 --- a/build/yamls/antrea-ipsec.yml +++ b/build/yamls/antrea-ipsec.yml @@ -223,6 +223,9 @@ spec: - required: - pod - namespace + - required: + - service + - namespace - required: - ip properties: @@ -233,6 +236,8 @@ spec: type: string pod: type: string + service: + type: string type: object packet: properties: diff --git a/build/yamls/antrea.yml b/build/yamls/antrea.yml index 842a1d6e9ed..d6bc87abbc2 100644 --- a/build/yamls/antrea.yml +++ b/build/yamls/antrea.yml @@ -223,6 +223,9 @@ spec: - required: - pod - namespace + - required: + - service + - namespace - required: - ip properties: @@ -233,6 +236,8 @@ spec: type: string pod: type: string + service: + type: string type: object packet: properties: diff --git a/build/yamls/base/crds.yml b/build/yamls/base/crds.yml index bf0c36fa014..49a3e51b3ca 100644 --- a/build/yamls/base/crds.yml +++ b/build/yamls/base/crds.yml @@ -104,6 +104,8 @@ spec: properties: pod: type: string + service: + type: string namespace: type: string ip: @@ -111,6 +113,7 @@ spec: format: ipv4 oneOf: - required: ["pod", "namespace"] + - required: ["service", "namespace"] - required: ["ip"] packet: type: object diff --git a/cmd/antrea-agent/agent.go b/cmd/antrea-agent/agent.go index 85905ff39f8..30bd58e8b52 100644 --- a/cmd/antrea-agent/agent.go +++ b/cmd/antrea-agent/agent.go @@ -138,6 +138,7 @@ func run(o *Options) error { if features.DefaultFeatureGate.Enabled(features.Traceflow) { traceflowController = traceflow.NewTraceflowController( k8sClient, + informerFactory, crdClient, traceflowInformer, ofClient, diff --git a/cmd/antrea-controller/controller.go b/cmd/antrea-controller/controller.go index 34ccad7808b..b8616f1f864 100644 --- a/cmd/antrea-controller/controller.go +++ b/cmd/antrea-controller/controller.go @@ -93,7 +93,7 @@ func run(o *Options) error { var traceflowController *traceflow.Controller if features.DefaultFeatureGate.Enabled(features.Traceflow) { - traceflowController = traceflow.NewTraceflowController(crdClient, traceflowInformer) + traceflowController = traceflow.NewTraceflowController(crdClient, podInformer, traceflowInformer) } apiServerConfig, err := createAPIServerConfig(o.config.ClientConnection.Kubeconfig, diff --git a/pkg/agent/controller/traceflow/packetin.go b/pkg/agent/controller/traceflow/packetin.go index 98ca4e8ccc7..d14d40c87cf 100644 --- a/pkg/agent/controller/traceflow/packetin.go +++ b/pkg/agent/controller/traceflow/packetin.go @@ -22,6 +22,7 @@ import ( "time" "github.com/contiv/libOpenflow/openflow13" + "github.com/contiv/libOpenflow/protocol" "github.com/contiv/ofnet/ofctrl" v1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/client-go/util/retry" @@ -100,6 +101,27 @@ func (c *Controller) parsePacketIn(pktIn *ofctrl.PacketIn) (*opsv1alpha1.Tracefl obs = append(obs, *ob) } + // Collect Service DNAT. + if pktIn.Data.Ethertype == 0x800 { + ipPacket, ok := pktIn.Data.Data.(*protocol.IPv4) + if !ok { + return nil, nil, errors.New("invalid traceflow IPv4 packet") + } + ctNwDst, err := getInfoInCtNwDstField(matchers) + if err != nil { + return nil, nil, err + } + ipDst := ipPacket.NWDst.String() + if ctNwDst != "" && ipDst != ctNwDst { + ob := &opsv1alpha1.Observation{ + Component: opsv1alpha1.LB, + Action: opsv1alpha1.Forwarded, + TranslatedDstIP: ipDst, + } + obs = append(obs, *ob) + } + } + // Collect egress conjunctionID and get NetworkPolicy from cache. if match = getMatchRegField(matchers, uint32(openflow.EgressReg)); match != nil { egressInfo, err := getInfoInReg(match, nil) @@ -194,3 +216,15 @@ func getInfoInTunnelDst(regMatch *ofctrl.MatchField) (string, error) { } return regValue.String(), nil } + +func getInfoInCtNwDstField(matchers *ofctrl.Matchers) (string, error) { + match := matchers.GetMatchByName("NXM_NX_CT_NW_DST") + if match == nil { + return "", nil + } + regValue, ok := match.GetValue().(net.IP) + if !ok { + return "", errors.New("packet-in conntrack IP destination value cannot be retrieved from metadata") + } + return regValue.String(), nil +} diff --git a/pkg/agent/controller/traceflow/traceflow_controller.go b/pkg/agent/controller/traceflow/traceflow_controller.go index cf0ecd037a2..d2da98d7f86 100644 --- a/pkg/agent/controller/traceflow/traceflow_controller.go +++ b/pkg/agent/controller/traceflow/traceflow_controller.go @@ -26,7 +26,9 @@ import ( metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/types" "k8s.io/apimachinery/pkg/util/wait" + "k8s.io/client-go/informers" clientset "k8s.io/client-go/kubernetes" + corelisters "k8s.io/client-go/listers/core/v1" "k8s.io/client-go/tools/cache" "k8s.io/client-go/util/workqueue" "k8s.io/klog/v2" @@ -38,6 +40,7 @@ import ( clientsetversioned "github.com/vmware-tanzu/antrea/pkg/client/clientset/versioned" opsinformers "github.com/vmware-tanzu/antrea/pkg/client/informers/externalversions/ops/v1alpha1" opslisters "github.com/vmware-tanzu/antrea/pkg/client/listers/ops/v1alpha1" + "github.com/vmware-tanzu/antrea/pkg/features" "github.com/vmware-tanzu/antrea/pkg/ovs/ovsconfig" ) @@ -65,6 +68,8 @@ const ( // the switch for traceflow request. type Controller struct { kubeClient clientset.Interface + serviceLister corelisters.ServiceLister + serviceListerSynced cache.InformerSynced traceflowClient clientsetversioned.Interface traceflowInformer opsinformers.TraceflowInformer traceflowLister opslisters.TraceflowLister @@ -85,6 +90,7 @@ type Controller struct { // events. func NewTraceflowController( kubeClient clientset.Interface, + informerFactory informers.SharedInformerFactory, traceflowClient clientsetversioned.Interface, traceflowInformer opsinformers.TraceflowInformer, client openflow.Client, @@ -118,6 +124,11 @@ func NewTraceflowController( ) // Register packetInHandler c.ofClient.RegisterPacketInHandler("traceflow", c) + // Add serviceLister if AntreaProxy enabled + if features.DefaultFeatureGate.Enabled(features.AntreaProxy) { + c.serviceLister = informerFactory.Core().V1().Services().Lister() + c.serviceListerSynced = informerFactory.Core().V1().Services().Informer().HasSynced + } return c } @@ -135,6 +146,12 @@ func (c *Controller) Run(stopCh <-chan struct{}) { defer klog.Infof("Shutting down %s", controllerName) klog.Infof("Waiting for caches to sync for %s", controllerName) + if features.DefaultFeatureGate.Enabled(features.AntreaProxy) { + if !cache.WaitForCacheSync(stopCh, c.serviceListerSynced) { + klog.Errorf("Unable to sync service cache for %s", controllerName) + return + } + } if !cache.WaitForCacheSync(stopCh, c.traceflowListerSynced) { klog.Errorf("Unable to sync caches for %s", controllerName) return @@ -244,17 +261,21 @@ func (c *Controller) syncTraceflow(traceflowName string) error { // startTraceflow deploys OVS flow entries for Traceflow and inject packet if current Node // is Sender Node. func (c *Controller) startTraceflow(tf *opsv1alpha1.Traceflow) error { - // Deploy flow entries for traceflow - klog.V(2).Infof("Deploy flow entries for Traceflow %s", tf.Name) - err := c.ofClient.InstallTraceflowFlows(tf.Status.DataplaneTag) + err := c.validateTraceflow(tf) defer func() { if err != nil { - c.errorTraceflowCRD(tf, fmt.Sprintf("Node: %s, error: %+v", tf.Name, err)) + c.errorTraceflowCRD(tf, fmt.Sprintf("Node: %s, error: %+v", c.nodeConfig.Name, err)) } }() if err != nil { return err } + // Deploy flow entries for traceflow + klog.V(2).Infof("Deploy flow entries for Traceflow %s", tf.Name) + err = c.ofClient.InstallTraceflowFlows(tf.Status.DataplaneTag) + if err != nil { + return err + } // TODO: let controller compute the source Node, and the source Node can just return an error, // if fails to find the Pod. @@ -268,6 +289,13 @@ func (c *Controller) startTraceflow(tf *opsv1alpha1.Traceflow) error { return err } +func (c *Controller) validateTraceflow(tf *opsv1alpha1.Traceflow) error { + if tf.Spec.Destination.Service != "" && !features.DefaultFeatureGate.Enabled(features.AntreaProxy) { + return errors.New("using Service destination requires AntreaProxy feature enabled") + } + return nil +} + func (c *Controller) injectPacket(tf *opsv1alpha1.Traceflow) error { podInterfaces := c.interfaceStore.GetContainerInterfacesByPod(tf.Spec.Source.Pod, tf.Spec.Source.Namespace) // Update Traceflow phase to Running. @@ -285,7 +313,7 @@ func (c *Controller) injectPacket(tf *opsv1alpha1.Traceflow) error { if hasInterface { dstMAC = dstPodInterface.MAC.String() } - } else { + } else if tf.Spec.Destination.Pod != "" { dstPodInterfaces := c.interfaceStore.GetContainerInterfacesByPod(tf.Spec.Destination.Pod, tf.Spec.Destination.Namespace) if len(dstPodInterfaces) > 0 { dstMAC = dstPodInterfaces[0].MAC.String() @@ -299,11 +327,18 @@ func (c *Controller) injectPacket(tf *opsv1alpha1.Traceflow) error { dstIP = dstPod.Status.PodIP dstNodeIP = dstPod.Status.HostIP } + } else if tf.Spec.Destination.Service != "" { + dstSvc, err := c.serviceLister.Services(tf.Spec.Destination.Namespace).Get(tf.Spec.Destination.Service) + if err != nil { + return err + } + dstIP = dstSvc.Spec.ClusterIP } - if dstNodeIP != "" { + // Check encap status if no dstMAC found which means the destination is Service or the destination Pod/IP is not on local Node. + if dstMAC == "" { peerIP := net.ParseIP(dstNodeIP) - if c.networkConfig.TunnelType == ovsconfig.GeneveTunnel && peerIP != nil && c.networkConfig.TrafficEncapMode.NeedsEncapToPeer(peerIP, c.nodeConfig.NodeIPAddr) { - // Wait a small period for other Nodes. + if c.networkConfig.TunnelType == ovsconfig.GeneveTunnel && (tf.Spec.Destination.Pod == "" || c.networkConfig.TrafficEncapMode.NeedsEncapToPeer(peerIP, c.nodeConfig.NodeIPAddr)) { + // If the destination is Service/IP or the packet will be encapsulated to remote Node, wait a small period for other Nodes. time.Sleep(time.Duration(injectPacketDelay) * time.Second) } else { // Inter-node traceflow is only available when the packet is encapsulated in Geneve tunnel. diff --git a/pkg/agent/openflow/pipeline.go b/pkg/agent/openflow/pipeline.go index f29116cf08c..86fac5b3081 100644 --- a/pkg/agent/openflow/pipeline.go +++ b/pkg/agent/openflow/pipeline.go @@ -513,12 +513,19 @@ func (c *client) connectionTrackFlows(category cookie.Category) []binding.Flow { // avoid unexpected packet drop in Traceflow. func (c *client) traceflowConnectionTrackFlows(dataplaneTag uint8, category cookie.Category) binding.Flow { connectionTrackStateTable := c.pipeline[conntrackStateTable] - return connectionTrackStateTable.BuildFlow(priorityNormal+2). + flowBuilder := connectionTrackStateTable.BuildFlow(priorityLow+2). MatchRegRange(int(TraceflowReg), uint32(dataplaneTag), OfTraceflowMarkRange). SetHardTimeout(300). - Action().ResubmitToTable(connectionTrackStateTable.GetNext()). - Cookie(c.cookieAllocator.Request(category).Raw()). - Done() + Cookie(c.cookieAllocator.Request(category).Raw()) + if c.enableProxy { + flowBuilder = flowBuilder. + Action().ResubmitToTable(sessionAffinityTable). + Action().ResubmitToTable(serviceLBTable) + } else { + flowBuilder = flowBuilder. + Action().ResubmitToTable(connectionTrackStateTable.GetNext()) + } + return flowBuilder.Done() } // reEntranceBypassCTFlow generates flow that bypass CT for traffic re-entering host network space. diff --git a/pkg/controller/traceflow/controller.go b/pkg/controller/traceflow/controller.go index 462a200b54a..69d4604f25b 100644 --- a/pkg/controller/traceflow/controller.go +++ b/pkg/controller/traceflow/controller.go @@ -18,13 +18,16 @@ import ( "context" "encoding/json" "errors" + "fmt" "sync" "time" + corev1 "k8s.io/api/core/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/labels" "k8s.io/apimachinery/pkg/types" "k8s.io/apimachinery/pkg/util/wait" + coreinformers "k8s.io/client-go/informers/core/v1" "k8s.io/client-go/tools/cache" "k8s.io/client-go/util/workqueue" "k8s.io/klog/v2" @@ -48,6 +51,9 @@ const ( // dataplaneTag=15 is reserved. minTagNum uint8 = 1 maxTagNum uint8 = 14 + + // PodIP index name for Pod cache. + podIPIndex = "podIP" ) var ( @@ -58,6 +64,7 @@ var ( // Controller is for traceflow. type Controller struct { client versioned.Interface + podInformer coreinformers.PodInformer traceflowInformer opsinformers.TraceflowInformer traceflowLister opslisters.TraceflowLister traceflowListerSynced cache.InformerSynced @@ -66,10 +73,11 @@ type Controller struct { runningTraceflows map[uint8]string // tag->traceflowName if tf.Status.Phase is Running. } -// NewTraceflowController creates a new traceflow controller. -func NewTraceflowController(client versioned.Interface, traceflowInformer opsinformers.TraceflowInformer) *Controller { +// NewTraceflowController creates a new traceflow controller and adds podIP indexer to podInformer. +func NewTraceflowController(client versioned.Interface, podInformer coreinformers.PodInformer, traceflowInformer opsinformers.TraceflowInformer) *Controller { c := &Controller{ client: client, + podInformer: podInformer, traceflowInformer: traceflowInformer, traceflowLister: traceflowInformer.Lister(), traceflowListerSynced: traceflowInformer.Informer().HasSynced, @@ -84,9 +92,21 @@ func NewTraceflowController(client versioned.Interface, traceflowInformer opsinf }, resyncPeriod, ) + podInformer.Informer().AddIndexers(cache.Indexers{podIPIndex: podIPIndexFunc}) return c } +func podIPIndexFunc(obj interface{}) ([]string, error) { + pod, ok := obj.(*corev1.Pod) + if !ok { + return nil, fmt.Errorf("obj is not pod: %+v", obj) + } + if pod.Status.PodIP != "" && pod.Status.Phase != corev1.PodSucceeded && pod.Status.Phase != corev1.PodFailed { + return []string{pod.Status.PodIP}, nil + } + return nil, nil +} + // enqueueTraceflow adds an object to the controller work queue. func (c *Controller) enqueueTraceflow(tf *opsv1alpha1.Traceflow) { c.queue.Add(tf.Name) @@ -224,14 +244,28 @@ func (c *Controller) checkTraceflowStatus(tf *opsv1alpha1.Traceflow) (retry bool retry = false sender := false receiver := false - for _, nodeResult := range tf.Status.Results { - for _, ob := range nodeResult.Observations { + for i, nodeResult := range tf.Status.Results { + for j, ob := range nodeResult.Observations { if ob.Component == opsv1alpha1.SpoofGuard { sender = true } if ob.Action == opsv1alpha1.Delivered || ob.Action == opsv1alpha1.Dropped { receiver = true } + if ob.TranslatedDstIP != "" { + // Add Pod ns/name to observation if TranslatedDstIP (a.k.a. Service Endpoint address) is Pod IP. + pods, err := c.podInformer.Informer().GetIndexer().ByIndex("podIP", ob.TranslatedDstIP) + if err != nil { + klog.Infof("Unable to find Pod from IP, error: %+v", err) + } else if len(pods) > 0 { + pod, ok := pods[0].(*corev1.Pod) + if !ok { + klog.Warningf("Invalid Pod obj in cache") + } else { + tf.Status.Results[i].Observations[j].Pod = fmt.Sprintf("%s/%s", pod.Namespace, pod.Name) + } + } + } } } if sender && receiver { diff --git a/test/e2e/traceflow_test.go b/test/e2e/traceflow_test.go index 29ad19c8e97..1285c1ea295 100644 --- a/test/e2e/traceflow_test.go +++ b/test/e2e/traceflow_test.go @@ -21,6 +21,7 @@ import ( "testing" "time" + "github.com/stretchr/testify/require" networkingv1 "k8s.io/api/networking/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/util/wait" @@ -58,6 +59,12 @@ func TestTraceflow(t *testing.T) { defer node1CleanupFn() defer node2CleanupFn() + require.NoError(t, data.createNginxPod("nginx", node2)) + nginxIP, err := data.podWaitForIP(defaultTimeout, "nginx", testNamespace) + require.NoError(t, err) + svc, err := data.createNginxClusterIPService(false) + require.NoError(t, err) + // Setup 2 NetworkPolicies: // 1. Allow all egress traffic. // 2. Deny ingress traffic on pod with label antrea-e2e = node1Pods[1]. So flow node1Pods[0] -> node1Pods[1] will be dropped. @@ -96,6 +103,7 @@ func TestTraceflow(t *testing.T) { // 2. node1Pods[0] -> node2Pods[0], inter node1 and node2. // 3. node1Pods[0] -> node1IPs[1], intra node1. // 4. node1Pods[0] -> node2IPs[0], inter node1 and node2. + // 5. node1Pods[0] -> service, inter node1 and node2. testcases := []testcase{ { name: "intraNodeTraceflow", @@ -320,13 +328,84 @@ func TestTraceflow(t *testing.T) { }, }, }, + { + name: "serviceTraceflow", + tf: &v1alpha1.Traceflow{ + ObjectMeta: metav1.ObjectMeta{ + Name: randName(fmt.Sprintf("%s-%s-to-svc-%s-", testNamespace, node1Pods[0], svc.Name)), + }, + Spec: v1alpha1.TraceflowSpec{ + Source: v1alpha1.Source{ + Namespace: testNamespace, + Pod: node1Pods[0], + }, + Destination: v1alpha1.Destination{ + Namespace: testNamespace, + Service: svc.Name, + }, + Packet: v1alpha1.Packet{ + IPHeader: v1alpha1.IPHeader{ + Protocol: 6, + }, + TransportHeader: v1alpha1.TransportHeader{ + TCP: &v1alpha1.TCPHeader{ + DstPort: 80, + Flags: 2, + }, + }, + }, + }, + }, + expectedPhase: v1alpha1.Succeeded, + expectedResults: []v1alpha1.NodeResult{ + { + Node: node1, + Observations: []v1alpha1.Observation{ + { + Component: v1alpha1.SpoofGuard, + Action: v1alpha1.Forwarded, + }, + { + Component: v1alpha1.LB, + Pod: fmt.Sprintf("%s/%s", testNamespace, "nginx"), + TranslatedDstIP: nginxIP, + Action: v1alpha1.Forwarded, + }, + { + Component: v1alpha1.NetworkPolicy, + ComponentInfo: "EgressRule", + Action: v1alpha1.Forwarded, + }, + { + Component: v1alpha1.Forwarding, + ComponentInfo: "Output", + Action: v1alpha1.Forwarded, + }, + }, + }, + { + Node: node2, + Observations: []v1alpha1.Observation{ + { + Component: v1alpha1.Forwarding, + ComponentInfo: "Classification", + Action: v1alpha1.Received, + }, + { + Component: v1alpha1.Forwarding, + ComponentInfo: "Output", + Action: v1alpha1.Delivered, + }, + }, + }, + }, + }, } t.Run("traceflowGroupTest", func(t *testing.T) { for _, tc := range testcases { tc := tc t.Run(tc.name, func(t *testing.T) { - t.Parallel() if _, err := data.crdClient.OpsV1alpha1().Traceflows().Create(context.TODO(), tc.tf, metav1.CreateOptions{}); err != nil { t.Fatalf("Error when creating traceflow: %v", err) } @@ -404,6 +483,7 @@ func (data *TestData) enableTraceflow(t *testing.T) error { configMap.Data["antrea-controller.conf"] = antreaControllerConf antreaAgentConf, _ := configMap.Data["antrea-agent.conf"] antreaAgentConf = strings.Replace(antreaAgentConf, "# Traceflow: false", " Traceflow: true", 1) + antreaAgentConf = strings.Replace(antreaAgentConf, "# AntreaProxy: false", " AntreaProxy: true", 1) antreaAgentConf = strings.Replace(antreaAgentConf, "#tunnelType: geneve", "tunnelType: geneve", 1) configMap.Data["antrea-agent.conf"] = antreaAgentConf @@ -435,6 +515,8 @@ func compareObservations(expected v1alpha1.NodeResult, actual v1alpha1.NodeResul for i := 0; i < len(exObs); i++ { if exObs[i].Component != acObs[i].Component || exObs[i].ComponentInfo != acObs[i].ComponentInfo || + exObs[i].Pod != acObs[i].Pod || + exObs[i].TranslatedDstIP != acObs[i].TranslatedDstIP || exObs[i].Action != acObs[i].Action { return fmt.Errorf("Observations should be %v, but got %v", exObs, acObs) }