@geekmidas/cli 0.53.0 → 1.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.
Files changed (156) hide show
  1. package/CHANGELOG.md +17 -0
  2. package/README.md +26 -5
  3. package/dist/CachedStateProvider-D73dCqfH.cjs +60 -0
  4. package/dist/CachedStateProvider-D73dCqfH.cjs.map +1 -0
  5. package/dist/CachedStateProvider-DVyKfaMm.mjs +54 -0
  6. package/dist/CachedStateProvider-DVyKfaMm.mjs.map +1 -0
  7. package/dist/CachedStateProvider-D_uISMmJ.cjs +3 -0
  8. package/dist/CachedStateProvider-OiFUGr7p.mjs +3 -0
  9. package/dist/HostingerProvider-DUV9-Tzg.cjs +210 -0
  10. package/dist/HostingerProvider-DUV9-Tzg.cjs.map +1 -0
  11. package/dist/HostingerProvider-DqUq6e9i.mjs +210 -0
  12. package/dist/HostingerProvider-DqUq6e9i.mjs.map +1 -0
  13. package/dist/LocalStateProvider-CdspeSVL.cjs +43 -0
  14. package/dist/LocalStateProvider-CdspeSVL.cjs.map +1 -0
  15. package/dist/LocalStateProvider-DxoSaWUV.mjs +42 -0
  16. package/dist/LocalStateProvider-DxoSaWUV.mjs.map +1 -0
  17. package/dist/Route53Provider-CpRIqu69.cjs +157 -0
  18. package/dist/Route53Provider-CpRIqu69.cjs.map +1 -0
  19. package/dist/Route53Provider-KUAX3vz9.mjs +156 -0
  20. package/dist/Route53Provider-KUAX3vz9.mjs.map +1 -0
  21. package/dist/SSMStateProvider-BxAPU99a.cjs +53 -0
  22. package/dist/SSMStateProvider-BxAPU99a.cjs.map +1 -0
  23. package/dist/SSMStateProvider-C4wp4AZe.mjs +52 -0
  24. package/dist/SSMStateProvider-C4wp4AZe.mjs.map +1 -0
  25. package/dist/{bundler-DGry2vaR.mjs → bundler-BqTN5Dj5.mjs} +3 -3
  26. package/dist/{bundler-DGry2vaR.mjs.map → bundler-BqTN5Dj5.mjs.map} +1 -1
  27. package/dist/{bundler-BB-kETMd.cjs → bundler-tHLLwYuU.cjs} +3 -3
  28. package/dist/{bundler-BB-kETMd.cjs.map → bundler-tHLLwYuU.cjs.map} +1 -1
  29. package/dist/{config-HYiM3iQJ.cjs → config-BGeJsW1r.cjs} +2 -2
  30. package/dist/{config-HYiM3iQJ.cjs.map → config-BGeJsW1r.cjs.map} +1 -1
  31. package/dist/{config-C3LSBNSl.mjs → config-C6awcFBx.mjs} +2 -2
  32. package/dist/{config-C3LSBNSl.mjs.map → config-C6awcFBx.mjs.map} +1 -1
  33. package/dist/config.cjs +2 -2
  34. package/dist/config.d.cts +1 -1
  35. package/dist/config.d.mts +2 -2
  36. package/dist/config.mjs +2 -2
  37. package/dist/credentials-C8DWtnMY.cjs +174 -0
  38. package/dist/credentials-C8DWtnMY.cjs.map +1 -0
  39. package/dist/credentials-DT1dSxIx.mjs +126 -0
  40. package/dist/credentials-DT1dSxIx.mjs.map +1 -0
  41. package/dist/deploy/sniffer-envkit-patch.cjs.map +1 -1
  42. package/dist/deploy/sniffer-envkit-patch.mjs.map +1 -1
  43. package/dist/deploy/sniffer-loader.cjs +1 -1
  44. package/dist/{dokploy-api-94KzmTVf.mjs → dokploy-api-7k3t7_zd.mjs} +1 -1
  45. package/dist/{dokploy-api-94KzmTVf.mjs.map → dokploy-api-7k3t7_zd.mjs.map} +1 -1
  46. package/dist/dokploy-api-CHa8G51l.mjs +3 -0
  47. package/dist/{dokploy-api-YD8WCQfW.cjs → dokploy-api-CQvhV6Hd.cjs} +1 -1
  48. package/dist/{dokploy-api-YD8WCQfW.cjs.map → dokploy-api-CQvhV6Hd.cjs.map} +1 -1
  49. package/dist/dokploy-api-CWc02yyg.cjs +3 -0
  50. package/dist/{encryption-DaCB_NmS.cjs → encryption-BE0UOb8j.cjs} +1 -1
  51. package/dist/{encryption-DaCB_NmS.cjs.map → encryption-BE0UOb8j.cjs.map} +1 -1
  52. package/dist/{encryption-Biq0EZ4m.cjs → encryption-Cv3zips0.cjs} +1 -1
  53. package/dist/{encryption-BC4MAODn.mjs → encryption-JtMsiGNp.mjs} +1 -1
  54. package/dist/{encryption-BC4MAODn.mjs.map → encryption-JtMsiGNp.mjs.map} +1 -1
  55. package/dist/encryption-UUmaWAmz.mjs +3 -0
  56. package/dist/{index-pOA56MWT.d.cts → index-B5rGIc4g.d.cts} +553 -196
  57. package/dist/index-B5rGIc4g.d.cts.map +1 -0
  58. package/dist/{index-A70abJ1m.d.mts → index-KFEbMIRa.d.mts} +554 -197
  59. package/dist/index-KFEbMIRa.d.mts.map +1 -0
  60. package/dist/index.cjs +2242 -568
  61. package/dist/index.cjs.map +1 -1
  62. package/dist/index.mjs +2219 -545
  63. package/dist/index.mjs.map +1 -1
  64. package/dist/{openapi-C3C-BzIZ.mjs → openapi-BMFmLnX6.mjs} +51 -7
  65. package/dist/openapi-BMFmLnX6.mjs.map +1 -0
  66. package/dist/{openapi-D7WwlpPF.cjs → openapi-D1KXv2Ml.cjs} +51 -7
  67. package/dist/openapi-D1KXv2Ml.cjs.map +1 -0
  68. package/dist/{openapi-react-query-C_MxpBgF.cjs → openapi-react-query-BeXvk-wa.cjs} +1 -1
  69. package/dist/{openapi-react-query-C_MxpBgF.cjs.map → openapi-react-query-BeXvk-wa.cjs.map} +1 -1
  70. package/dist/{openapi-react-query-ZoP9DPbY.mjs → openapi-react-query-DGEkD39r.mjs} +1 -1
  71. package/dist/{openapi-react-query-ZoP9DPbY.mjs.map → openapi-react-query-DGEkD39r.mjs.map} +1 -1
  72. package/dist/openapi-react-query.cjs +1 -1
  73. package/dist/openapi-react-query.mjs +1 -1
  74. package/dist/openapi.cjs +3 -3
  75. package/dist/openapi.d.cts +1 -1
  76. package/dist/openapi.d.mts +2 -2
  77. package/dist/openapi.mjs +3 -3
  78. package/dist/{storage-Dhst7BhI.mjs → storage-BMW6yLu3.mjs} +1 -1
  79. package/dist/{storage-Dhst7BhI.mjs.map → storage-BMW6yLu3.mjs.map} +1 -1
  80. package/dist/{storage-fOR8dMu5.cjs → storage-C7pmBq1u.cjs} +1 -1
  81. package/dist/{storage-BPRgh3DU.cjs → storage-CoCNe0Pt.cjs} +1 -1
  82. package/dist/{storage-BPRgh3DU.cjs.map → storage-CoCNe0Pt.cjs.map} +1 -1
  83. package/dist/{storage-DNj_I11J.mjs → storage-D8XzjVaO.mjs} +1 -1
  84. package/dist/{types-BtGL-8QS.d.mts → types-BldpmqQX.d.mts} +1 -1
  85. package/dist/{types-BtGL-8QS.d.mts.map → types-BldpmqQX.d.mts.map} +1 -1
  86. package/dist/workspace/index.cjs +1 -1
  87. package/dist/workspace/index.d.cts +1 -1
  88. package/dist/workspace/index.d.mts +2 -2
  89. package/dist/workspace/index.mjs +1 -1
  90. package/dist/{workspace-CaVW6j2q.cjs → workspace-BFRUOOrh.cjs} +309 -25
  91. package/dist/workspace-BFRUOOrh.cjs.map +1 -0
  92. package/dist/{workspace-DLFRaDc-.mjs → workspace-DAxG3_H2.mjs} +309 -25
  93. package/dist/workspace-DAxG3_H2.mjs.map +1 -0
  94. package/package.json +12 -8
  95. package/src/build/__tests__/handler-templates.spec.ts +115 -47
  96. package/src/deploy/CachedStateProvider.ts +86 -0
  97. package/src/deploy/LocalStateProvider.ts +57 -0
  98. package/src/deploy/SSMStateProvider.ts +93 -0
  99. package/src/deploy/StateProvider.ts +171 -0
  100. package/src/deploy/__tests__/CachedStateProvider.spec.ts +228 -0
  101. package/src/deploy/__tests__/HostingerProvider.spec.ts +347 -0
  102. package/src/deploy/__tests__/LocalStateProvider.spec.ts +126 -0
  103. package/src/deploy/__tests__/Route53Provider.spec.ts +402 -0
  104. package/src/deploy/__tests__/SSMStateProvider.spec.ts +177 -0
  105. package/src/deploy/__tests__/__fixtures__/env-parsers/throwing-env-parser.ts +1 -3
  106. package/src/deploy/__tests__/__fixtures__/route-apps/endpoints/auth.ts +16 -0
  107. package/src/deploy/__tests__/__fixtures__/route-apps/endpoints/health.ts +13 -0
  108. package/src/deploy/__tests__/__fixtures__/route-apps/endpoints/users.ts +15 -0
  109. package/src/deploy/__tests__/__fixtures__/route-apps/services.ts +55 -0
  110. package/src/deploy/__tests__/createDnsProvider.spec.ts +172 -0
  111. package/src/deploy/__tests__/createStateProvider.spec.ts +116 -0
  112. package/src/deploy/__tests__/dns-orchestration.spec.ts +192 -0
  113. package/src/deploy/__tests__/dns-verification.spec.ts +2 -2
  114. package/src/deploy/__tests__/env-resolver.spec.ts +41 -17
  115. package/src/deploy/__tests__/sniffer.spec.ts +168 -10
  116. package/src/deploy/__tests__/state.spec.ts +13 -5
  117. package/src/deploy/dns/DnsProvider.ts +163 -0
  118. package/src/deploy/dns/HostingerProvider.ts +100 -0
  119. package/src/deploy/dns/Route53Provider.ts +256 -0
  120. package/src/deploy/dns/index.ts +257 -165
  121. package/src/deploy/env-resolver.ts +12 -5
  122. package/src/deploy/index.ts +16 -13
  123. package/src/deploy/sniffer-envkit-patch.ts +3 -1
  124. package/src/deploy/sniffer-routes-worker.ts +104 -0
  125. package/src/deploy/sniffer.ts +130 -5
  126. package/src/deploy/state-commands.ts +274 -0
  127. package/src/dev/__tests__/entry.spec.ts +8 -2
  128. package/src/dev/__tests__/index.spec.ts +1 -3
  129. package/src/dev/index.ts +9 -3
  130. package/src/docker/__tests__/templates.spec.ts +3 -1
  131. package/src/docker/templates.ts +3 -3
  132. package/src/index.ts +88 -0
  133. package/src/init/__tests__/generators.spec.ts +273 -0
  134. package/src/init/__tests__/init.spec.ts +3 -3
  135. package/src/init/generators/auth.ts +1 -0
  136. package/src/init/generators/config.ts +2 -0
  137. package/src/init/generators/models.ts +6 -1
  138. package/src/init/generators/monorepo.ts +3 -0
  139. package/src/init/generators/ui.ts +1472 -0
  140. package/src/init/generators/web.ts +134 -87
  141. package/src/init/index.ts +22 -3
  142. package/src/init/templates/api.ts +109 -3
  143. package/src/openapi.ts +99 -13
  144. package/src/workspace/__tests__/schema.spec.ts +107 -0
  145. package/src/workspace/schema.ts +314 -4
  146. package/src/workspace/types.ts +22 -36
  147. package/dist/dokploy-api-CItuaWTq.mjs +0 -3
  148. package/dist/dokploy-api-DBNE8MDt.cjs +0 -3
  149. package/dist/encryption-CQXBZGkt.mjs +0 -3
  150. package/dist/index-A70abJ1m.d.mts.map +0 -1
  151. package/dist/index-pOA56MWT.d.cts.map +0 -1
  152. package/dist/openapi-C3C-BzIZ.mjs.map +0 -1
  153. package/dist/openapi-D7WwlpPF.cjs.map +0 -1
  154. package/dist/workspace-CaVW6j2q.cjs.map +0 -1
  155. package/dist/workspace-DLFRaDc-.mjs.map +0 -1
  156. package/tsconfig.tsbuildinfo +0 -1
@@ -0,0 +1,256 @@
1
+ /**
2
+ * Route53 DNS Provider
3
+ *
4
+ * Implements DnsProvider interface using AWS Route53.
5
+ */
6
+
7
+ import {
8
+ ChangeResourceRecordSetsCommand,
9
+ ListHostedZonesByNameCommand,
10
+ ListResourceRecordSetsCommand,
11
+ Route53Client,
12
+ type RRType,
13
+ } from '@aws-sdk/client-route-53';
14
+ import { fromIni } from '@aws-sdk/credential-providers';
15
+ import type {
16
+ DnsProvider,
17
+ DnsRecord,
18
+ DnsRecordType,
19
+ UpsertDnsRecord,
20
+ UpsertResult,
21
+ } from './DnsProvider';
22
+
23
+ export interface Route53ProviderOptions {
24
+ /** AWS region (optional - uses AWS_REGION env var if not provided) */
25
+ region?: string;
26
+ /** AWS profile name (optional - uses default credential chain if not provided) */
27
+ profile?: string;
28
+ /** Hosted zone ID (optional - auto-detected from domain if not provided) */
29
+ hostedZoneId?: string;
30
+ /** Custom endpoint for testing with localstack */
31
+ endpoint?: string;
32
+ }
33
+
34
+ /**
35
+ * Route53 DNS provider implementation.
36
+ *
37
+ * Uses AWS default credential chain for authentication.
38
+ * Region can be specified or will use AWS_REGION/AWS_DEFAULT_REGION env vars.
39
+ * Profile can be specified to use a named profile from ~/.aws/credentials.
40
+ */
41
+ export class Route53Provider implements DnsProvider {
42
+ readonly name = 'route53';
43
+ private client: Route53Client;
44
+ private hostedZoneId?: string;
45
+ private hostedZoneCache: Map<string, string> = new Map();
46
+
47
+ constructor(options: Route53ProviderOptions = {}) {
48
+ this.client = new Route53Client({
49
+ ...(options.region && { region: options.region }),
50
+ ...(options.endpoint && { endpoint: options.endpoint }),
51
+ ...(options.profile && {
52
+ credentials: fromIni({ profile: options.profile }),
53
+ }),
54
+ });
55
+ this.hostedZoneId = options.hostedZoneId;
56
+ }
57
+
58
+ /**
59
+ * Get the hosted zone ID for a domain.
60
+ * Uses cache to avoid repeated API calls.
61
+ */
62
+ private async getHostedZoneId(domain: string): Promise<string> {
63
+ // Use configured zone ID if provided
64
+ if (this.hostedZoneId) {
65
+ return this.hostedZoneId;
66
+ }
67
+
68
+ // Check cache
69
+ if (this.hostedZoneCache.has(domain)) {
70
+ return this.hostedZoneCache.get(domain)!;
71
+ }
72
+
73
+ // Auto-detect from domain
74
+ const command = new ListHostedZonesByNameCommand({
75
+ DNSName: domain,
76
+ MaxItems: 1,
77
+ });
78
+
79
+ const response = await this.client.send(command);
80
+ const zones = response.HostedZones ?? [];
81
+
82
+ // Find exact match (domain with trailing dot)
83
+ const normalizedDomain = domain.endsWith('.') ? domain : `${domain}.`;
84
+ const zone = zones.find((z) => z.Name === normalizedDomain);
85
+
86
+ if (!zone?.Id) {
87
+ throw new Error(
88
+ `No hosted zone found for domain: ${domain}. Create one in Route53 or provide hostedZoneId in config.`,
89
+ );
90
+ }
91
+
92
+ // Zone ID comes as "/hostedzone/Z1234567890" - extract just the ID
93
+ const zoneId = zone.Id.replace('/hostedzone/', '');
94
+ this.hostedZoneCache.set(domain, zoneId);
95
+ return zoneId;
96
+ }
97
+
98
+ /**
99
+ * Convert Route53 record type to our DnsRecordType.
100
+ * Excludes NS and SOA which are auto-managed by Route53 for the zone.
101
+ */
102
+ private toRecordType(type: string): DnsRecordType | null {
103
+ // Exclude NS and SOA which are auto-managed zone records
104
+ const managedTypes = ['NS', 'SOA'];
105
+ if (managedTypes.includes(type)) {
106
+ return null;
107
+ }
108
+
109
+ const validTypes: DnsRecordType[] = [
110
+ 'A',
111
+ 'AAAA',
112
+ 'CNAME',
113
+ 'MX',
114
+ 'TXT',
115
+ 'SRV',
116
+ 'CAA',
117
+ ];
118
+ return validTypes.includes(type as DnsRecordType)
119
+ ? (type as DnsRecordType)
120
+ : null;
121
+ }
122
+
123
+ /**
124
+ * Extract subdomain from full record name relative to domain.
125
+ */
126
+ private extractSubdomain(recordName: string, domain: string): string {
127
+ const normalizedDomain = domain.endsWith('.') ? domain : `${domain}.`;
128
+ const normalizedName = recordName.endsWith('.')
129
+ ? recordName
130
+ : `${recordName}.`;
131
+
132
+ if (normalizedName === normalizedDomain) {
133
+ return '@';
134
+ }
135
+
136
+ // Remove the domain suffix
137
+ const subdomain = normalizedName.replace(`.${normalizedDomain}`, '');
138
+ return subdomain.replace(/\.$/, ''); // Remove trailing dot if any
139
+ }
140
+
141
+ async getRecords(domain: string): Promise<DnsRecord[]> {
142
+ const zoneId = await this.getHostedZoneId(domain);
143
+ const records: DnsRecord[] = [];
144
+
145
+ let nextRecordName: string | undefined;
146
+ let nextRecordType: RRType | undefined;
147
+
148
+ // Paginate through all records
149
+ do {
150
+ const command = new ListResourceRecordSetsCommand({
151
+ HostedZoneId: zoneId,
152
+ StartRecordName: nextRecordName,
153
+ StartRecordType: nextRecordType,
154
+ MaxItems: 100,
155
+ });
156
+
157
+ const response = await this.client.send(command);
158
+
159
+ for (const recordSet of response.ResourceRecordSets ?? []) {
160
+ const type = this.toRecordType(recordSet.Type ?? '');
161
+ if (!type || !recordSet.Name) continue;
162
+
163
+ const values = (recordSet.ResourceRecords ?? [])
164
+ .map((r) => r.Value)
165
+ .filter((v): v is string => !!v);
166
+
167
+ records.push({
168
+ name: this.extractSubdomain(recordSet.Name, domain),
169
+ type,
170
+ ttl: recordSet.TTL ?? 300,
171
+ values,
172
+ });
173
+ }
174
+
175
+ if (response.IsTruncated) {
176
+ nextRecordName = response.NextRecordName;
177
+ nextRecordType = response.NextRecordType;
178
+ } else {
179
+ nextRecordName = undefined;
180
+ }
181
+ } while (nextRecordName);
182
+
183
+ return records;
184
+ }
185
+
186
+ async upsertRecords(
187
+ domain: string,
188
+ records: UpsertDnsRecord[],
189
+ ): Promise<UpsertResult[]> {
190
+ const zoneId = await this.getHostedZoneId(domain);
191
+ const results: UpsertResult[] = [];
192
+
193
+ // Get existing records to determine if creating or updating
194
+ const existingRecords = await this.getRecords(domain);
195
+
196
+ // Process records in batches (Route53 allows max 1000 changes per request)
197
+ const batchSize = 100;
198
+ for (let i = 0; i < records.length; i += batchSize) {
199
+ const batch = records.slice(i, i + batchSize);
200
+ const changes = [];
201
+
202
+ for (const record of batch) {
203
+ const existing = existingRecords.find(
204
+ (r) => r.name === record.name && r.type === record.type,
205
+ );
206
+
207
+ const existingValue = existing?.values?.[0];
208
+
209
+ if (existing && existingValue === record.value) {
210
+ // Record exists with same value - unchanged
211
+ results.push({
212
+ record,
213
+ created: false,
214
+ unchanged: true,
215
+ });
216
+ continue;
217
+ }
218
+
219
+ // Build full record name
220
+ const recordName =
221
+ record.name === '@' ? domain : `${record.name}.${domain}`;
222
+
223
+ changes.push({
224
+ Action: 'UPSERT' as const,
225
+ ResourceRecordSet: {
226
+ Name: recordName,
227
+ Type: record.type,
228
+ TTL: record.ttl,
229
+ ResourceRecords: [{ Value: record.value }],
230
+ },
231
+ });
232
+
233
+ results.push({
234
+ record,
235
+ created: !existing,
236
+ unchanged: false,
237
+ });
238
+ }
239
+
240
+ // Execute batch if there are changes
241
+ if (changes.length > 0) {
242
+ const command = new ChangeResourceRecordSetsCommand({
243
+ HostedZoneId: zoneId,
244
+ ChangeBatch: {
245
+ Comment: 'Upsert by gkm deploy',
246
+ Changes: changes,
247
+ },
248
+ });
249
+
250
+ await this.client.send(command);
251
+ }
252
+ }
253
+
254
+ return results;
255
+ }
256
+ }