Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

don't remove representative peers in any-namespace #352

Merged
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
7 changes: 7 additions & 0 deletions pkg/netpol/connlist/connlist_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -913,4 +913,11 @@ var goodPathTests = []struct {
focusWorkload: "ratings-v1-b6994bb9",
outputFormats: ExposureValidFormats,
},
{
// test that when the rule enable any-namespace with podSelector, a representative peer is created even
// if there is a matching pod in a specific namespace
testDirName: "test_exposure_to_any_namespace_with_podSelector",
exposureAnalysis: true,
outputFormats: ExposureValidFormats,
},
}
7 changes: 7 additions & 0 deletions pkg/netpol/eval/exposure.go
Original file line number Diff line number Diff line change
Expand Up @@ -68,12 +68,19 @@ func (pe *PolicyEngine) extractLabelsAndRefineRepresentativePeers(podObj *k8s.Po

// refineRepresentativePeersMatchingLabels removes from the policy engine all representative peers
// with labels matching the given labels of a real pod
// representative peers matching any-namespace will not be removed.
func (pe *PolicyEngine) refineRepresentativePeersMatchingLabels(realPodLabels, realNsLabels map[string]string) {
keysToDelete := make([]string, 0)
// look for representative peers with labels matching the given real pod's (and its namespace) labels
for key, peer := range pe.representativePeersMap {
potentialPodSelector := labels.SelectorFromSet(labels.Set(peer.Pod.Labels))
potentialNsSelector := labels.SelectorFromSet(labels.Set(peer.PotentialNamespaceLabels))
if potentialNsSelector.Empty() { // representative peer that matches any-namespace, will not be removed
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

does it differentiate between

namespaceSelector: {}
podSelector: {k: v}

(which is any namespace)

to

podSelector: {k: v}

(which is the network policy namespace)?

Copy link
Collaborator

@adisos adisos May 15, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

and what about

namespaceSelector: {k: v}
podSelector: {}

this creates exposure to all pods in a certain namespace..
does this require indication as exposure?
it's not as matching pod with certain labels..

do we have any example test capturing this case with current implementation?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

does it differentiate between

namespaceSelector: {}
podSelector: {k: v}

(which is any namespace)

to

podSelector: {k: v}

(which is the network policy namespace)?

yes, it does
examples:

  1. tests/test_conn_with_pod_selector_in_any_ns with :
  • from:
    - namespaceSelector: {}
    podSelector:
    matchLabels:
    role: monitoring

(output: test_outputs\connlist\exposure_test_conn_with_pod_selector_in_any_ns_connlist_output.dot.png)

  1. tests/test_conn_with_only_pod_selector
  • from:
    - podSelector:
    matchLabels:
    role: monitoring
    (output:test_outputs\connlist\exposure_test_conn_with_only_pod_selector_connlist_output.dot.png)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

and what about

namespaceSelector: {k: v}
podSelector: {}

this creates exposure to all pods in a certain namespace.. does this require indication as exposure? it's not as matching pod with certain labels..

do we have any example test capturing this case with current implementation?

we have following example:
test: tests\test_conn_to_all_pods_in_a_new_ns
with:

  • from:
    - namespaceSelector:
    matchLabels:
    kubernetes.io/metadata.name: backend
    podSelector: {}

output: test_outputs\connlist\exposure_test_conn_to_all_pods_in_a_new_ns_connlist_output.dot.png

in this example the backend ns is not in the manifests and we have no real pods in this ns so we see an exposure to representative peers backend\[all pods].

if we had a real pod in the namespace backend : it would be a match and we won't see the exposure line.(like all cases where the podSelector was not used, if there is a pod in the namespace the it is a match)

Copy link
Collaborator

@adisos adisos May 16, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

in this example the backend ns is not in the manifests and we have no real pods in this ns

I would add another example where there is a real pod in the namespace backend

if we had a real pod in the namespace backend : it would be a match and we won't see the exposure line.

ok, the question is - should we consider changing this behavior (when the policy permits all pods in a real namespace)? if all pods in a namespace are permitted, it might be relevant as exposure info, even if there is a certain real pod in the namespace. (BTW there could be a real pod in that namespace on which there is a certain policy that blocks such a connection for this pod.. in such a case the connection to this pod is not in the report, nor the exposure to this namespace). [consider adding such an example as well]

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I meant to a policy that selects all pods in its namespace.. (let it be foo namespace) let's assume we do not have a pod in that namespace but another pod (bar/pod1) has exposure to this namespace (e.g. policy that permits egress to foo) the question is - do we consider such policy on a representative peer (namespace)? (for example, a policy that is a default deny for the foo namespace)

I will add an example to see the behavior and update

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

added an example locally, the exposure analysis of bar still shows the line:
bar/workload-a[Deployment] => foo/[all pods] : All Connections
even that we have default-deny-in-foo for ingress

should fix the bug in this PR?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

create new item and implement in a separate PR .
analysis should generate representative peers by also considering policies that capture them, and separate into sub-sections (sub-peers) that are having different connectivity by the policies (for example pods in ns with certain labels, and pods in ns without these labels).

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

if we had a real pod in the namespace backend : it would be a match and we won't see the exposure line.

ok, the question is - should we consider changing this behavior (when the policy permits all pods in a real namespace)? if all pods in a namespace are permitted, it might be relevant as exposure info, even if there is a certain real pod in the namespace.

I see, yes we can change this. our assumption was that if there is a rule selecting a namespace and there was a real pod in that namespace so the "developer" knows that connections are allowed to that namespace and didn't add it to the exposure lines. so by changing this we want to emphasize that conns with all pods in that namespace are allowed . ( however this will affect also tests with only namespaceSelector, where the namespace is real and there is a pod in it in the manifests )

(BTW there could be a real pod in that namespace on which there is a certain policy that blocks such a connection for this pod.. in such a case the connection to this pod is not in the report, nor the exposure to this namespace). [consider adding such an example as well]

I prefer to add such examples within the adding more complicated tests PR

ok, let's have this change in this PR , including adding the relevant examples , and mark by the example which has inaccurate output, a TODO comment, to make sure it is fixed in the next PR .

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done

shireenf-ibm marked this conversation as resolved.
Show resolved Hide resolved
continue
}
// remove representative peer matching both realPodLabels and realNsLabels.
// a representative peer that matches any-pod in a specific namespace, and real Labels are for a specific pod in same namespace,
// the representative peer will be removed (we have at least one real pod in that namespace)
if potentialPodSelector.Matches(labels.Set(realPodLabels)) && potentialNsSelector.Matches(labels.Set(realNsLabels)) {
keysToDelete = append(keysToDelete, key)
}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
digraph {
subgraph "cluster_default" {
color="black"
fontcolor="black"
"default/backend[Deployment]" [label="backend[Deployment]" color="blue" fontcolor="blue"]
"default/frontend[Deployment]" [label="frontend[Deployment]" color="blue" fontcolor="blue"]
label="default"
}
subgraph "cluster_all namespaces" {
color="red2"
fontcolor="red2"
"pod with {app=frontend}_in_all namespaces" [label="pod with {app=frontend}" color="red2" fontcolor="red2"]
label="all namespaces"
}
"default/frontend[Deployment]" -> "default/backend[Deployment]" [label="TCP 9090" color="gold2" fontcolor="darkgreen"]
"pod with {app=frontend}_in_all namespaces" -> "default/backend[Deployment]" [label="TCP 9090" color="gold2" fontcolor="darkgreen"]
}
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
default/frontend[Deployment] => default/backend[Deployment] : TCP 9090

Exposure Analysis Result:

Ingress Exposure:
default/backend[Deployment] <= [all namespaces]/[pod with {app=frontend}] : TCP 9090
53 changes: 53 additions & 0 deletions tests/test_exposure_to_any_namespace_with_podSelector/backend.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
---
apiVersion: apps/v1
kind: Deployment
metadata:
name: backend
spec:
selector:
matchLabels:
app: backendservice
template:
metadata:
labels:
app: backendservice
spec:
containers:
- name: server
image: backendservice
ports:
- containerPort: 9090
readinessProbe:
initialDelaySeconds: 10
httpGet:
path: "/_healthz"
port: 9090
livenessProbe:
initialDelaySeconds: 10
httpGet:
path: "/_healthz"
port: 9090
env:
- name: PORT
value: "9090"
resources:
requests:
cpu: 100m
memory: 64Mi
limits:
cpu: 200m
memory: 128Mi
---
apiVersion: v1
kind: Service
metadata:
name: backendservice
spec:
type: ClusterIP
selector:
app: backendservice
ports:
- name: http
port: 9090
targetPort: 9090

Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
---
apiVersion: apps/v1
kind: Deployment
metadata:
name: frontend
spec:
selector:
matchLabels:
app: frontend
template:
metadata:
labels:
app: frontend
spec:
containers:
- name: server
image: frontend
ports:
- containerPort: 8080
readinessProbe:
initialDelaySeconds: 10
httpGet:
path: "/_healthz"
port: 8080
livenessProbe:
initialDelaySeconds: 10
httpGet:
path: "/_healthz"
port: 8080
env:
- name: PORT
value: "8080"
- name: BACKEND_SERVICE_ADDR
value: "backendservice:9090"
resources:
requests:
cpu: 100m
memory: 64Mi
limits:
cpu: 200m
memory: 128Mi
---
apiVersion: v1
kind: Service
metadata:
name: frontend
spec:
type: ClusterIP
selector:
app: frontend
ports:
- name: http
port: 80
targetPort: 8080
---
apiVersion: v1
kind: Service
metadata:
name: frontend-external
spec:
type: LoadBalancer
selector:
app: frontend
ports:
- name: http
port: 80
targetPort: 8080
59 changes: 59 additions & 0 deletions tests/test_exposure_to_any_namespace_with_podSelector/netpols.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
apiVersion: networking.k8s.io/v1
kind: NetworkPolicy
metadata:
creationTimestamp: null
name: backend-netpol
spec:
ingress:
- from:
- namespaceSelector: {}
podSelector:
matchLabels:
app: frontend
ports:
- port: 9090
protocol: TCP
podSelector:
matchLabels:
app: backendservice
policyTypes:
- Ingress
- Egress
status: {}

---
apiVersion: networking.k8s.io/v1
kind: NetworkPolicy
metadata:
creationTimestamp: null
name: frontend-netpol
spec:
egress:
- ports:
- port: 9090
protocol: TCP
to:
- podSelector:
matchLabels:
app: backendservice
podSelector:
matchLabels:
app: frontend
policyTypes:
- Ingress
- Egress
status: {}

---
apiVersion: networking.k8s.io/v1
kind: NetworkPolicy
metadata:
creationTimestamp: null
name: default-deny-in-namespace
spec:
podSelector: {}
policyTypes:
- Ingress
- Egress
status: {}