Skip to content

Commit

Permalink
Merge branch 'branch/v16' into bot/backport-45366-branch/v16
Browse files Browse the repository at this point in the history
  • Loading branch information
smallinsky authored Aug 22, 2024
2 parents 12ebd77 + 76f2a52 commit bcd9748
Show file tree
Hide file tree
Showing 64 changed files with 1,015 additions and 510 deletions.
56 changes: 26 additions & 30 deletions api/client/client.go
Original file line number Diff line number Diff line change
Expand Up @@ -3788,6 +3788,31 @@ func GetUnifiedResourcePage(ctx context.Context, clt ListUnifiedResourcesClient,
}
}

// GetAllUnifiedResources is a helper for getting all existing resources that match the provided request. In addition to
// iterating pages, it also correctly handles downsizing pages when LimitExceeded errors are encountered.
func GetAllUnifiedResources(ctx context.Context, clt ListUnifiedResourcesClient, req *proto.ListUnifiedResourcesRequest) ([]*types.EnrichedResource, error) {
var out []*types.EnrichedResource

// Set the limit to the default size.
req.Limit = int32(defaults.DefaultChunkSize)
for {
resources, nextKey, err := GetUnifiedResourcePage(ctx, clt, req)
if err != nil {
return nil, trace.Wrap(err)
}

out = append(out, resources...)

if nextKey == "" || len(resources) == 0 {
break
}

req.StartKey = nextKey
}

return out, nil
}

// GetEnrichedResourcePage is a helper for getting a single page of enriched resources.
func GetEnrichedResourcePage(ctx context.Context, clt GetResourcesClient, req *proto.ListResourcesRequest) (ResourcePage[*types.EnrichedResource], error) {
var out ResourcePage[*types.EnrichedResource]
Expand Down Expand Up @@ -4064,36 +4089,7 @@ func GetKubernetesResourcesWithFilters(ctx context.Context, clt kubeproto.KubeSe
// but may result in confusing behavior if it is used outside of those contexts.
func (c *Client) GetSSHTargets(ctx context.Context, req *proto.GetSSHTargetsRequest) (*proto.GetSSHTargetsResponse, error) {
rsp, err := c.grpc.GetSSHTargets(ctx, req)
if err := trace.Wrap(err); !trace.IsNotImplemented(err) {
return rsp, err
}

// if we got a not implemented error, fallback to client-side filtering
servers, err := GetAllResources[*types.ServerV2](ctx, c, &proto.ListResourcesRequest{
ResourceType: types.KindNode,
UseSearchAsRoles: true,
})
if err != nil {
return nil, trace.Wrap(err)
}

// we only get here if we hit a NotImplementedError from GetSSHTargets, which means
// we should be performing client-side filtering with default parameters instead.
routeMatcher := utils.NewSSHRouteMatcher(req.Host, req.Port, false)

// do client-side filtering
filtered := servers[:0]
for _, srv := range servers {
if !routeMatcher.RouteToServer(srv) {
continue
}

filtered = append(filtered, srv)
}

return &proto.GetSSHTargetsResponse{
Servers: filtered,
}, nil
return rsp, trace.Wrap(err)
}

// CreateSessionTracker creates a tracker resource for an active session.
Expand Down
2 changes: 2 additions & 0 deletions docs/cspell.json
Original file line number Diff line number Diff line change
Expand Up @@ -121,6 +121,7 @@
"Keycloak",
"Keyspaces",
"knownhosts",
"Kubelet",
"Kubernetes",
"Kubes",
"LDAPS",
Expand Down Expand Up @@ -292,6 +293,7 @@
"argoproj",
"armv",
"atburke",
"attested",
"attrname",
"attobj",
"auditctl",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -148,6 +148,7 @@ Teleport's Workload Identity currently only supports issuing X.509-SVIDs.

- [Getting Started](./workload-identity/getting-started.mdx): How to configure Teleport for Workload Identity.
- [Best Practices](./workload-identity/best-practices.mdx): Best practices for using Workload Identity in Production.
- [Workload Attestation](./workload-identity/workload-attestation.mdx): Learn about using Workload Attestation to securely issue SVIDs to specific workloads.
- [TSH Support](./workload-identity/tsh.mdx): How to use `tsh` with Workload Identity to issue SVIDs to users.
- [AWS Roles Anywhere](./workload-identity/aws-roles-anywhere.mdx): Configuring AWS to accept Workload ID certificates as authentication using AWS Roles Anywhere.

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,239 @@
---
title: Workload Attestation
description: An overview of the Teleport Workload Identity Workload Attestation feature.
---

<Admonition type="tip" title="Preview">
Teleport Workload Identity is currently in Preview. This means that some
features may be missing. We're actively looking for design partners to help us
shape the future of Workload Identity and would love to
[hear your feedback](mailto:[email protected]).
</Admonition>

Workload Attestation is the process completed by `tbot` to assert the identity
of a workload that has connected to the Workload API and requested certificates.
The information gathered during attestation is used to decide which, if any,
SPIFFE IDs should be encoded into an SVID and issued to the workload.

Workload Attestors are the individual components that perform this attestation.
They use the process ID of the workload to gather information about the workload
from platform-specific APIs. For example, the Kubernetes Workload Attestor
queries the local Kubelet API to determine which Kubernetes pod the process
belongs to.

The result of the attestation process is known as attestation metadata. This
attestation metadata is referred to by the rules you configured for `tbot`'s
Workload API service. For example, you may state that only workloads running in
a specific Kubernetes namespace should be issued a specific SPIFFE ID.

Additionally, this metadata is included in the log messages output by `tbot`
when it issues an SVID. This allows you to audit the issuance of SVIDs and
understand why a specific SPIFFE ID was issued to a workload.

## Unix

The Unix Workload Attestor is the most basic attestor and allows you to restrict
the issuance of SVIDs to specific Unix processes based on a range of criteria.

### Attestation Metadata

The following metadata is produced by the Unix Workload Attestor and is
available to be used when configuring rules for `tbot`'s Workload API service:

| Field | Description |
|-------------------|------------------------------------------------------------------------------|
| `unix.attested` | Indicates that the workload has been attested by the Unix Workload Attestor. |
| `unix.pid` | The process ID of the attested workload. |
| `unix.uid` | The effective user ID of the attested workload. |
| `unix.gid` | The effective primary group ID of the attested workload. |

## Kubernetes

The Kubernetes Workload Attestor allows you to restrict the issuance of SVIDs
to specific Kubernetes workloads based on a range of criteria.

It works by first determining the pod ID for a given process ID and then by
querying the local kubelet API for details about that pod.

### Attestation Metadata

The following metadata is produced by the Kubernetes Workload Attestor and is
available to be used when configuring rules for `tbot`'s Workload API service:

| Field | Description |
|------------------------------|------------------------------------------------------------------------------------|
| `kubernetes.attested` | Indicates that the workload has been attested by the Kubernetes Workload Attestor. |
| `kubernetes.namespace` | The namespace of the Kubernetes Pod. |
| `kubernetes.service_account` | The service account of the Kubernetes Pod. |
| `kubernetes.pod_name` | The name of the Kubernetes Pod. |

### Deployment Guidance

To use Kubernetes Workload Attestation, `tbot` must be deployed as a daemon
set. This is because the unix domain socket can only be accessed by pods on the
same node as the agent. Additionally, the daemon set must have the `hostPID`
property set to `true` to allow the agent to access information about
processes within other containers.

The daemon set must also have a service account assigned that allows it to query
the Kubelet API. This is an example role with the required RBAC:

```yaml
kind: ClusterRole
apiVersion: rbac.authorization.k8s.io/v1
metadata:
name: tbot
rules:
- resources: ["pods","nodes","nodes/proxy"]
apiGroups: [""]
verbs: ["get"]
```
Mapping the Workload API Unix domain socket into the containers of workloads
can be done in two ways:
- Directly configuring a hostPath volume for the `tbot` daemonset and workloads
which will need to connect to it.
- Using [spiffe-csi-driver](https://github.com/spiffe/spiffe-csi).

Example manifests for required Kubernetes resources:

```yaml
kind: ClusterRole
apiVersion: rbac.authorization.k8s.io/v1
metadata:
name: tbot
rules:
- resources: ["pods","nodes","nodes/proxy"]
apiGroups: [""]
verbs: ["get"]
---
kind: ClusterRoleBinding
apiVersion: rbac.authorization.k8s.io/v1
metadata:
name: tbot
subjects:
- kind: ServiceAccount
name: tbot
namespace: default
roleRef:
kind: ClusterRole
name: tbot
apiGroup: rbac.authorization.k8s.io
---
apiVersion: v1
kind: ServiceAccount
metadata:
name: tbot
namespace: default
---
apiVersion: v1
kind: ConfigMap
metadata:
name: tbot-config
namespace: default
data:
tbot.yaml: |
version: v2
onboarding:
join_method: kubernetes
# replace with the name of a join token you have created.
token: example-token
storage:
type: memory
# ensure this is configured to the address of your Teleport Proxy Service.
proxy_server: example.teleport.sh:443
services:
- type: spiffe-workload-api
listen: unix:///run/tbot/sockets/workload.sock
attestor:
kubernetes:
enabled: true
kubelet:
# skip verification of the Kubelet API certificate as this is not
# usually issued by the cluster CA.
skip_verify: true
# replace the svid entries with the SPIFFE IDs that you wish to issue,
# using the `rules` blocks to restrict these to specific Kubernetes
# workloads.
svids:
- path: /my-service
rules:
- kubernetes:
namespace: default
service_account: example-sa
---
apiVersion: apps/v1
kind: DaemonSet
metadata:
name: tbot
spec:
selector:
matchLabels:
app: tbot
template:
metadata:
labels:
app: tbot
spec:
securityContext:
runAsUser: 0
runAsGroup: 0
hostPID: true
containers:
- name: tbot
image: public.ecr.aws/gravitational/tbot-distroless:(=teleport.version=)
imagePullPolicy: IfNotPresent
securityContext:
privileged: true
args:
- start
- -c
- /config/tbot.yaml
- --log-format
- json
volumeMounts:
- mountPath: /config
name: config
- mountPath: /var/run/secrets/tokens
name: join-sa-token
- name: tbot-sockets
mountPath: /run/tbot/sockets
readOnly: false
env:
- name: TELEPORT_NODE_NAME
valueFrom:
fieldRef:
fieldPath: spec.nodeName
- name: KUBERNETES_TOKEN_PATH
value: /var/run/secrets/tokens/join-sa-token
serviceAccountName: tbot
volumes:
- name: tbot-sockets
hostPath:
path: /run/tbot/sockets
type: DirectoryOrCreate
- name: config
configMap:
name: tbot-config
- name: join-sa-token
projected:
sources:
- serviceAccountToken:
path: join-sa-token
# 600 seconds is the minimum that Kubernetes supports. We
# recommend this value is used.
expirationSeconds: 600
# `example.teleport.sh` must be replaced with the name of
# your Teleport cluster.
audience: example.teleport.sh
```
## Next steps
- [Workload Identity Overview](../workload-identity.mdx): Overview of Teleport
Workload Identity.
- [Best Practices](./best-practices.mdx): Best practices for using Workload
Identity in Production.
- Read the [configuration reference](../../../reference/machine-id/configuration.mdx) to explore
all the available configuration options.
Loading

0 comments on commit bcd9748

Please sign in to comment.