@polymorphism-tech/morph-spec 4.7.0 → 4.7.2

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.
Files changed (232) hide show
  1. package/.morph/.morphversion +5 -0
  2. package/.morph/analytics/threads-log.jsonl +5 -0
  3. package/.morph/config/config.json +8 -0
  4. package/.morph/framework/agents.json +1815 -0
  5. package/.morph/framework/hooks/README.md +205 -0
  6. package/.morph/framework/hooks/claude-code/notification/approval-reminder.js +54 -0
  7. package/.morph/framework/hooks/claude-code/post-tool-use/dispatch.js +83 -0
  8. package/.morph/framework/hooks/claude-code/post-tool-use/handle-tool-failure.js +42 -0
  9. package/.morph/framework/hooks/claude-code/pre-compact/save-morph-context.js +61 -0
  10. package/.morph/framework/hooks/claude-code/pre-tool-use/enforce-phase-writes.js +71 -0
  11. package/.morph/framework/hooks/claude-code/pre-tool-use/protect-readonly-files.js +58 -0
  12. package/.morph/framework/hooks/claude-code/pre-tool-use/protect-spec-files.js +64 -0
  13. package/.morph/framework/hooks/claude-code/session-start/inject-morph-context.js +94 -0
  14. package/.morph/framework/hooks/claude-code/statusline.py +538 -0
  15. package/.morph/framework/hooks/claude-code/statusline.sh +7 -0
  16. package/.morph/framework/hooks/claude-code/stop/validate-completion.js +88 -0
  17. package/.morph/framework/hooks/claude-code/user-prompt/enrich-prompt.js +91 -0
  18. package/.morph/framework/hooks/git/commit-msg/conventional-commits.sh +33 -0
  19. package/.morph/framework/hooks/git/pre-commit/agents.sh +25 -0
  20. package/.morph/framework/hooks/git/pre-commit/orchestrator.sh +64 -0
  21. package/.morph/framework/hooks/git/pre-commit/specs.sh +50 -0
  22. package/.morph/framework/hooks/git/pre-push/run-tests.sh +44 -0
  23. package/.morph/framework/hooks/shared/hook-response.js +45 -0
  24. package/.morph/framework/hooks/shared/phase-utils.js +129 -0
  25. package/.morph/framework/hooks/shared/state-reader.js +138 -0
  26. package/.morph/framework/hooks/shared/stdin-reader.js +26 -0
  27. package/.morph/framework/standards/STANDARDS.json +933 -0
  28. package/.morph/framework/standards/ai-agents/blazor-ui.md +364 -0
  29. package/.morph/framework/standards/ai-agents/production.md +415 -0
  30. package/.morph/framework/standards/ai-agents/setup.md +418 -0
  31. package/.morph/framework/standards/ai-agents/team-orchestration.md +479 -0
  32. package/.morph/framework/standards/ai-agents/workflows.md +354 -0
  33. package/.morph/framework/standards/architecture/ddd/aggregates.md +120 -0
  34. package/.morph/framework/standards/architecture/ddd/bounded-contexts.md +105 -0
  35. package/.morph/framework/standards/architecture/ddd/complexity-levels.md +108 -0
  36. package/.morph/framework/standards/architecture/ddd/entities.md +99 -0
  37. package/.morph/framework/standards/architecture/ddd/ubiquitous-language.md +58 -0
  38. package/.morph/framework/standards/architecture/ddd/value-objects.md +124 -0
  39. package/.morph/framework/standards/backend/api/minimal-api.md +494 -0
  40. package/.morph/framework/standards/backend/api/rest.md +492 -0
  41. package/.morph/framework/standards/backend/api/validation.md +88 -0
  42. package/.morph/framework/standards/backend/authentication/passkeys.md +428 -0
  43. package/.morph/framework/standards/backend/database/ef-core.md +199 -0
  44. package/.morph/framework/standards/backend/database/migrations.md +393 -0
  45. package/.morph/framework/standards/backend/database/postgresql/database.md +352 -0
  46. package/.morph/framework/standards/backend/database/repository-patterns.md +528 -0
  47. package/.morph/framework/standards/backend/database/vector-search-rag.md +541 -0
  48. package/.morph/framework/standards/backend/dotnet/async.md +366 -0
  49. package/.morph/framework/standards/backend/dotnet/core.md +117 -0
  50. package/.morph/framework/standards/backend/dotnet/di.md +439 -0
  51. package/.morph/framework/standards/backend/dotnet/program-cs-checklist.md +92 -0
  52. package/.morph/framework/standards/backend/integrations/asaas/asaas-api.md +216 -0
  53. package/.morph/framework/standards/backend/integrations/clerk/clerk-auth.md +290 -0
  54. package/.morph/framework/standards/backend/integrations/hangfire/hangfire-jobs.md +350 -0
  55. package/.morph/framework/standards/backend/integrations/resend/resend-email.md +385 -0
  56. package/.morph/framework/standards/context/analytics.md +96 -0
  57. package/.morph/framework/standards/context/bundles.md +110 -0
  58. package/.morph/framework/standards/context/priming.md +78 -0
  59. package/.morph/framework/standards/core/architecture.md +185 -0
  60. package/.morph/framework/standards/core/coding.md +214 -0
  61. package/.morph/framework/standards/core/git-branching-strategy.md +403 -0
  62. package/.morph/framework/standards/core/git.md +185 -0
  63. package/.morph/framework/standards/core/testing.md +295 -0
  64. package/.morph/framework/standards/data/nosql/blob-storage.md +102 -0
  65. package/.morph/framework/standards/data/nosql/cache/redis.md +97 -0
  66. package/.morph/framework/standards/data/nosql/cosmos-db.md +118 -0
  67. package/.morph/framework/standards/data/vector-search/azure-ai-search.md +121 -0
  68. package/.morph/framework/standards/data/vector-search/rag-chunking.md +104 -0
  69. package/.morph/framework/standards/frontend/blazor/design-checklist.md +222 -0
  70. package/.morph/framework/standards/frontend/blazor/fluent-ui-setup.md +595 -0
  71. package/.morph/framework/standards/frontend/blazor/fluent-ui.md +137 -0
  72. package/.morph/framework/standards/frontend/blazor/html-conversion.md +184 -0
  73. package/.morph/framework/standards/frontend/blazor/lifecycle.md +195 -0
  74. package/.morph/framework/standards/frontend/blazor/pitfalls.md +198 -0
  75. package/.morph/framework/standards/frontend/blazor/state.md +191 -0
  76. package/.morph/framework/standards/frontend/design-system/animations.md +151 -0
  77. package/.morph/framework/standards/frontend/design-system/naming.md +64 -0
  78. package/.morph/framework/standards/frontend/nextjs/app-router.md +123 -0
  79. package/.morph/framework/standards/frontend/nextjs/components.md +132 -0
  80. package/.morph/framework/standards/frontend/nextjs/data-fetching.md +126 -0
  81. package/.morph/framework/standards/frontend/nextjs/forms.md +128 -0
  82. package/.morph/framework/standards/frontend/nextjs/naming-conventions.md +67 -0
  83. package/.morph/framework/standards/frontend/nextjs/nextjs-patterns.md +215 -0
  84. package/.morph/framework/standards/frontend/nextjs/project-structure.md +102 -0
  85. package/.morph/framework/standards/frontend/nextjs/state-management.md +72 -0
  86. package/.morph/framework/standards/frontend/nextjs/testing.md +111 -0
  87. package/.morph/framework/standards/infrastructure/azure/azure.md +624 -0
  88. package/.morph/framework/standards/infrastructure/azure/bicep/bicep-patterns.md +422 -0
  89. package/.morph/framework/standards/infrastructure/azure/devops/azure-devops-setup.md +516 -0
  90. package/.morph/framework/standards/infrastructure/azure/devops/local-development.md +520 -0
  91. package/.morph/framework/standards/infrastructure/azure/services/functions.md +486 -0
  92. package/.morph/framework/standards/infrastructure/azure/services/service-bus.md +459 -0
  93. package/.morph/framework/standards/infrastructure/azure/services/storage.md +407 -0
  94. package/.morph/framework/standards/infrastructure/docker/easypanel-deploy.md +196 -0
  95. package/.morph/framework/standards/infrastructure/supabase/mcp-setup.md +252 -0
  96. package/.morph/framework/standards/infrastructure/supabase/supabase-auth.md +176 -0
  97. package/.morph/framework/standards/infrastructure/supabase/supabase-pgvector.md +169 -0
  98. package/.morph/framework/standards/infrastructure/supabase/supabase-rls.md +184 -0
  99. package/.morph/framework/standards/infrastructure/supabase/supabase-storage.md +153 -0
  100. package/.morph/framework/standards/integration/api/graphql.md +91 -0
  101. package/.morph/framework/standards/integration/api/grpc.md +114 -0
  102. package/.morph/framework/standards/integration/api/rest-design.md +95 -0
  103. package/.morph/framework/standards/integration/event-driven/cqrs.md +101 -0
  104. package/.morph/framework/standards/integration/event-driven/event-sourcing.md +124 -0
  105. package/.morph/framework/standards/integration/event-driven/service-bus.md +95 -0
  106. package/.morph/framework/standards/integration/mcp/mcp-tools.md +384 -0
  107. package/.morph/framework/standards/observability/logging.md +131 -0
  108. package/.morph/framework/standards/observability/metrics.md +121 -0
  109. package/.morph/framework/standards/observability/monitoring.md +114 -0
  110. package/.morph/framework/standards/observability/tracing.md +132 -0
  111. package/.morph/framework/standards/workflows/parallel-execution.md +112 -0
  112. package/.morph/framework/standards/workflows/thread-management.md +113 -0
  113. package/.morph/framework/templates/.idea/morph-templates.xml +92 -0
  114. package/.morph/framework/templates/.vscode/morph-templates.code-snippets +186 -0
  115. package/.morph/framework/templates/IDE-SNIPPETS.md +266 -0
  116. package/.morph/framework/templates/README.md +814 -0
  117. package/.morph/framework/templates/REGISTRY.json +1888 -0
  118. package/.morph/framework/templates/code/dotnet/backend/repository.cs +141 -0
  119. package/.morph/framework/templates/code/dotnet/backend/service.cs +139 -0
  120. package/.morph/framework/templates/code/dotnet/contracts/Commands.cs +74 -0
  121. package/.morph/framework/templates/code/dotnet/contracts/Entities.cs +25 -0
  122. package/.morph/framework/templates/code/dotnet/contracts/Queries.cs +74 -0
  123. package/.morph/framework/templates/code/dotnet/contracts/README.md +74 -0
  124. package/.morph/framework/templates/code/dotnet/contracts/api-contracts.cs +173 -0
  125. package/.morph/framework/templates/code/dotnet/contracts/contracts-level1.cs +69 -0
  126. package/.morph/framework/templates/code/dotnet/contracts/contracts-level2.cs +86 -0
  127. package/.morph/framework/templates/code/dotnet/contracts/contracts-level3.cs +41 -0
  128. package/.morph/framework/templates/code/dotnet/database/migration.cs +83 -0
  129. package/.morph/framework/templates/code/dotnet/frontend/component.razor +239 -0
  130. package/.morph/framework/templates/code/dotnet/jobs/agent.cs +163 -0
  131. package/.morph/framework/templates/code/dotnet/jobs/job.cs +171 -0
  132. package/.morph/framework/templates/code/dotnet/test.cs +239 -0
  133. package/.morph/framework/templates/code/sql/rls-policy.sql +57 -0
  134. package/.morph/framework/templates/code/sql/supabase-migration.sql +100 -0
  135. package/.morph/framework/templates/code/sql/supabase-migration.template.sql +113 -0
  136. package/.morph/framework/templates/code/typescript/contracts.ts +168 -0
  137. package/.morph/framework/templates/context/CONTEXT-FEATURE.md +276 -0
  138. package/.morph/framework/templates/context/CONTEXT.md +181 -0
  139. package/.morph/framework/templates/docs/clarifications.md +253 -0
  140. package/.morph/framework/templates/docs/onboarding.md +123 -0
  141. package/.morph/framework/templates/docs/proposal.md +182 -0
  142. package/.morph/framework/templates/docs/schema-analysis.md +119 -0
  143. package/.morph/framework/templates/docs/spec.md +198 -0
  144. package/.morph/framework/templates/docs/ui-components.md +124 -0
  145. package/.morph/framework/templates/docs/ui-design-system.md +76 -0
  146. package/.morph/framework/templates/docs/ui-flows.md +167 -0
  147. package/.morph/framework/templates/docs/ui-mockups.md +98 -0
  148. package/.morph/framework/templates/docs/user-stories.md +34 -0
  149. package/.morph/framework/templates/examples/design-system-examples.md +357 -0
  150. package/.morph/framework/templates/examples/spec-examples.md +90 -0
  151. package/.morph/framework/templates/feature/decisions.md +187 -0
  152. package/.morph/framework/templates/feature/recap.md +146 -0
  153. package/.morph/framework/templates/feature/tasks.md +199 -0
  154. package/.morph/framework/templates/frontend/nextjs/Dockerfile.nextjs.hbs +43 -0
  155. package/.morph/framework/templates/frontend/nextjs/client-component.tsx.hbs +26 -0
  156. package/.morph/framework/templates/frontend/nextjs/env.mjs.hbs +32 -0
  157. package/.morph/framework/templates/frontend/nextjs/feature-form.tsx.hbs +56 -0
  158. package/.morph/framework/templates/frontend/nextjs/page.tsx.hbs +22 -0
  159. package/.morph/framework/templates/frontend/nextjs/tsconfig.json.hbs +26 -0
  160. package/.morph/framework/templates/frontend/nextjs/use-feature.ts.hbs +54 -0
  161. package/.morph/framework/templates/infrastructure/azure/Dockerfile.example +82 -0
  162. package/.morph/framework/templates/infrastructure/azure/README.md +286 -0
  163. package/.morph/framework/templates/infrastructure/azure/app-insights.bicep +63 -0
  164. package/.morph/framework/templates/infrastructure/azure/app-service.bicep +164 -0
  165. package/.morph/framework/templates/infrastructure/azure/container-app-env.bicep +49 -0
  166. package/.morph/framework/templates/infrastructure/azure/container-app.bicep +156 -0
  167. package/.morph/framework/templates/infrastructure/azure/deploy-checklist.md +426 -0
  168. package/.morph/framework/templates/infrastructure/azure/deploy.ps1 +229 -0
  169. package/.morph/framework/templates/infrastructure/azure/deploy.sh +208 -0
  170. package/.morph/framework/templates/infrastructure/azure/key-vault.bicep +91 -0
  171. package/.morph/framework/templates/infrastructure/azure/main.bicep +189 -0
  172. package/.morph/framework/templates/infrastructure/azure/parameters.dev.json +29 -0
  173. package/.morph/framework/templates/infrastructure/azure/parameters.prod.json +29 -0
  174. package/.morph/framework/templates/infrastructure/azure/parameters.staging.json +29 -0
  175. package/.morph/framework/templates/infrastructure/azure/sql-database.bicep +103 -0
  176. package/.morph/framework/templates/infrastructure/azure/storage.bicep +106 -0
  177. package/.morph/framework/templates/infrastructure/docker/Dockerfile.template +58 -0
  178. package/.morph/framework/templates/infrastructure/docker/docker-compose.template.yml +67 -0
  179. package/.morph/framework/templates/infrastructure/docker/dockerfile-api.dockerfile +38 -0
  180. package/.morph/framework/templates/infrastructure/docker/dockerfile-web.dockerfile +48 -0
  181. package/.morph/framework/templates/infrastructure/docker/easypanel.template.json +54 -0
  182. package/.morph/framework/templates/infrastructure/github/README.md +593 -0
  183. package/.morph/framework/templates/infrastructure/github/actions/azure-auth/action.yml.hbs +22 -0
  184. package/.morph/framework/templates/infrastructure/github/actions/docker-build-push/action.yml.hbs +45 -0
  185. package/.morph/framework/templates/infrastructure/github/actions/health-check/action.yml.hbs +27 -0
  186. package/.morph/framework/templates/infrastructure/github/workflows/deploy-azure-app-service.yml.hbs +61 -0
  187. package/.morph/framework/templates/infrastructure/github/workflows/deploy-easypanel.yml.hbs +31 -0
  188. package/.morph/framework/templates/infrastructure/github/workflows/docker-build-push.yml.hbs +59 -0
  189. package/.morph/framework/templates/infrastructure/github/workflows/dotnet-build.yml.hbs +39 -0
  190. package/.morph/framework/templates/integrations/asaas-client.cs +387 -0
  191. package/.morph/framework/templates/integrations/asaas-webhook.cs +351 -0
  192. package/.morph/framework/templates/integrations/azure-identity-config.cs +288 -0
  193. package/.morph/framework/templates/integrations/clerk-config.cs +258 -0
  194. package/.morph/framework/templates/meta-prompts/fusion/fusion-agent.md +76 -0
  195. package/.morph/framework/templates/meta-prompts/fusion/fusion-aggregator.md +100 -0
  196. package/.morph/framework/templates/meta-prompts/hops/hop-retry.md +78 -0
  197. package/.morph/framework/templates/meta-prompts/hops/hop-validation.md +97 -0
  198. package/.morph/framework/templates/meta-prompts/hops/hop-wrapper.md +36 -0
  199. package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-coordinator.md +113 -0
  200. package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-worker.md +80 -0
  201. package/.morph/framework/templates/meta-prompts/squad-leaders/backend-squad.md +90 -0
  202. package/.morph/framework/templates/meta-prompts/squad-leaders/frontend-squad.md +126 -0
  203. package/.morph/framework/templates/meta-prompts/squad-leaders/squad-leader.md +43 -0
  204. package/.morph/framework/templates/meta-prompts/validators/checkpoint-validator.md +107 -0
  205. package/.morph/framework/templates/meta-prompts/validators/pre-commit-validator.md +95 -0
  206. package/.morph/framework/templates/project-structure/dotnet-ddd.md +70 -0
  207. package/.morph/framework/templates/saas/subscription.cs +347 -0
  208. package/.morph/framework/templates/saas/tenant.cs +338 -0
  209. package/.morph/framework/templates/state.template.json +17 -0
  210. package/.morph/framework/templates/ui/FluentDesignTheme.cs +149 -0
  211. package/.morph/framework/templates/ui/MudTheme.cs +281 -0
  212. package/.morph/framework/templates/ui/design-system.css +226 -0
  213. package/.morph/logs/tool-failures.log +17 -0
  214. package/.morph/memory/pre-compact-2026-02-24T17-43-30-049Z.json +16 -0
  215. package/.morph/plans/eager-watching-bunny.md +105 -0
  216. package/.morph/plans/temporal-seeking-nebula.md +45 -0
  217. package/.morph/state.json +48 -0
  218. package/CLAUDE.md +1 -1
  219. package/README.md +119 -99
  220. package/bin/morph-spec.js +0 -9
  221. package/framework/CLAUDE.md +1 -1
  222. package/framework/hooks/README.md +10 -6
  223. package/framework/hooks/claude-code/notification/approval-reminder.js +2 -0
  224. package/framework/hooks/claude-code/post-tool-use/dispatch.js +1 -1
  225. package/framework/hooks/claude-code/stop/validate-completion.js +1 -1
  226. package/framework/hooks/claude-code/user-prompt/enrich-prompt.js +1 -1
  227. package/package.json +1 -1
  228. package/src/commands/project/init.js +15 -42
  229. package/src/commands/project/update.js +22 -37
  230. package/src/lib/installers/mcp-installer.js +18 -3
  231. package/src/utils/hooks-installer.js +5 -15
  232. package/src/commands/project/detect.js +0 -114
@@ -0,0 +1,208 @@
1
+ #!/bin/bash
2
+ # ==============================================================================
3
+ # MORPH-SPEC - Deploy Script
4
+ # Automated deployment of Azure infrastructure
5
+ # ==============================================================================
6
+
7
+ set -e # Exit on error
8
+
9
+ # ==============================================================================
10
+ # CONFIGURATION
11
+ # ==============================================================================
12
+
13
+ # Required variables (override via environment)
14
+ APP_NAME="${APP_NAME:-myapp}"
15
+ ENVIRONMENT="${ENVIRONMENT:-dev}"
16
+ LOCATION="${LOCATION:-eastus}"
17
+ SUBSCRIPTION_ID="${SUBSCRIPTION_ID:-}"
18
+
19
+ # Optional variables
20
+ HOSTING_TYPE="${HOSTING_TYPE:-appservice}" # appservice or containerapp
21
+ APP_SERVICE_SKU="${APP_SERVICE_SKU:-F1}"
22
+ CONTAINER_IMAGE="${CONTAINER_IMAGE:-mcr.microsoft.com/hello-world:latest}"
23
+
24
+ # Derived variables
25
+ RESOURCE_GROUP="rg-${APP_NAME}-${ENVIRONMENT}"
26
+ DEPLOYMENT_NAME="deploy-${APP_NAME}-$(date +%Y%m%d-%H%M%S)"
27
+
28
+ # ==============================================================================
29
+ # COLORS
30
+ # ==============================================================================
31
+
32
+ RED='\033[0;31m'
33
+ GREEN='\033[0;32m'
34
+ YELLOW='\033[1;33m'
35
+ BLUE='\033[0;34m'
36
+ NC='\033[0m' # No Color
37
+
38
+ # ==============================================================================
39
+ # FUNCTIONS
40
+ # ==============================================================================
41
+
42
+ log_info() {
43
+ echo -e "${BLUE}ℹ️ $1${NC}"
44
+ }
45
+
46
+ log_success() {
47
+ echo -e "${GREEN}✅ $1${NC}"
48
+ }
49
+
50
+ log_warning() {
51
+ echo -e "${YELLOW}⚠️ $1${NC}"
52
+ }
53
+
54
+ log_error() {
55
+ echo -e "${RED}❌ $1${NC}"
56
+ }
57
+
58
+ check_prerequisites() {
59
+ log_info "Checking prerequisites..."
60
+
61
+ # Check Azure CLI
62
+ if ! command -v az &> /dev/null; then
63
+ log_error "Azure CLI not found. Install from: https://aka.ms/azure-cli"
64
+ exit 1
65
+ fi
66
+
67
+ # Check login
68
+ if ! az account show &> /dev/null; then
69
+ log_error "Not logged in to Azure. Run: az login"
70
+ exit 1
71
+ fi
72
+
73
+ # Set subscription if provided
74
+ if [ -n "$SUBSCRIPTION_ID" ]; then
75
+ log_info "Setting subscription to: $SUBSCRIPTION_ID"
76
+ az account set --subscription "$SUBSCRIPTION_ID"
77
+ fi
78
+
79
+ log_success "Prerequisites checked"
80
+ }
81
+
82
+ create_resource_group() {
83
+ log_info "Creating resource group: $RESOURCE_GROUP"
84
+
85
+ if az group exists -n "$RESOURCE_GROUP" | grep -q true; then
86
+ log_warning "Resource group already exists"
87
+ else
88
+ az group create \
89
+ --name "$RESOURCE_GROUP" \
90
+ --location "$LOCATION" \
91
+ --tags \
92
+ environment="$ENVIRONMENT" \
93
+ application="$APP_NAME" \
94
+ managedBy="bicep" \
95
+ framework="morph-spec"
96
+
97
+ log_success "Resource group created"
98
+ fi
99
+ }
100
+
101
+ generate_sql_password() {
102
+ # Generate secure random password
103
+ SQL_PASSWORD=$(openssl rand -base64 32 | tr -d "=+/" | cut -c1-25)
104
+ log_success "SQL password generated (stored in Key Vault after deploy)"
105
+ }
106
+
107
+ deploy_infrastructure() {
108
+ log_info "Deploying infrastructure..."
109
+ log_info " App Name: $APP_NAME"
110
+ log_info " Environment: $ENVIRONMENT"
111
+ log_info " Hosting Type: $HOSTING_TYPE"
112
+ log_info " Location: $LOCATION"
113
+
114
+ # Prepare parameters file
115
+ PARAMS_FILE="parameters.${ENVIRONMENT}.json"
116
+
117
+ if [ ! -f "$PARAMS_FILE" ]; then
118
+ log_error "Parameters file not found: $PARAMS_FILE"
119
+ exit 1
120
+ fi
121
+
122
+ # Deploy
123
+ az deployment group create \
124
+ --resource-group "$RESOURCE_GROUP" \
125
+ --name "$DEPLOYMENT_NAME" \
126
+ --template-file main.bicep \
127
+ --parameters "@$PARAMS_FILE" \
128
+ --parameters \
129
+ appName="$APP_NAME" \
130
+ environment="$ENVIRONMENT" \
131
+ location="$LOCATION" \
132
+ hostingType="$HOSTING_TYPE" \
133
+ appServiceSku="$APP_SERVICE_SKU" \
134
+ containerImage="$CONTAINER_IMAGE" \
135
+ sqlAdminPassword="$SQL_PASSWORD" \
136
+ --output table
137
+
138
+ log_success "Infrastructure deployed"
139
+ }
140
+
141
+ show_outputs() {
142
+ log_info "Retrieving deployment outputs..."
143
+
144
+ APP_URL=$(az deployment group show \
145
+ -g "$RESOURCE_GROUP" \
146
+ -n "$DEPLOYMENT_NAME" \
147
+ --query properties.outputs.appUrl.value -o tsv)
148
+
149
+ SQL_CONNECTION=$(az deployment group show \
150
+ -g "$RESOURCE_GROUP" \
151
+ -n "$DEPLOYMENT_NAME" \
152
+ --query properties.outputs.sqlConnectionString.value -o tsv)
153
+
154
+ APPINSIGHTS_CONNECTION=$(az deployment group show \
155
+ -g "$RESOURCE_GROUP" \
156
+ -n "$DEPLOYMENT_NAME" \
157
+ --query properties.outputs.appInsightsConnectionString.value -o tsv)
158
+
159
+ echo ""
160
+ echo "╔════════════════════════════════════════════════════════════════╗"
161
+ echo "║ DEPLOYMENT SUCCESSFUL ║"
162
+ echo "╚════════════════════════════════════════════════════════════════╝"
163
+ echo ""
164
+ echo "🌐 Application URL:"
165
+ echo " $APP_URL"
166
+ echo ""
167
+ echo "🗄️ SQL Connection String:"
168
+ echo " $SQL_CONNECTION"
169
+ echo ""
170
+ echo "📊 App Insights Connection String:"
171
+ echo " $APPINSIGHTS_CONNECTION"
172
+ echo ""
173
+ echo "💡 Next steps:"
174
+
175
+ if [ "$HOSTING_TYPE" = "appservice" ]; then
176
+ echo " 1. Deploy your code: az webapp up --name app-${APP_NAME}-${ENVIRONMENT}"
177
+ echo " 2. View logs: az webapp log tail --name app-${APP_NAME}-${ENVIRONMENT} -g $RESOURCE_GROUP"
178
+ else
179
+ echo " 1. Build and push container: az acr build --registry <ACR> --image ${APP_NAME}:latest ."
180
+ echo " 2. Update container app: az containerapp update -n ca-${APP_NAME}-${ENVIRONMENT} -g $RESOURCE_GROUP --image <IMAGE>"
181
+ echo " 3. View logs: az containerapp logs show -n ca-${APP_NAME}-${ENVIRONMENT} -g $RESOURCE_GROUP --follow"
182
+ fi
183
+
184
+ echo ""
185
+ }
186
+
187
+ # ==============================================================================
188
+ # MAIN
189
+ # ==============================================================================
190
+
191
+ main() {
192
+ echo ""
193
+ echo "╔════════════════════════════════════════════════════════════════╗"
194
+ echo "║ MORPH-SPEC - Azure Infrastructure Deploy ║"
195
+ echo "╚════════════════════════════════════════════════════════════════╝"
196
+ echo ""
197
+
198
+ check_prerequisites
199
+ create_resource_group
200
+ generate_sql_password
201
+ deploy_infrastructure
202
+ show_outputs
203
+
204
+ log_success "Deployment complete! 🚀"
205
+ }
206
+
207
+ # Run main function
208
+ main
@@ -0,0 +1,91 @@
1
+ // ==============================================================================
2
+ // MORPH-SPEC - Key Vault
3
+ // Azure Key Vault for secrets management
4
+ // ==============================================================================
5
+
6
+ @description('Key Vault name')
7
+ @minLength(3)
8
+ @maxLength(24)
9
+ param name string
10
+
11
+ @description('Location')
12
+ param location string
13
+
14
+ @description('Tags')
15
+ param tags object = {}
16
+
17
+ @description('Enable soft delete')
18
+ param enableSoftDelete bool = true
19
+
20
+ @description('Soft delete retention days')
21
+ @minValue(7)
22
+ @maxValue(90)
23
+ param softDeleteRetentionDays int = 30
24
+
25
+ @description('Enable purge protection')
26
+ param enablePurgeProtection bool = false
27
+
28
+ @description('Object IDs to grant access (optional)')
29
+ param accessPoliciesObjectIds array = []
30
+
31
+ // ==============================================================================
32
+ // KEY VAULT
33
+ // ==============================================================================
34
+
35
+ resource keyVault 'Microsoft.KeyVault/vaults@2023-07-01' = {
36
+ name: name
37
+ location: location
38
+ tags: tags
39
+ properties: {
40
+ tenantId: subscription().tenantId
41
+ sku: {
42
+ family: 'A'
43
+ name: 'standard'
44
+ }
45
+ enabledForDeployment: true
46
+ enabledForDiskEncryption: false
47
+ enabledForTemplateDeployment: true
48
+ enableSoftDelete: enableSoftDelete
49
+ softDeleteRetentionInDays: softDeleteRetentionDays
50
+ enablePurgeProtection: enablePurgeProtection ? true : null
51
+ enableRbacAuthorization: true
52
+ publicNetworkAccess: 'Enabled'
53
+ networkAcls: {
54
+ defaultAction: 'Allow'
55
+ bypass: 'AzureServices'
56
+ }
57
+ }
58
+ }
59
+
60
+ // ==============================================================================
61
+ // ACCESS POLICIES (Optional - if not using RBAC)
62
+ // ==============================================================================
63
+
64
+ resource accessPolicies 'Microsoft.KeyVault/vaults/accessPolicies@2023-07-01' = if (length(accessPoliciesObjectIds) > 0) {
65
+ parent: keyVault
66
+ name: 'add'
67
+ properties: {
68
+ accessPolicies: [for objectId in accessPoliciesObjectIds: {
69
+ tenantId: subscription().tenantId
70
+ objectId: objectId
71
+ permissions: {
72
+ secrets: ['get', 'list', 'set', 'delete']
73
+ keys: ['get', 'list', 'create', 'delete']
74
+ certificates: ['get', 'list', 'create', 'delete']
75
+ }
76
+ }]
77
+ }
78
+ }
79
+
80
+ // ==============================================================================
81
+ // OUTPUTS
82
+ // ==============================================================================
83
+
84
+ @description('Key Vault ID')
85
+ output id string = keyVault.id
86
+
87
+ @description('Key Vault name')
88
+ output name string = keyVault.name
89
+
90
+ @description('Key Vault URI')
91
+ output uri string = keyVault.properties.vaultUri
@@ -0,0 +1,189 @@
1
+ // ==============================================================================
2
+ // MORPH-SPEC - Main Bicep Template
3
+ // Entry point para infraestrutura Azure
4
+ // ==============================================================================
5
+
6
+ targetScope = 'resourceGroup'
7
+
8
+ // ==============================================================================
9
+ // PARAMETERS
10
+ // ==============================================================================
11
+
12
+ @description('Environment name (dev, staging, prod)')
13
+ @allowed(['dev', 'staging', 'prod'])
14
+ param environment string = 'dev'
15
+
16
+ @description('Location for all resources')
17
+ param location string = resourceGroup().location
18
+
19
+ @description('Application name (used for naming resources)')
20
+ @minLength(3)
21
+ @maxLength(15)
22
+ param appName string
23
+
24
+ @description('SQL Server administrator password')
25
+ @secure()
26
+ param sqlAdminPassword string
27
+
28
+ @description('Container image to deploy (only for Container Apps)')
29
+ param containerImage string = 'mcr.microsoft.com/hello-world:latest'
30
+
31
+ @description('Hosting type: appservice or containerapp')
32
+ @allowed(['appservice', 'containerapp'])
33
+ param hostingType string = 'appservice'
34
+
35
+ @description('App Service SKU (only for App Service hosting)')
36
+ @allowed(['F1', 'B1', 'S1', 'P1v2'])
37
+ param appServiceSku string = 'F1'
38
+
39
+ // ==============================================================================
40
+ // VARIABLES
41
+ // ==============================================================================
42
+
43
+ var resourcePrefix = '${appName}-${environment}'
44
+ var tags = {
45
+ environment: environment
46
+ application: appName
47
+ managedBy: 'bicep'
48
+ framework: 'morph-spec'
49
+ }
50
+
51
+ // ==============================================================================
52
+ // LOG ANALYTICS WORKSPACE
53
+ // Required for Container Apps and Application Insights
54
+ // ==============================================================================
55
+
56
+ resource logAnalytics 'Microsoft.OperationalInsights/workspaces@2022-10-01' = {
57
+ name: '${resourcePrefix}-logs'
58
+ location: location
59
+ tags: tags
60
+ properties: {
61
+ sku: {
62
+ name: 'PerGB2018'
63
+ }
64
+ retentionInDays: environment == 'prod' ? 90 : 30
65
+ }
66
+ }
67
+
68
+ // ==============================================================================
69
+ // MODULES
70
+ // ==============================================================================
71
+
72
+ // Application Insights
73
+ module appInsights 'app-insights.bicep' = {
74
+ name: 'appInsights-${uniqueString(resourceGroup().id)}'
75
+ params: {
76
+ name: '${resourcePrefix}-insights'
77
+ location: location
78
+ tags: tags
79
+ logAnalyticsWorkspaceId: logAnalytics.id
80
+ }
81
+ }
82
+
83
+ // Key Vault
84
+ module keyVault 'key-vault.bicep' = {
85
+ name: 'keyVault-${uniqueString(resourceGroup().id)}'
86
+ params: {
87
+ name: '${resourcePrefix}-kv'
88
+ location: location
89
+ tags: tags
90
+ }
91
+ }
92
+
93
+ // Storage Account
94
+ module storage 'storage.bicep' = {
95
+ name: 'storage-${uniqueString(resourceGroup().id)}'
96
+ params: {
97
+ name: replace('${resourcePrefix}st', '-', '')
98
+ location: location
99
+ tags: tags
100
+ sku: environment == 'prod' ? 'Standard_GRS' : 'Standard_LRS'
101
+ }
102
+ }
103
+
104
+ // SQL Database
105
+ module sqlDatabase 'sql-database.bicep' = {
106
+ name: 'sqlDatabase-${uniqueString(resourceGroup().id)}'
107
+ params: {
108
+ serverName: '${resourcePrefix}-sql'
109
+ databaseName: appName
110
+ location: location
111
+ tags: tags
112
+ adminPassword: sqlAdminPassword
113
+ useFree: environment == 'dev'
114
+ }
115
+ }
116
+
117
+ // ==============================================================================
118
+ // HOSTING - App Service (Conditional)
119
+ // ==============================================================================
120
+
121
+ module appService 'app-service.bicep' = if (hostingType == 'appservice') {
122
+ name: 'appService-${uniqueString(resourceGroup().id)}'
123
+ params: {
124
+ name: 'app-${resourcePrefix}'
125
+ location: location
126
+ tags: tags
127
+ sku: appServiceSku
128
+ appInsightsConnectionString: appInsights.outputs.connectionString
129
+ alwaysOn: appServiceSku != 'F1' // Only available on paid tiers
130
+ }
131
+ }
132
+
133
+ // ==============================================================================
134
+ // HOSTING - Container Apps (Conditional)
135
+ // ==============================================================================
136
+
137
+ // Container Apps Environment
138
+ module containerAppEnv 'container-app-env.bicep' = if (hostingType == 'containerapp') {
139
+ name: 'containerAppEnv-${uniqueString(resourceGroup().id)}'
140
+ params: {
141
+ name: '${resourcePrefix}-env'
142
+ location: location
143
+ tags: tags
144
+ logAnalyticsWorkspaceId: logAnalytics.id
145
+ }
146
+ }
147
+
148
+ // Container App
149
+ module containerApp 'container-app.bicep' = if (hostingType == 'containerapp') {
150
+ name: 'containerApp-${uniqueString(resourceGroup().id)}'
151
+ params: {
152
+ name: 'ca-${resourcePrefix}'
153
+ location: location
154
+ tags: tags
155
+ environmentId: hostingType == 'containerapp' ? containerAppEnv.outputs.id : ''
156
+ containerImage: containerImage
157
+ appInsightsConnectionString: appInsights.outputs.connectionString
158
+ minReplicas: environment == 'prod' ? 1 : 0
159
+ maxReplicas: environment == 'prod' ? 10 : 3
160
+ }
161
+ }
162
+
163
+ // ==============================================================================
164
+ // OUTPUTS
165
+ // ==============================================================================
166
+
167
+ @description('Application URL')
168
+ output appUrl string = hostingType == 'appservice' ? appService.outputs.url : containerApp.outputs.url
169
+
170
+ @description('Hosting Type')
171
+ output hostingType string = hostingType
172
+
173
+ @description('SQL Server connection string')
174
+ output sqlConnectionString string = sqlDatabase.outputs.connectionString
175
+
176
+ @description('Key Vault URI')
177
+ output keyVaultUri string = keyVault.outputs.uri
178
+
179
+ @description('Storage Account connection string')
180
+ output storageConnectionString string = storage.outputs.connectionString
181
+
182
+ @description('Application Insights connection string')
183
+ output appInsightsConnectionString string = appInsights.outputs.connectionString
184
+
185
+ @description('Log Analytics Workspace ID')
186
+ output logAnalyticsWorkspaceId string = logAnalytics.id
187
+
188
+ @description('App Service Principal ID (for Managed Identity)')
189
+ output appPrincipalId string = hostingType == 'appservice' ? appService.outputs.principalId : ''
@@ -0,0 +1,29 @@
1
+ {
2
+ "$schema": "https://schema.management.azure.com/schemas/2019-04-01/deploymentParameters.json#",
3
+ "contentVersion": "1.0.0.0",
4
+ "parameters": {
5
+ "environment": {
6
+ "value": "dev"
7
+ },
8
+ "appName": {
9
+ "value": "{{APP_NAME}}"
10
+ },
11
+ "sqlAdminPassword": {
12
+ "reference": {
13
+ "keyVault": {
14
+ "id": "/subscriptions/{{SUBSCRIPTION_ID}}/resourceGroups/{{RESOURCE_GROUP}}/providers/Microsoft.KeyVault/vaults/{{KEY_VAULT_NAME}}"
15
+ },
16
+ "secretName": "sql-admin-password"
17
+ }
18
+ },
19
+ "hostingType": {
20
+ "value": "appservice"
21
+ },
22
+ "appServiceSku": {
23
+ "value": "F1"
24
+ },
25
+ "containerImage": {
26
+ "value": "{{ACR_NAME}}.azurecr.io/{{APP_NAME}}:latest"
27
+ }
28
+ }
29
+ }
@@ -0,0 +1,29 @@
1
+ {
2
+ "$schema": "https://schema.management.azure.com/schemas/2019-04-01/deploymentParameters.json#",
3
+ "contentVersion": "1.0.0.0",
4
+ "parameters": {
5
+ "environment": {
6
+ "value": "prod"
7
+ },
8
+ "appName": {
9
+ "value": "{{APP_NAME}}"
10
+ },
11
+ "sqlAdminPassword": {
12
+ "reference": {
13
+ "keyVault": {
14
+ "id": "/subscriptions/{{SUBSCRIPTION_ID}}/resourceGroups/{{RESOURCE_GROUP}}/providers/Microsoft.KeyVault/vaults/{{KEY_VAULT_NAME}}"
15
+ },
16
+ "secretName": "sql-admin-password"
17
+ }
18
+ },
19
+ "hostingType": {
20
+ "value": "containerapp"
21
+ },
22
+ "appServiceSku": {
23
+ "value": "P1v2"
24
+ },
25
+ "containerImage": {
26
+ "value": "{{ACR_NAME}}.azurecr.io/{{APP_NAME}}:{{VERSION}}"
27
+ }
28
+ }
29
+ }
@@ -0,0 +1,29 @@
1
+ {
2
+ "$schema": "https://schema.management.azure.com/schemas/2019-04-01/deploymentParameters.json#",
3
+ "contentVersion": "1.0.0.0",
4
+ "parameters": {
5
+ "environment": {
6
+ "value": "staging"
7
+ },
8
+ "appName": {
9
+ "value": "{{APP_NAME}}"
10
+ },
11
+ "sqlAdminPassword": {
12
+ "reference": {
13
+ "keyVault": {
14
+ "id": "/subscriptions/{{SUBSCRIPTION_ID}}/resourceGroups/{{RESOURCE_GROUP}}/providers/Microsoft.KeyVault/vaults/{{KEY_VAULT_NAME}}"
15
+ },
16
+ "secretName": "sql-admin-password"
17
+ }
18
+ },
19
+ "hostingType": {
20
+ "value": "containerapp"
21
+ },
22
+ "appServiceSku": {
23
+ "value": "B1"
24
+ },
25
+ "containerImage": {
26
+ "value": "{{ACR_NAME}}.azurecr.io/{{APP_NAME}}:latest"
27
+ }
28
+ }
29
+ }
@@ -0,0 +1,103 @@
1
+ // ==============================================================================
2
+ // MORPH-SPEC - SQL Database
3
+ // Azure SQL Server with Database (supports Free tier)
4
+ // ==============================================================================
5
+
6
+ @description('SQL Server name')
7
+ param serverName string
8
+
9
+ @description('Database name')
10
+ param databaseName string
11
+
12
+ @description('Location')
13
+ param location string
14
+
15
+ @description('Tags')
16
+ param tags object = {}
17
+
18
+ @description('Admin username')
19
+ param adminUsername string = 'sqladmin'
20
+
21
+ @description('Admin password')
22
+ @secure()
23
+ param adminPassword string
24
+
25
+ @description('Use free tier (32GB, limited DTUs)')
26
+ param useFree bool = true
27
+
28
+ // ==============================================================================
29
+ // SQL SERVER
30
+ // ==============================================================================
31
+
32
+ resource sqlServer 'Microsoft.Sql/servers@2023-05-01-preview' = {
33
+ name: serverName
34
+ location: location
35
+ tags: tags
36
+ properties: {
37
+ administratorLogin: adminUsername
38
+ administratorLoginPassword: adminPassword
39
+ version: '12.0'
40
+ minimalTlsVersion: '1.2'
41
+ publicNetworkAccess: 'Enabled'
42
+ }
43
+ }
44
+
45
+ // ==============================================================================
46
+ // SQL DATABASE
47
+ // ==============================================================================
48
+
49
+ resource sqlDatabase 'Microsoft.Sql/servers/databases@2023-05-01-preview' = {
50
+ parent: sqlServer
51
+ name: databaseName
52
+ location: location
53
+ tags: tags
54
+ sku: useFree ? {
55
+ name: 'Free'
56
+ tier: 'Free'
57
+ } : {
58
+ name: 'Basic'
59
+ tier: 'Basic'
60
+ capacity: 5
61
+ }
62
+ properties: {
63
+ collation: 'SQL_Latin1_General_CP1_CI_AS'
64
+ maxSizeBytes: useFree ? 32212254720 : 2147483648 // 32GB free, 2GB basic
65
+ catalogCollation: 'SQL_Latin1_General_CP1_CI_AS'
66
+ zoneRedundant: false
67
+ readScale: 'Disabled'
68
+ requestedBackupStorageRedundancy: 'Local'
69
+ }
70
+ }
71
+
72
+ // ==============================================================================
73
+ // FIREWALL RULES
74
+ // ==============================================================================
75
+
76
+ // Allow Azure services
77
+ resource firewallAzure 'Microsoft.Sql/servers/firewallRules@2023-05-01-preview' = {
78
+ parent: sqlServer
79
+ name: 'AllowAllAzureIps'
80
+ properties: {
81
+ startIpAddress: '0.0.0.0'
82
+ endIpAddress: '0.0.0.0'
83
+ }
84
+ }
85
+
86
+ // ==============================================================================
87
+ // OUTPUTS
88
+ // ==============================================================================
89
+
90
+ @description('SQL Server ID')
91
+ output serverId string = sqlServer.id
92
+
93
+ @description('SQL Server FQDN')
94
+ output serverFqdn string = sqlServer.properties.fullyQualifiedDomainName
95
+
96
+ @description('Database ID')
97
+ output databaseId string = sqlDatabase.id
98
+
99
+ @description('Connection string (password placeholder)')
100
+ output connectionString string = 'Server=tcp:${sqlServer.properties.fullyQualifiedDomainName},1433;Database=${databaseName};User ID=${adminUsername};Password=${adminPassword};Encrypt=true;TrustServerCertificate=false;Connection Timeout=30;'
101
+
102
+ @description('Connection string template (no password)')
103
+ output connectionStringTemplate string = 'Server=tcp:${sqlServer.properties.fullyQualifiedDomainName},1433;Database=${databaseName};User ID=${adminUsername};Password={your_password};Encrypt=true;TrustServerCertificate=false;Connection Timeout=30;'