-
Notifications
You must be signed in to change notification settings - Fork 1
Expand file tree
/
Copy pathMakefile
More file actions
176 lines (153 loc) · 7.23 KB
/
Makefile
File metadata and controls
176 lines (153 loc) · 7.23 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
.PHONY: docker
# Change detection configuration
BASE_REF ?= HEAD^1
TEST_DATABASE_URL ?= postgresql://connect_admin:passpass@localhost:5432/connect_test
TEST_ADMIN_DATABASE_URL ?= postgresql://admin:passpass@localhost:5432/postgres
TEST_MCP_ENCRYPTION_KEY ?= 0000000000000000000000000000000000000000000000000000000000000000
# Local image tags (no GCP references)
localApiImage = caseai-connect/api:local
localWorkersImage = caseai-connect/workers:local
smokeComposeFile = infra/docker-compose.api-workers-smoke.yaml
# ==============================================================================
# Change Detection
# ==============================================================================
# Paths that affect API deployment
API_PATHS := \
apps/api/src \
apps/api/package.json \
apps/api/nest-cli.json \
apps/api/tsconfig.json \
apps/api/Dockerfile \
packages/api-contracts \
package.json \
package-lock.json \
turbo.json
# Patterns that should NOT trigger deployment (extended regex format)
API_EXCLUDE_PATTERNS := \.md$$|\.spec\.ts$$|\.e2e-spec\.ts$$|apps/api/test/|apps/api/README\.md
# Check if API has meaningful changes
check-api-changes:
@echo "Checking for API changes between $(BASE_REF) and HEAD..."
@CHANGED=$$(git diff --name-only $(BASE_REF) HEAD -- $(API_PATHS) | \
grep -v -E '$(API_EXCLUDE_PATTERNS)' || true); \
if [ -n "$$CHANGED" ]; then \
echo "✓ API changes detected:"; \
echo "$$CHANGED" | sed 's/^/ - /'; \
exit 0; \
else \
echo "✗ No API changes detected (skipping deployment)"; \
exit 1; \
fi
# Check if frontend has meaningful changes (for future use)
check-web-changes:
@echo "Checking for web changes between $(BASE_REF) and HEAD..."
@CHANGED=$$(git diff --name-only $(BASE_REF) HEAD -- apps/web packages/ui | \
grep -v -E '\.md$$|\.spec\.ts$$|\.test\.ts$$' || true); \
if [ -n "$$CHANGED" ]; then \
echo "✓ Web changes detected:"; \
echo "$$CHANGED" | sed 's/^/ - /'; \
exit 0; \
else \
echo "✗ No web changes detected"; \
exit 1; \
fi
# ==============================================================================
# Docker & Deployment
# ==============================================================================
trivy-scan: docker-build
trivy image --ignore-unfixed --pkg-types os,library --severity CRITICAL,HIGH --ignorefile .trivyignore.yaml ${localApiImage}
trivy image --ignore-unfixed --pkg-types os,library --severity CRITICAL,HIGH --ignorefile .trivyignore.yaml ${localWorkersImage}
docker-build: docker-build-api docker-build-workers
docker-build-api:
docker build --platform=linux/amd64 --target api-runtime -t ${localApiImage} -f apps/api/Dockerfile .
docker-build-workers:
docker build --platform=linux/amd64 --target workers-runtime -t ${localWorkersImage} -f apps/api/Dockerfile .
docker-check: docker-build-api
@echo "Starting docker container and checking for successful startup..."
@CONTAINER_ID=$$(docker run -d -p "3003:3000" ${localApiImage}); \
echo "Container ID: $$CONTAINER_ID"; \
i=1; \
while [ $$i -le 30 ]; do \
echo "Checking logs (attempt $$i/30)..."; \
LOGS=$$(docker logs $$CONTAINER_ID 2>&1); \
echo "$$LOGS"; \
if echo "$$LOGS" | grep -q "Starting Nest application..."; then \
echo "✓ Docker container started successfully"; \
docker kill $$CONTAINER_ID >/dev/null 2>&1; \
exit 0; \
fi; \
sleep 1; \
i=$$((i + 1)); \
done; \
echo "✗ Failed to find 'Starting Nest application...' in docker logs"; \
docker kill $$CONTAINER_ID >/dev/null 2>&1; \
exit 1
docker-workers-check: docker-build-workers
@echo "Starting docker workers with smoke dependencies and checking for successful startup..."
@API_IMAGE=${localApiImage} WORKERS_IMAGE=${localWorkersImage} docker compose -f ${smokeComposeFile} up -d postgres redis workers; \
CONTAINER_ID=$$(API_IMAGE=${localApiImage} WORKERS_IMAGE=${localWorkersImage} docker compose -f ${smokeComposeFile} ps -q workers); \
echo "Container ID: $$CONTAINER_ID"; \
echo "Verifying Docling CLI in workers container..."; \
j=1; \
DOC_VERSION=""; \
while [ $$j -le 15 ]; do \
DOC_VERSION=$$(docker exec $$CONTAINER_ID docling --version 2>/dev/null || true); \
if [ -n "$$DOC_VERSION" ]; then \
echo "✓ Docling CLI available: $$DOC_VERSION"; \
break; \
fi; \
if ! docker ps -q --no-trunc | grep -q "$$CONTAINER_ID"; then \
echo "✗ Workers container exited before Docling CLI check completed"; \
docker logs $$CONTAINER_ID 2>&1; \
API_IMAGE=${localApiImage} WORKERS_IMAGE=${localWorkersImage} docker compose -f ${smokeComposeFile} down -v >/dev/null 2>&1; \
exit 1; \
fi; \
sleep 1; \
j=$$((j + 1)); \
done; \
if [ -z "$$DOC_VERSION" ]; then \
echo "✗ Docling CLI check failed in workers container"; \
docker logs $$CONTAINER_ID 2>&1; \
API_IMAGE=${localApiImage} WORKERS_IMAGE=${localWorkersImage} docker compose -f ${smokeComposeFile} down -v >/dev/null 2>&1; \
exit 1; \
fi; \
i=1; \
while [ $$i -le 45 ]; do \
echo "Checking logs (attempt $$i/45)..."; \
LOGS=$$(docker logs $$CONTAINER_ID 2>&1); \
echo "$$LOGS"; \
if echo "$$LOGS" | grep -q "Workers app started"; then \
echo "✓ Docker workers container started successfully"; \
API_IMAGE=${localApiImage} WORKERS_IMAGE=${localWorkersImage} docker compose -f ${smokeComposeFile} down -v >/dev/null 2>&1; \
exit 0; \
fi; \
sleep 1; \
i=$$((i + 1)); \
done; \
echo "✗ Failed to find 'Workers app started' in docker logs"; \
API_IMAGE=${localApiImage} WORKERS_IMAGE=${localWorkersImage} docker compose -f ${smokeComposeFile} down -v >/dev/null 2>&1; \
exit 1
docker-smoke-up: docker-build
API_IMAGE=${localApiImage} WORKERS_IMAGE=${localWorkersImage} docker compose -f ${smokeComposeFile} up -d --build
docker-smoke-ps:
API_IMAGE=${localApiImage} WORKERS_IMAGE=${localWorkersImage} docker compose -f ${smokeComposeFile} ps
docker-smoke-logs:
API_IMAGE=${localApiImage} WORKERS_IMAGE=${localWorkersImage} docker compose -f ${smokeComposeFile} logs --tail=200 api workers postgres redis
docker-smoke-check: docker-smoke-up
@echo "Waiting for services to settle..."
@sleep 8
@API_IMAGE=${localApiImage} WORKERS_IMAGE=${localWorkersImage} docker compose -f ${smokeComposeFile} ps
@if API_IMAGE=${localApiImage} WORKERS_IMAGE=${localWorkersImage} docker compose -f ${smokeComposeFile} ps --status exited | grep -Eq "api|workers"; then \
echo "✗ API or workers exited unexpectedly. Check logs with: make docker-smoke-logs"; \
exit 1; \
fi
@echo "✓ API and workers are still running"
docker-smoke-down:
API_IMAGE=${localApiImage} WORKERS_IMAGE=${localWorkersImage} docker compose -f ${smokeComposeFile} down -v
ci-checks:
npm ci && npm run biome:ci && npm run typecheck
db-tests:
docker compose -f infra/database/docker-compose.yaml up -d
tests: db-tests ci-checks
cd apps/api && DATABASE_URL=${TEST_DATABASE_URL} MCP_ENCRYPTION_KEY=${TEST_MCP_ENCRYPTION_KEY} npm run migration:test:run && DATABASE_URL=${TEST_DATABASE_URL} MCP_ENCRYPTION_KEY=${TEST_MCP_ENCRYPTION_KEY} npm run test
tests-parallel: db-tests ci-checks
cd apps/api && DATABASE_URL=${TEST_DATABASE_URL} MCP_ENCRYPTION_KEY=${TEST_MCP_ENCRYPTION_KEY} npm run migration:test:run && TEST_ADMIN_DATABASE_URL=${TEST_ADMIN_DATABASE_URL} TEST_MAX_WORKERS=${TEST_MAX_WORKERS} DATABASE_URL=${TEST_DATABASE_URL} MCP_ENCRYPTION_KEY=${TEST_MCP_ENCRYPTION_KEY} npm run test:parallel