[G1-Sync] Manual knowledge update
This commit is contained in:
@@ -0,0 +1,380 @@
|
||||
---
|
||||
id: devops-crossplane-tekton
|
||||
title: Crossplane / Tekton — K8s 안 Cloud / CI
|
||||
category: Coding
|
||||
status: draft
|
||||
source_trust_level: B
|
||||
verification_status: conceptual
|
||||
created_at: 2026-05-09
|
||||
updated_at: 2026-05-09
|
||||
tags: [devops, crossplane, tekton, vibe-coding]
|
||||
tech_stack: { language: "YAML / K8s", applicable_to: ["DevOps"] }
|
||||
applied_in: []
|
||||
aliases: [Crossplane, Tekton, Composition, K8s native CI, infrastructure as data]
|
||||
---
|
||||
|
||||
# Crossplane / Tekton
|
||||
|
||||
> Kubernetes-native infra. **Crossplane = K8s 가 cloud 자원 관리 (Terraform 의 K8s alternative). Tekton = K8s native CI**. Argo Workflows / GitOps 전체 통합.
|
||||
|
||||
## 📖 핵심 개념
|
||||
- Crossplane: cloud 자원 = K8s resource.
|
||||
- Tekton: pipeline as Custom Resource.
|
||||
- Composition: 자체 high-level resource.
|
||||
- Provider: AWS / GCP / Azure 등.
|
||||
|
||||
## 💻 코드 패턴
|
||||
|
||||
### Crossplane 설치
|
||||
```bash
|
||||
helm install crossplane crossplane-stable/crossplane -n crossplane-system --create-namespace
|
||||
|
||||
# AWS provider
|
||||
kubectl apply -f - <<EOF
|
||||
apiVersion: pkg.crossplane.io/v1
|
||||
kind: Provider
|
||||
metadata: { name: provider-aws }
|
||||
spec:
|
||||
package: xpkg.upbound.io/upbound/provider-aws-rds:v1.0
|
||||
EOF
|
||||
```
|
||||
|
||||
### S3 bucket 만들기 (Crossplane)
|
||||
```yaml
|
||||
apiVersion: s3.aws.upbound.io/v1beta1
|
||||
kind: Bucket
|
||||
metadata: { name: my-bucket }
|
||||
spec:
|
||||
forProvider:
|
||||
region: us-east-1
|
||||
providerConfigRef:
|
||||
name: aws-default
|
||||
```
|
||||
|
||||
```bash
|
||||
kubectl apply -f bucket.yaml
|
||||
# → AWS 에 S3 bucket 생성 + status reflect
|
||||
```
|
||||
|
||||
### RDS instance
|
||||
```yaml
|
||||
apiVersion: rds.aws.upbound.io/v1beta1
|
||||
kind: Instance
|
||||
metadata: { name: my-db }
|
||||
spec:
|
||||
forProvider:
|
||||
region: us-east-1
|
||||
instanceClass: db.t3.micro
|
||||
allocatedStorage: 20
|
||||
engine: postgres
|
||||
engineVersion: "16"
|
||||
masterUsername: app
|
||||
autoGeneratePassword: true
|
||||
masterPasswordSecretRef:
|
||||
name: my-db-creds
|
||||
namespace: default
|
||||
key: password
|
||||
skipFinalSnapshot: true
|
||||
writeConnectionSecretToRef:
|
||||
name: my-db-conn
|
||||
namespace: default
|
||||
```
|
||||
|
||||
### Composition (high-level abstraction)
|
||||
```yaml
|
||||
apiVersion: apiextensions.crossplane.io/v1
|
||||
kind: CompositeResourceDefinition
|
||||
metadata:
|
||||
name: xappdatabases.acme.io
|
||||
spec:
|
||||
group: acme.io
|
||||
names: { kind: XAppDatabase, plural: xappdatabases }
|
||||
claimNames: { kind: AppDatabase, plural: appdatabases }
|
||||
versions:
|
||||
- name: v1
|
||||
served: true
|
||||
referenceable: true
|
||||
schema:
|
||||
openAPIV3Schema:
|
||||
type: object
|
||||
properties:
|
||||
spec:
|
||||
type: object
|
||||
properties:
|
||||
size: { type: string, enum: [small, medium, large] }
|
||||
region: { type: string }
|
||||
```
|
||||
|
||||
```yaml
|
||||
# Composition
|
||||
apiVersion: apiextensions.crossplane.io/v1
|
||||
kind: Composition
|
||||
metadata: { name: app-database-aws }
|
||||
spec:
|
||||
compositeTypeRef:
|
||||
apiVersion: acme.io/v1
|
||||
kind: XAppDatabase
|
||||
|
||||
resources:
|
||||
- name: rds
|
||||
base:
|
||||
apiVersion: rds.aws.upbound.io/v1beta1
|
||||
kind: Instance
|
||||
spec:
|
||||
forProvider:
|
||||
engine: postgres
|
||||
allocatedStorage: 20
|
||||
patches:
|
||||
- fromFieldPath: spec.region
|
||||
toFieldPath: spec.forProvider.region
|
||||
- fromFieldPath: spec.size
|
||||
toFieldPath: spec.forProvider.instanceClass
|
||||
transforms:
|
||||
- type: map
|
||||
map:
|
||||
small: db.t3.micro
|
||||
medium: db.t3.medium
|
||||
large: db.r6g.xlarge
|
||||
```
|
||||
|
||||
### Use (developer 친화)
|
||||
```yaml
|
||||
# Developer 만 작성 — 단순
|
||||
apiVersion: acme.io/v1
|
||||
kind: AppDatabase
|
||||
metadata: { name: orders-db }
|
||||
spec:
|
||||
size: medium
|
||||
region: us-east-1
|
||||
```
|
||||
|
||||
→ Crossplane 가 RDS instance + secret 자동 생성.
|
||||
|
||||
### Tekton Pipeline
|
||||
```yaml
|
||||
apiVersion: tekton.dev/v1beta1
|
||||
kind: Pipeline
|
||||
metadata: { name: build-and-deploy }
|
||||
spec:
|
||||
params:
|
||||
- name: repo-url
|
||||
type: string
|
||||
- name: image-name
|
||||
type: string
|
||||
|
||||
workspaces:
|
||||
- name: shared
|
||||
|
||||
tasks:
|
||||
- name: clone
|
||||
taskRef: { name: git-clone }
|
||||
params:
|
||||
- name: url
|
||||
value: $(params.repo-url)
|
||||
workspaces:
|
||||
- { name: output, workspace: shared }
|
||||
|
||||
- name: test
|
||||
runAfter: [clone]
|
||||
taskSpec:
|
||||
steps:
|
||||
- name: test
|
||||
image: node:20
|
||||
workingDir: $(workspaces.source.path)
|
||||
script: yarn install && yarn test
|
||||
workspaces:
|
||||
- { name: source, workspace: shared }
|
||||
|
||||
- name: build
|
||||
runAfter: [test]
|
||||
taskRef: { name: kaniko }
|
||||
params:
|
||||
- { name: IMAGE, value: $(params.image-name) }
|
||||
workspaces:
|
||||
- { name: source, workspace: shared }
|
||||
|
||||
- name: deploy
|
||||
runAfter: [build]
|
||||
taskRef: { name: kubectl-deploy }
|
||||
params:
|
||||
- { name: image, value: $(params.image-name) }
|
||||
```
|
||||
|
||||
### PipelineRun
|
||||
```yaml
|
||||
apiVersion: tekton.dev/v1beta1
|
||||
kind: PipelineRun
|
||||
metadata: { generateName: build-and-deploy- }
|
||||
spec:
|
||||
pipelineRef: { name: build-and-deploy }
|
||||
params:
|
||||
- { name: repo-url, value: https://github.com/myorg/app }
|
||||
- { name: image-name, value: ghcr.io/myorg/app:v1 }
|
||||
workspaces:
|
||||
- name: shared
|
||||
volumeClaimTemplate:
|
||||
spec:
|
||||
accessModes: [ReadWriteOnce]
|
||||
resources: { requests: { storage: 5Gi } }
|
||||
```
|
||||
|
||||
### Trigger (webhook)
|
||||
```yaml
|
||||
apiVersion: triggers.tekton.dev/v1beta1
|
||||
kind: EventListener
|
||||
metadata: { name: github-listener }
|
||||
spec:
|
||||
triggers:
|
||||
- name: github-push
|
||||
interceptors:
|
||||
- ref: { name: github }
|
||||
params:
|
||||
- name: secretRef
|
||||
value:
|
||||
secretName: github-secret
|
||||
secretKey: token
|
||||
- name: eventTypes
|
||||
value: [push]
|
||||
bindings:
|
||||
- ref: github-push-binding
|
||||
template:
|
||||
ref: build-and-deploy-template
|
||||
```
|
||||
|
||||
→ GitHub push → Pipeline 자동 trigger.
|
||||
|
||||
### Tekton Task (재사용 step)
|
||||
```yaml
|
||||
apiVersion: tekton.dev/v1beta1
|
||||
kind: Task
|
||||
metadata: { name: kaniko }
|
||||
spec:
|
||||
params:
|
||||
- name: IMAGE
|
||||
type: string
|
||||
workspaces:
|
||||
- name: source
|
||||
steps:
|
||||
- name: build
|
||||
image: gcr.io/kaniko-project/executor:latest
|
||||
args:
|
||||
- --dockerfile=$(workspaces.source.path)/Dockerfile
|
||||
- --context=$(workspaces.source.path)
|
||||
- --destination=$(params.IMAGE)
|
||||
```
|
||||
|
||||
### Tekton Hub (catalog)
|
||||
```bash
|
||||
# 공유 task 사용
|
||||
tkn hub install task git-clone
|
||||
tkn hub install task kaniko
|
||||
tkn hub install task npm
|
||||
```
|
||||
|
||||
### Tekton vs GitHub Actions / GitLab CI
|
||||
```
|
||||
Tekton:
|
||||
+ K8s native (자원 효율)
|
||||
+ Reusable (Task / Pipeline)
|
||||
+ Cloud-agnostic
|
||||
- 학습 곡선
|
||||
- UI 약함
|
||||
|
||||
GitHub Actions:
|
||||
+ 가장 인기
|
||||
+ Marketplace 커
|
||||
+ UI / 통합 강
|
||||
|
||||
→ K8s 사용 + on-prem CI = Tekton. 일반 = GHA.
|
||||
```
|
||||
|
||||
### Argo Workflows (alternative)
|
||||
```
|
||||
Tekton 와 비슷 — K8s native workflow.
|
||||
ML pipeline 친화 (Argo MLOps).
|
||||
```
|
||||
|
||||
### Crossplane vs Terraform
|
||||
```
|
||||
Terraform:
|
||||
+ 인기, 큰 ecosystem
|
||||
+ 어떤 시스템 — K8s 외 OK
|
||||
+ Plan 명확
|
||||
|
||||
Crossplane:
|
||||
+ K8s native (declarative + reconcile loop)
|
||||
+ GitOps 친화 (ArgoCD 가 manage)
|
||||
+ Composition 으로 자체 abstraction
|
||||
- K8s required
|
||||
- 옛 자원 일부 X
|
||||
|
||||
→ K8s 중심 = Crossplane. 일반 = Terraform.
|
||||
```
|
||||
|
||||
### Backstage + Crossplane + Tekton
|
||||
```
|
||||
1. Dev 가 Backstage scaffolder 클릭 ("New service")
|
||||
2. Tekton 가 GitHub repo 생성 + CI setup
|
||||
3. Crossplane 가 RDS / S3 자동 생성
|
||||
4. ArgoCD 가 cluster 에 deploy
|
||||
5. Backstage 에 자동 등록
|
||||
|
||||
→ Self-service developer experience.
|
||||
```
|
||||
|
||||
### KubeVela (alternative)
|
||||
```
|
||||
Crossplane 비슷 — open application model (OAM).
|
||||
Less popular but cleaner abstraction.
|
||||
```
|
||||
|
||||
### Operator pattern (Crossplane 의 underpinning)
|
||||
```
|
||||
Custom Resource + Controller = Operator.
|
||||
- CR 가 desired state.
|
||||
- Controller 가 reconcile (current = desired).
|
||||
```
|
||||
|
||||
```go
|
||||
// Controller pseudocode
|
||||
for {
|
||||
desired = readCR()
|
||||
current = readCloud()
|
||||
if desired != current {
|
||||
apply(desired)
|
||||
}
|
||||
sleep(30s)
|
||||
}
|
||||
```
|
||||
|
||||
→ Crossplane 가 cloud 의 controller.
|
||||
|
||||
## 🤔 의사결정 기준
|
||||
| 작업 | 추천 |
|
||||
|---|---|
|
||||
| K8s 친화 IaC | Crossplane |
|
||||
| 일반 IaC | Terraform |
|
||||
| K8s native CI | Tekton |
|
||||
| 일반 CI | GitHub Actions |
|
||||
| ML pipeline | Argo Workflows |
|
||||
| 개발자 portal | Backstage |
|
||||
| Self-service | Backstage + Crossplane + Tekton |
|
||||
|
||||
## ❌ 안티패턴
|
||||
- **Crossplane + 큰 cloud account 첫 시도**: 학습 비용. 작은 부분 부터.
|
||||
- **Tekton 가 GHA 대체 (작은 팀)**: overkill.
|
||||
- **Composition 무 versioning**: breaking 시 모든 instance 영향.
|
||||
- **Provider config 매 namespace**: 한 곳.
|
||||
- **Terraform + Crossplane 같은 자원 동시**: drift / 충돌.
|
||||
- **Tekton step 큰 image**: 매 step pull 시간.
|
||||
|
||||
## 🤖 LLM 활용 힌트
|
||||
- K8s + GitOps = Crossplane + ArgoCD.
|
||||
- Self-service portal = Backstage.
|
||||
- Tekton 가 K8s native CI but GHA 가 일반.
|
||||
- Composition 가 high-level abstraction.
|
||||
|
||||
## 🔗 관련 문서
|
||||
- [[DevOps_Helm_Deep]]
|
||||
- [[DevOps_ArgoCD_GitOps]]
|
||||
- [[DevOps_Backstage_Platform]]
|
||||
Reference in New Issue
Block a user