# Build Stage
FROM mcr.microsoft.com/dotnet/sdk:8.0 AS build
WORKDIR /src
# Copy project files
COPY ["src/MyApp/MyApp.csproj", "MyApp/"]
RUN dotnet restore "MyApp/MyApp.csproj"
# Copy source and build
COPY src/MyApp/. MyApp/
WORKDIR "/src/MyApp"
RUN dotnet publish -c Release -o /app/publish
# Runtime Stage
FROM mcr.microsoft.com/dotnet/aspnet:8.0 AS runtime
WORKDIR /app
COPY --from=build /app/publish .
# Non-root user
RUN groupadd -r appgroup && useradd -r -g appgroup appuser
RUN chown -R appuser:appgroup /app
USER appuser
EXPOSE 8080
ENV ASPNETCORE_URLS=http+:8080
ENTRYPOINT ["dotnet", "MyApp.dll"]
# Build image
docker build -t myapp:latest .
# Run container
docker run -d -p 8080:8080 --name myapp myapp:latest
# Run với environment variables
docker run -d -p 8080:8080 \
-e "ASPNETCORE_ENVIRONMENT=Production" \
-e "ConnectionStrings__DefaultConnection=..." \
myapp:latest
# Docker Compose
docker-compose.yml
version: '3.8'
services:
app:
build: .
ports:
- "8080:8080"
depends_on:
- db
- redis
db:
image: mcr.microsoft.com/mssql/server
environment:
- ACCEPT_EULA=Y
- SA_PASSWORD=YourStrong!Passw0rd
redis:
image: redis:alpine
apiVersion: v1
kind: Pod
metadata:
name: myapp-pod
labels:
app: myapp
spec:
containers:
- name: myapp
image: myapp:latest
ports:
- containerPort: 8080
resources:
requests:
memory: "64Mi"
cpu: "250m"
limits:
memory: "128Mi"
cpu: "500m"
apiVersion: apps/v1
kind: Deployment
metadata:
name: myapp-deployment
spec:
replicas: 3
selector:
matchLabels:
app: myapp
template:
metadata:
labels:
app: myapp
spec:
containers:
- name: myapp
image: myapp:latest
ports:
- containerPort: 8080
env:
- name: ASPNETCORE_ENVIRONMENT
value: "Production"
livenessProbe:
httpGet:
path: /health
port: 8080
initialDelaySeconds: 30
periodSeconds: 10
readinessProbe:
httpGet:
path: /health/ready
port: 8080
initialDelaySeconds: 5
periodSeconds: 5
apiVersion: v1
kind: Service
metadata:
name: myapp-service
spec:
selector:
app: myapp
ports:
- protocol: TCP
port: 80
targetPort: 8080
type: LoadBalancer
# Azure Container Apps
az containerapp create \
--name myapp \
--resource-group mygroup \
--image myregistry.azurecr.io/myapp:latest \
--cpu 0.25 --memory 0.5Gi \
--ingress external \
--target-port 8080
# Azure Kubernetes Service
az aks create \
--resource-group mygroup \
--name mycluster \
--node-count 3 \
--enable-addons monitoring
# Amazon ECS
aws ecs create-cluster --cluster-name mycluster
# Amazon EKS
aws eks create-cluster \
--name mycluster \
--role-arn arn:aws:iam::123456789:role/EKSRole \
--resources-vpc-config subnetIds=subnet-12345
| Factor | Description |
| Codebase | One codebase tracked in version control |
| Dependencies | Explicitly declare dependencies |
| Config | Store config in environment |
| Backing Services | Treat backing services as attached resources |
| Build/Release/Run | Strictly separate build and run stages |
| Processes | Execute app as one or more stateless processes |
| Port Binding | Export HTTP as a service by port binding |
| Concurrency | Scale out via process model |
| Disposability | Fast startup and graceful shutdown |
| Dev/Prod Parity | Keep development, staging, production similar |
| Logs | Treat logs as event streams |
| Admin Processes | Run admin/maintenance tasks as one-off processes |