@polymorphism-tech/morph-spec 1.0.2 → 2.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CLAUDE.md +1381 -0
- package/LICENSE +72 -0
- package/README.md +114 -12
- package/bin/detect-agents.js +225 -0
- package/bin/morph-spec.js +120 -0
- package/bin/render-template.js +302 -0
- package/bin/semantic-detect-agents.js +246 -0
- package/bin/validate-agents-skills.js +239 -0
- package/bin/validate-agents.js +69 -0
- package/bin/validate-phase.js +263 -0
- package/content/.azure/README.md +293 -0
- package/content/.azure/docs/azure-devops-setup.md +454 -0
- package/content/.azure/docs/branch-strategy.md +398 -0
- package/content/.azure/docs/local-development.md +515 -0
- package/content/.azure/pipelines/pipeline-variables.yml +34 -0
- package/content/.azure/pipelines/prod-pipeline.yml +319 -0
- package/content/.azure/pipelines/staging-pipeline.yml +234 -0
- package/content/.azure/pipelines/templates/build-dotnet.yml +75 -0
- package/content/.azure/pipelines/templates/deploy-app-service.yml +94 -0
- package/content/.azure/pipelines/templates/deploy-container-app.yml +120 -0
- package/content/.azure/pipelines/templates/infra-deploy.yml +90 -0
- package/content/.claude/commands/morph-apply.md +118 -26
- package/content/.claude/commands/morph-archive.md +9 -9
- package/content/.claude/commands/morph-clarify.md +184 -0
- package/content/.claude/commands/morph-design.md +275 -0
- package/content/.claude/commands/morph-proposal.md +56 -15
- package/content/.claude/commands/morph-setup.md +100 -0
- package/content/.claude/commands/morph-status.md +47 -32
- package/content/.claude/commands/morph-tasks.md +319 -0
- package/content/.claude/commands/morph-uiux.md +211 -0
- package/content/.claude/skills/specialists/ai-system-architect.md +604 -0
- package/content/.claude/skills/specialists/ms-agent-expert.md +143 -89
- package/content/.claude/skills/specialists/ui-ux-designer.md +744 -9
- package/content/.claude/skills/stacks/dotnet-blazor.md +244 -8
- package/content/.claude/skills/stacks/dotnet-nextjs.md +2 -2
- package/content/.morph/.morphversion +5 -0
- package/content/.morph/config/agents.json +101 -8
- package/content/.morph/config/azure-pricing.json +70 -0
- package/content/.morph/config/azure-pricing.schema.json +50 -0
- package/content/.morph/config/config.template.json +15 -3
- package/content/.morph/docs/STORY-DRIVEN-DEVELOPMENT.md +392 -0
- package/content/.morph/hooks/README.md +239 -0
- package/content/.morph/hooks/pre-commit-agents.sh +24 -0
- package/content/.morph/hooks/pre-commit-all.sh +48 -0
- package/content/.morph/hooks/pre-commit-costs.sh +91 -0
- package/content/.morph/hooks/pre-commit-specs.sh +49 -0
- package/content/.morph/hooks/pre-commit-tests.sh +60 -0
- package/content/.morph/project.md +5 -4
- package/content/.morph/schemas/agent.schema.json +296 -0
- package/content/.morph/standards/agent-framework-setup.md +453 -0
- package/content/.morph/standards/architecture.md +142 -7
- package/content/.morph/standards/azure.md +218 -23
- package/content/.morph/standards/coding.md +47 -12
- package/content/.morph/standards/dotnet10-migration.md +494 -0
- package/content/.morph/standards/fluent-ui-setup.md +590 -0
- package/content/.morph/standards/migration-guide.md +514 -0
- package/content/.morph/standards/passkeys-auth.md +423 -0
- package/content/.morph/standards/vector-search-rag.md +536 -0
- package/content/.morph/state.json +18 -0
- package/content/.morph/templates/FluentDesignTheme.cs +149 -0
- package/content/.morph/templates/MudTheme.cs +281 -0
- package/content/.morph/templates/contracts.cs +55 -55
- package/content/.morph/templates/decisions.md +4 -4
- package/content/.morph/templates/design-system.css +226 -0
- package/content/.morph/templates/infra/.dockerignore.example +89 -0
- package/content/.morph/templates/infra/Dockerfile.example +82 -0
- package/content/.morph/templates/infra/README.md +286 -0
- package/content/.morph/templates/infra/app-service.bicep +164 -0
- package/content/.morph/templates/infra/deploy.ps1 +229 -0
- package/content/.morph/templates/infra/deploy.sh +208 -0
- package/content/.morph/templates/infra/main.bicep +41 -7
- package/content/.morph/templates/infra/parameters.dev.json +6 -0
- package/content/.morph/templates/infra/parameters.prod.json +6 -0
- package/content/.morph/templates/infra/parameters.staging.json +29 -0
- package/content/.morph/templates/proposal.md +3 -3
- package/content/.morph/templates/recap.md +3 -3
- package/content/.morph/templates/spec.md +9 -8
- package/content/.morph/templates/sprint-status.yaml +68 -0
- package/content/.morph/templates/state.template.json +222 -0
- package/content/.morph/templates/story.md +143 -0
- package/content/.morph/templates/tasks.md +1 -1
- package/content/.morph/templates/ui-components.md +276 -0
- package/content/.morph/templates/ui-design-system.md +286 -0
- package/content/.morph/templates/ui-flows.md +336 -0
- package/content/.morph/templates/ui-mockups.md +133 -0
- package/content/.morph/test-infra/example.bicep +59 -0
- package/content/CLAUDE.md +124 -0
- package/content/README.md +79 -0
- package/detectors/config-detector.js +223 -0
- package/detectors/conversation-analyzer.js +163 -0
- package/detectors/index.js +84 -0
- package/detectors/standards-generator.js +275 -0
- package/detectors/structure-detector.js +221 -0
- package/docs/README.md +149 -0
- package/docs/api/cost-calculator.js.html +513 -0
- package/docs/api/design-system-generator.js.html +382 -0
- package/docs/api/fonts/Montserrat/Montserrat-Bold.eot +0 -0
- package/docs/api/fonts/Montserrat/Montserrat-Bold.ttf +0 -0
- package/docs/api/fonts/Montserrat/Montserrat-Bold.woff +0 -0
- package/docs/api/fonts/Montserrat/Montserrat-Bold.woff2 +0 -0
- package/docs/api/fonts/Montserrat/Montserrat-Regular.eot +0 -0
- package/docs/api/fonts/Montserrat/Montserrat-Regular.ttf +0 -0
- package/docs/api/fonts/Montserrat/Montserrat-Regular.woff +0 -0
- package/docs/api/fonts/Montserrat/Montserrat-Regular.woff2 +0 -0
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-light-webfont.eot +0 -0
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-light-webfont.svg +978 -0
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-light-webfont.ttf +0 -0
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-light-webfont.woff +0 -0
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-light-webfont.woff2 +0 -0
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-regular-webfont.eot +0 -0
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-regular-webfont.svg +1049 -0
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-regular-webfont.ttf +0 -0
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-regular-webfont.woff +0 -0
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-regular-webfont.woff2 +0 -0
- package/docs/api/global.html +5263 -0
- package/docs/api/index.html +96 -0
- package/docs/api/scripts/collapse.js +39 -0
- package/docs/api/scripts/commonNav.js +28 -0
- package/docs/api/scripts/linenumber.js +25 -0
- package/docs/api/scripts/nav.js +12 -0
- package/docs/api/scripts/polyfill.js +4 -0
- package/docs/api/scripts/prettify/Apache-License-2.0.txt +202 -0
- package/docs/api/scripts/prettify/lang-css.js +2 -0
- package/docs/api/scripts/prettify/prettify.js +28 -0
- package/docs/api/scripts/search.js +99 -0
- package/docs/api/state-manager.js.html +423 -0
- package/docs/api/styles/jsdoc.css +776 -0
- package/docs/api/styles/prettify.css +80 -0
- package/docs/examples.md +328 -0
- package/docs/getting-started.md +302 -0
- package/docs/installation.md +361 -0
- package/docs/templates.md +418 -0
- package/docs/validation-checklist.md +266 -0
- package/package.json +39 -12
- package/src/commands/cost.js +181 -0
- package/src/commands/create-story.js +283 -0
- package/src/commands/detect.js +104 -0
- package/src/commands/doctor.js +67 -0
- package/src/commands/generate.js +149 -0
- package/src/commands/init.js +71 -46
- package/src/commands/shard-spec.js +224 -0
- package/src/commands/sprint-status.js +250 -0
- package/src/commands/state.js +333 -0
- package/src/commands/sync.js +167 -0
- package/src/commands/update-pricing.js +206 -0
- package/src/commands/update.js +88 -13
- package/src/lib/complexity-analyzer.js +292 -0
- package/src/lib/cost-calculator.js +429 -0
- package/src/lib/design-system-generator.js +298 -0
- package/src/lib/state-manager.js +340 -0
- package/src/utils/file-copier.js +63 -0
- package/src/utils/version-checker.js +175 -0
|
@@ -0,0 +1,454 @@
|
|
|
1
|
+
# Azure DevOps Setup - Workload Identity Federation
|
|
2
|
+
|
|
3
|
+
> **MORPH-SPEC Framework**
|
|
4
|
+
> Configuração de CI/CD com autenticação moderna (sem secrets)
|
|
5
|
+
|
|
6
|
+
---
|
|
7
|
+
|
|
8
|
+
## 📋 Índice
|
|
9
|
+
|
|
10
|
+
1. [Pré-requisitos](#pré-requisitos)
|
|
11
|
+
2. [Configurar Workload Identity Federation](#configurar-workload-identity-federation)
|
|
12
|
+
3. [Criar Service Connections](#criar-service-connections)
|
|
13
|
+
4. [Configurar Pipelines](#configurar-pipelines)
|
|
14
|
+
5. [Configurar Environments e Aprovações](#configurar-environments-e-aprovações)
|
|
15
|
+
6. [Troubleshooting](#troubleshooting)
|
|
16
|
+
|
|
17
|
+
---
|
|
18
|
+
|
|
19
|
+
## 🔑 Pré-requisitos
|
|
20
|
+
|
|
21
|
+
### Azure
|
|
22
|
+
- ✅ Subscription Azure ativa
|
|
23
|
+
- ✅ Permissões de Owner ou User Access Administrator na subscription
|
|
24
|
+
- ✅ Azure CLI instalado: https://aka.ms/azure-cli
|
|
25
|
+
|
|
26
|
+
### Azure DevOps
|
|
27
|
+
- ✅ Organização Azure DevOps criada
|
|
28
|
+
- ✅ Projeto criado
|
|
29
|
+
- ✅ Permissões de administrador do projeto
|
|
30
|
+
|
|
31
|
+
###Informações Necessárias
|
|
32
|
+
```bash
|
|
33
|
+
# Azure
|
|
34
|
+
SUBSCRIPTION_ID="<sua-subscription-id>"
|
|
35
|
+
TENANT_ID="<seu-tenant-id>"
|
|
36
|
+
|
|
37
|
+
# Azure DevOps
|
|
38
|
+
ADO_ORG="<sua-org>" # Ex: polymorphismtech
|
|
39
|
+
ADO_PROJECT="<seu-projeto>" # Ex: morph-app
|
|
40
|
+
|
|
41
|
+
# Application
|
|
42
|
+
APP_NAME="<nome-da-app>" # Ex: myapp
|
|
43
|
+
```
|
|
44
|
+
|
|
45
|
+
---
|
|
46
|
+
|
|
47
|
+
## 🌐 Configurar Workload Identity Federation
|
|
48
|
+
|
|
49
|
+
### Passo 1: Criar App Registration
|
|
50
|
+
|
|
51
|
+
```bash
|
|
52
|
+
# Login no Azure
|
|
53
|
+
az login
|
|
54
|
+
az account set --subscription $SUBSCRIPTION_ID
|
|
55
|
+
|
|
56
|
+
# Criar App Registration para Dev
|
|
57
|
+
APP_DEV_NAME="${APP_NAME}-dev-pipeline"
|
|
58
|
+
APP_DEV_ID=$(az ad app create \
|
|
59
|
+
--display-name "$APP_DEV_NAME" \
|
|
60
|
+
--query appId -o tsv)
|
|
61
|
+
|
|
62
|
+
echo "Dev App ID: $APP_DEV_ID"
|
|
63
|
+
|
|
64
|
+
# Criar Service Principal
|
|
65
|
+
SP_DEV_ID=$(az ad sp create \
|
|
66
|
+
--id $APP_DEV_ID \
|
|
67
|
+
--query id -o tsv)
|
|
68
|
+
|
|
69
|
+
echo "Dev Service Principal ID: $SP_DEV_ID"
|
|
70
|
+
|
|
71
|
+
# Repetir para Staging e Prod
|
|
72
|
+
APP_STAGING_NAME="${APP_NAME}-staging-pipeline"
|
|
73
|
+
APP_STAGING_ID=$(az ad app create --display-name "$APP_STAGING_NAME" --query appId -o tsv)
|
|
74
|
+
SP_STAGING_ID=$(az ad sp create --id $APP_STAGING_ID --query id -o tsv)
|
|
75
|
+
|
|
76
|
+
APP_PROD_NAME="${APP_NAME}-prod-pipeline"
|
|
77
|
+
APP_PROD_ID=$(az ad app create --display-name "$APP_PROD_NAME" --query appId -o tsv)
|
|
78
|
+
SP_PROD_ID=$(az ad sp create --id $APP_PROD_ID --query id -o tsv)
|
|
79
|
+
```
|
|
80
|
+
|
|
81
|
+
### Passo 2: Configurar Federated Credentials
|
|
82
|
+
|
|
83
|
+
```bash
|
|
84
|
+
# DEV Environment
|
|
85
|
+
cat <<EOF > federated-credential-dev.json
|
|
86
|
+
{
|
|
87
|
+
"name": "dev-pipeline-federated",
|
|
88
|
+
"issuer": "https://vstoken.dev.azure.com/<ADO_ORG_ID>",
|
|
89
|
+
"subject": "sc://$ADO_ORG/$ADO_PROJECT/Azure-Dev-Connection",
|
|
90
|
+
"description": "Federated credential for dev pipeline",
|
|
91
|
+
"audiences": [
|
|
92
|
+
"api://AzureADTokenExchange"
|
|
93
|
+
]
|
|
94
|
+
}
|
|
95
|
+
EOF
|
|
96
|
+
|
|
97
|
+
az ad app federated-credential create \
|
|
98
|
+
--id $APP_DEV_ID \
|
|
99
|
+
--parameters federated-credential-dev.json
|
|
100
|
+
|
|
101
|
+
# STAGING Environment
|
|
102
|
+
cat <<EOF > federated-credential-staging.json
|
|
103
|
+
{
|
|
104
|
+
"name": "staging-pipeline-federated",
|
|
105
|
+
"issuer": "https://vstoken.dev.azure.com/<ADO_ORG_ID>",
|
|
106
|
+
"subject": "sc://$ADO_ORG/$ADO_PROJECT/Azure-Staging-Connection",
|
|
107
|
+
"description": "Federated credential for staging pipeline",
|
|
108
|
+
"audiences": [
|
|
109
|
+
"api://AzureADTokenExchange"
|
|
110
|
+
]
|
|
111
|
+
}
|
|
112
|
+
EOF
|
|
113
|
+
|
|
114
|
+
az ad app federated-credential create \
|
|
115
|
+
--id $APP_STAGING_ID \
|
|
116
|
+
--parameters federated-credential-staging.json
|
|
117
|
+
|
|
118
|
+
# PROD Environment
|
|
119
|
+
cat <<EOF > federated-credential-prod.json
|
|
120
|
+
{
|
|
121
|
+
"name": "prod-pipeline-federated",
|
|
122
|
+
"issuer": "https://vstoken.dev.azure.com/<ADO_ORG_ID>",
|
|
123
|
+
"subject": "sc://$ADO_ORG/$ADO_PROJECT/Azure-Prod-Connection",
|
|
124
|
+
"description": "Federated credential for prod pipeline",
|
|
125
|
+
"audiences": [
|
|
126
|
+
"api://AzureADTokenExchange"
|
|
127
|
+
]
|
|
128
|
+
}
|
|
129
|
+
EOF
|
|
130
|
+
|
|
131
|
+
az ad app federated-credential create \
|
|
132
|
+
--id $APP_PROD_ID \
|
|
133
|
+
--parameters federated-credential-prod.json
|
|
134
|
+
```
|
|
135
|
+
|
|
136
|
+
**📌 Como obter ADO_ORG_ID:**
|
|
137
|
+
```bash
|
|
138
|
+
# Via Azure DevOps UI
|
|
139
|
+
# Vá em: Organization Settings → Overview → Organization ID
|
|
140
|
+
# Ou via API:
|
|
141
|
+
curl -u ":${AZURE_DEVOPS_PAT}" \
|
|
142
|
+
"https://dev.azure.com/${ADO_ORG}/_apis/connectionData"
|
|
143
|
+
```
|
|
144
|
+
|
|
145
|
+
### Passo 3: Atribuir Permissões Azure
|
|
146
|
+
|
|
147
|
+
```bash
|
|
148
|
+
# DEV - Contributor na resource group
|
|
149
|
+
RG_DEV="rg-${APP_NAME}-dev"
|
|
150
|
+
az role assignment create \
|
|
151
|
+
--assignee $SP_DEV_ID \
|
|
152
|
+
--role Contributor \
|
|
153
|
+
--scope "/subscriptions/$SUBSCRIPTION_ID/resourceGroups/$RG_DEV"
|
|
154
|
+
|
|
155
|
+
# STAGING - Contributor na resource group
|
|
156
|
+
RG_STAGING="rg-${APP_NAME}-staging"
|
|
157
|
+
az role assignment create \
|
|
158
|
+
--assignee $SP_STAGING_ID \
|
|
159
|
+
--role Contributor \
|
|
160
|
+
--scope "/subscriptions/$SUBSCRIPTION_ID/resourceGroups/$RG_STAGING"
|
|
161
|
+
|
|
162
|
+
# PROD - Contributor na resource group
|
|
163
|
+
RG_PROD="rg-${APP_NAME}-prod"
|
|
164
|
+
az role assignment create \
|
|
165
|
+
--assignee $SP_PROD_ID \
|
|
166
|
+
--role Contributor \
|
|
167
|
+
--scope "/subscriptions/$SUBSCRIPTION_ID/resourceGroups/$RG_PROD"
|
|
168
|
+
|
|
169
|
+
# ACR - AcrPush para todos
|
|
170
|
+
ACR_ID="/subscriptions/$SUBSCRIPTION_ID/resourceGroups/<rg-acr>/providers/Microsoft.ContainerRegistry/registries/<acr-name>"
|
|
171
|
+
|
|
172
|
+
az role assignment create --assignee $SP_DEV_ID --role AcrPush --scope $ACR_ID
|
|
173
|
+
az role assignment create --assignee $SP_STAGING_ID --role AcrPush --scope $ACR_ID
|
|
174
|
+
az role assignment create --assignee $SP_PROD_ID --role AcrPush --scope $ACR_ID
|
|
175
|
+
```
|
|
176
|
+
|
|
177
|
+
---
|
|
178
|
+
|
|
179
|
+
## 🔗 Criar Service Connections
|
|
180
|
+
|
|
181
|
+
### Via Azure DevOps UI
|
|
182
|
+
|
|
183
|
+
#### 1. Service Connection para Azure (Dev)
|
|
184
|
+
|
|
185
|
+
1. Vá em: **Project Settings** → **Service connections** → **New service connection**
|
|
186
|
+
2. Selecione: **Azure Resource Manager**
|
|
187
|
+
3. Authentication method: **Workload Identity federation (automatic)**
|
|
188
|
+
4. Scope level: **Subscription**
|
|
189
|
+
5. Preencha:
|
|
190
|
+
- **Subscription ID**: `<sua-subscription-id>`
|
|
191
|
+
- **Service connection name**: `Azure-Dev-Connection`
|
|
192
|
+
- **Service Principal ID**: `$APP_DEV_ID` (do Passo 1)
|
|
193
|
+
6. Marque: **Grant access permission to all pipelines** (ou configure por pipeline)
|
|
194
|
+
7. Click: **Save**
|
|
195
|
+
|
|
196
|
+
#### 2. Repetir para Staging e Prod
|
|
197
|
+
|
|
198
|
+
- **Staging**: Nome `Azure-Staging-Connection`, usar `$APP_STAGING_ID`
|
|
199
|
+
- **Prod**: Nome `Azure-Prod-Connection`, usar `$APP_PROD_ID`
|
|
200
|
+
|
|
201
|
+
#### 3. Service Connection para ACR
|
|
202
|
+
|
|
203
|
+
1. **New service connection** → **Docker Registry**
|
|
204
|
+
2. Registry type: **Azure Container Registry**
|
|
205
|
+
3. Authentication type: **Workload Identity federation**
|
|
206
|
+
4. Preencha:
|
|
207
|
+
- **Azure subscription**: Selecione a subscription
|
|
208
|
+
- **Azure container registry**: Selecione seu ACR
|
|
209
|
+
- **Service connection name**: `ACR-Connection`
|
|
210
|
+
5. **Save**
|
|
211
|
+
|
|
212
|
+
### Via Azure CLI (Alternativa)
|
|
213
|
+
|
|
214
|
+
```bash
|
|
215
|
+
# Requer Azure DevOps extension
|
|
216
|
+
az extension add --name azure-devops
|
|
217
|
+
|
|
218
|
+
# Login
|
|
219
|
+
az devops configure --defaults organization=https://dev.azure.com/$ADO_ORG project=$ADO_PROJECT
|
|
220
|
+
|
|
221
|
+
# Criar service connection (exemplo simplificado)
|
|
222
|
+
# Nota: Workload Identity via CLI é complexo, recomenda-se usar UI
|
|
223
|
+
```
|
|
224
|
+
|
|
225
|
+
---
|
|
226
|
+
|
|
227
|
+
## ⚙️ Configurar Pipelines
|
|
228
|
+
|
|
229
|
+
### Passo 1: Importar Pipelines
|
|
230
|
+
|
|
231
|
+
1. Vá em: **Pipelines** → **New pipeline**
|
|
232
|
+
2. Selecione: **Azure Repos Git** (ou seu SCM)
|
|
233
|
+
3. Selecione seu repositório
|
|
234
|
+
4. **Existing Azure Pipelines YAML file**
|
|
235
|
+
5. Path: `.azure/pipelines/dev-pipeline.yml`
|
|
236
|
+
6. **Continue** → **Save** (não run ainda)
|
|
237
|
+
|
|
238
|
+
Repetir para:
|
|
239
|
+
- `.azure/pipelines/staging-pipeline.yml`
|
|
240
|
+
- `.azure/pipelines/prod-pipeline.yml`
|
|
241
|
+
|
|
242
|
+
### Passo 2: Configurar Variáveis
|
|
243
|
+
|
|
244
|
+
#### Variáveis no Pipeline Level
|
|
245
|
+
|
|
246
|
+
Para cada pipeline, adicione as variáveis:
|
|
247
|
+
|
|
248
|
+
**Dev Pipeline** → **Edit** → **Variables**:
|
|
249
|
+
```
|
|
250
|
+
ACR_NAME: <seu-acr-name>
|
|
251
|
+
APP_NAME: <seu-app-name>
|
|
252
|
+
SUBSCRIPTION_ID: <subscription-id>
|
|
253
|
+
```
|
|
254
|
+
|
|
255
|
+
**Staging Pipeline** - mesmas variáveis
|
|
256
|
+
|
|
257
|
+
**Prod Pipeline** - mesmas variáveis
|
|
258
|
+
|
|
259
|
+
#### Variáveis no Group Level (Opcional)
|
|
260
|
+
|
|
261
|
+
1. **Pipelines** → **Library** → **+ Variable group**
|
|
262
|
+
2. Nome: `morph-common-vars`
|
|
263
|
+
3. Adicionar:
|
|
264
|
+
```
|
|
265
|
+
ACR_NAME: <seu-acr>
|
|
266
|
+
APP_NAME: <seu-app>
|
|
267
|
+
SUBSCRIPTION_ID: <subscription-id>
|
|
268
|
+
```
|
|
269
|
+
4. Linkar aos pipelines:
|
|
270
|
+
```yaml
|
|
271
|
+
variables:
|
|
272
|
+
- group: morph-common-vars
|
|
273
|
+
- template: pipeline-variables.yml
|
|
274
|
+
```
|
|
275
|
+
|
|
276
|
+
---
|
|
277
|
+
|
|
278
|
+
## 🛡️ Configurar Environments e Aprovações
|
|
279
|
+
|
|
280
|
+
### Passo 1: Criar Environments
|
|
281
|
+
|
|
282
|
+
1. **Pipelines** → **Environments** → **New environment**
|
|
283
|
+
2. Criar 3 environments:
|
|
284
|
+
|
|
285
|
+
**Dev Environment:**
|
|
286
|
+
- Name: `dev`
|
|
287
|
+
- Resource: None
|
|
288
|
+
- Approvals: **Nenhuma** (deploy automático)
|
|
289
|
+
|
|
290
|
+
**Staging Environment:**
|
|
291
|
+
- Name: `staging`
|
|
292
|
+
- Resource: None
|
|
293
|
+
- Approvals: **Opcional** (recomendado nenhuma para deploy rápido)
|
|
294
|
+
- Se desejar: Add approver selecione você mesmo
|
|
295
|
+
- Timeout: 24 hours
|
|
296
|
+
|
|
297
|
+
**Production Environment:**
|
|
298
|
+
- Name: `production`
|
|
299
|
+
- Resource: None
|
|
300
|
+
- Approvals: **OBRIGATÓRIO**
|
|
301
|
+
- Add approvers: Selecione você mesmo
|
|
302
|
+
- Timeout: 48 hours
|
|
303
|
+
- **Checks**: Adicionar "Invoke REST API" para verificações adicionais (opcional)
|
|
304
|
+
|
|
305
|
+
### Passo 2: Configurar Branch Policies (Opcional)
|
|
306
|
+
|
|
307
|
+
Para `main` branch:
|
|
308
|
+
|
|
309
|
+
1. **Repos** → **Branches** → `main` → **Branch policies**
|
|
310
|
+
2. Habilitar:
|
|
311
|
+
- **Require a minimum number of reviewers**: 0 (self-review via approval gate)
|
|
312
|
+
- **Check for linked work items**: Recommended
|
|
313
|
+
- **Build validation**: Link prod pipeline
|
|
314
|
+
|
|
315
|
+
---
|
|
316
|
+
|
|
317
|
+
## 🧪 Testar Configuração
|
|
318
|
+
|
|
319
|
+
### Teste 1: Dev Pipeline
|
|
320
|
+
|
|
321
|
+
```bash
|
|
322
|
+
# Criar branch develop
|
|
323
|
+
git checkout -b develop
|
|
324
|
+
git push origin develop
|
|
325
|
+
|
|
326
|
+
# Fazer um commit qualquer
|
|
327
|
+
echo "test" > test.txt
|
|
328
|
+
git add test.txt
|
|
329
|
+
git commit -m "test: trigger dev pipeline"
|
|
330
|
+
git push origin develop
|
|
331
|
+
```
|
|
332
|
+
|
|
333
|
+
**Verificar:**
|
|
334
|
+
- Pipeline triggou automaticamente
|
|
335
|
+
- Build passou
|
|
336
|
+
- Deploy para App Service Free foi bem-sucedido
|
|
337
|
+
- Health check passou
|
|
338
|
+
|
|
339
|
+
### Teste 2: Staging Pipeline
|
|
340
|
+
|
|
341
|
+
```bash
|
|
342
|
+
# Merge develop em main
|
|
343
|
+
git checkout main
|
|
344
|
+
git merge develop
|
|
345
|
+
git push origin main
|
|
346
|
+
```
|
|
347
|
+
|
|
348
|
+
**Verificar:**
|
|
349
|
+
- Staging pipeline triggou
|
|
350
|
+
- Container foi buildado e pushed para ACR
|
|
351
|
+
- Deploy para Container Apps funcionou
|
|
352
|
+
- Integration tests passaram
|
|
353
|
+
|
|
354
|
+
### Teste 3: Prod Pipeline (Manual)
|
|
355
|
+
|
|
356
|
+
1. **Pipelines** → **prod-pipeline** → **Run pipeline**
|
|
357
|
+
2. Verificar aprovação manual aparece
|
|
358
|
+
3. Aprovar deploy
|
|
359
|
+
4. Verificar deployment bem-sucedido
|
|
360
|
+
|
|
361
|
+
---
|
|
362
|
+
|
|
363
|
+
## 🆘 Troubleshooting
|
|
364
|
+
|
|
365
|
+
### Erro: "Failed to get federated token"
|
|
366
|
+
|
|
367
|
+
**Causa:** Subject no federated credential não match com service connection.
|
|
368
|
+
|
|
369
|
+
**Solução:**
|
|
370
|
+
```bash
|
|
371
|
+
# Verificar subject correto
|
|
372
|
+
# Deve ser: sc://<ORG>/<PROJECT>/<SERVICE_CONNECTION_NAME>
|
|
373
|
+
|
|
374
|
+
# Recriar federated credential com subject correto
|
|
375
|
+
az ad app federated-credential delete \
|
|
376
|
+
--id $APP_ID \
|
|
377
|
+
--federated-credential-id <credential-id>
|
|
378
|
+
|
|
379
|
+
# Criar novamente com subject correto
|
|
380
|
+
az ad app federated-credential create \
|
|
381
|
+
--id $APP_ID \
|
|
382
|
+
--parameters federated-credential.json
|
|
383
|
+
```
|
|
384
|
+
|
|
385
|
+
### Erro: "Insufficient permissions"
|
|
386
|
+
|
|
387
|
+
**Causa:** Service Principal não tem permissões na subscription/resource group.
|
|
388
|
+
|
|
389
|
+
**Solução:**
|
|
390
|
+
```bash
|
|
391
|
+
# Verificar role assignments
|
|
392
|
+
az role assignment list \
|
|
393
|
+
--assignee $SP_ID \
|
|
394
|
+
--output table
|
|
395
|
+
|
|
396
|
+
# Adicionar Contributor se necessário
|
|
397
|
+
az role assignment create \
|
|
398
|
+
--assignee $SP_ID \
|
|
399
|
+
--role Contributor \
|
|
400
|
+
--scope "/subscriptions/$SUBSCRIPTION_ID/resourceGroups/$RG_NAME"
|
|
401
|
+
```
|
|
402
|
+
|
|
403
|
+
### Erro: "Container registry not found"
|
|
404
|
+
|
|
405
|
+
**Causa:** Service Principal não tem permissão no ACR.
|
|
406
|
+
|
|
407
|
+
**Solução:**
|
|
408
|
+
```bash
|
|
409
|
+
# Adicionar AcrPush role
|
|
410
|
+
az role assignment create \
|
|
411
|
+
--assignee $SP_ID \
|
|
412
|
+
--role AcrPush \
|
|
413
|
+
--scope $ACR_ID
|
|
414
|
+
```
|
|
415
|
+
|
|
416
|
+
### Erro: "Pipeline not authorized to access service connection"
|
|
417
|
+
|
|
418
|
+
**Causa:** Pipeline não foi autorizado a usar a service connection.
|
|
419
|
+
|
|
420
|
+
**Solução:**
|
|
421
|
+
1. **Project Settings** → **Service connections**
|
|
422
|
+
2. Click na service connection
|
|
423
|
+
3. **Security** → Adicionar pipeline específico ou marcar "Grant access to all pipelines"
|
|
424
|
+
|
|
425
|
+
---
|
|
426
|
+
|
|
427
|
+
## 📚 Referências
|
|
428
|
+
|
|
429
|
+
- [Workload Identity Federation](https://learn.microsoft.com/azure/devops/pipelines/library/connect-to-azure)
|
|
430
|
+
- [Azure Pipelines YAML Schema](https://learn.microsoft.com/azure/devops/pipelines/yaml-schema)
|
|
431
|
+
- [Environments](https://learn.microsoft.com/azure/devops/pipelines/process/environments)
|
|
432
|
+
- [Service Connections](https://learn.microsoft.com/azure/devops/pipelines/library/service-endpoints)
|
|
433
|
+
|
|
434
|
+
---
|
|
435
|
+
|
|
436
|
+
## ✅ Checklist Final
|
|
437
|
+
|
|
438
|
+
Antes de ir para produção:
|
|
439
|
+
|
|
440
|
+
- [ ] Workload Identity configurada para dev/staging/prod
|
|
441
|
+
- [ ] Service connections criadas e testadas
|
|
442
|
+
- [ ] Variáveis configuradas (ACR_NAME, APP_NAME, SUBSCRIPTION_ID)
|
|
443
|
+
- [ ] Environments criados (dev, staging, production)
|
|
444
|
+
- [ ] Aprovações configuradas (production requer aprovação manual)
|
|
445
|
+
- [ ] Dev pipeline testado com sucesso
|
|
446
|
+
- [ ] Staging pipeline testado com sucesso
|
|
447
|
+
- [ ] Prod pipeline testado com aprovação
|
|
448
|
+
- [ ] Health checks funcionando
|
|
449
|
+
- [ ] Monitoring configurado (Application Insights)
|
|
450
|
+
- [ ] Rollback plan documentado
|
|
451
|
+
|
|
452
|
+
---
|
|
453
|
+
|
|
454
|
+
*MORPH-SPEC by Polymorphism Tech*
|