@geekmidas/cli 0.10.0 → 0.13.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 (146) hide show
  1. package/README.md +525 -0
  2. package/dist/bundler-B1qy9b-j.cjs +112 -0
  3. package/dist/bundler-B1qy9b-j.cjs.map +1 -0
  4. package/dist/bundler-DskIqW2t.mjs +111 -0
  5. package/dist/bundler-DskIqW2t.mjs.map +1 -0
  6. package/dist/{config-C9aXOHBe.cjs → config-AmInkU7k.cjs} +8 -8
  7. package/dist/config-AmInkU7k.cjs.map +1 -0
  8. package/dist/{config-BrkUalUh.mjs → config-DYULeEv8.mjs} +3 -3
  9. package/dist/config-DYULeEv8.mjs.map +1 -0
  10. package/dist/config.cjs +1 -1
  11. package/dist/config.d.cts +1 -1
  12. package/dist/config.d.mts +1 -1
  13. package/dist/config.mjs +1 -1
  14. package/dist/encryption-C8H-38Yy.mjs +42 -0
  15. package/dist/encryption-C8H-38Yy.mjs.map +1 -0
  16. package/dist/encryption-Dyf_r1h-.cjs +44 -0
  17. package/dist/encryption-Dyf_r1h-.cjs.map +1 -0
  18. package/dist/index.cjs +2123 -179
  19. package/dist/index.cjs.map +1 -1
  20. package/dist/index.mjs +2141 -192
  21. package/dist/index.mjs.map +1 -1
  22. package/dist/{openapi-CZLI4QTr.mjs → openapi-BfFlOBCG.mjs} +801 -38
  23. package/dist/openapi-BfFlOBCG.mjs.map +1 -0
  24. package/dist/{openapi-BeHLKcwP.cjs → openapi-Bt_1FDpT.cjs} +794 -31
  25. package/dist/openapi-Bt_1FDpT.cjs.map +1 -0
  26. package/dist/{openapi-react-query-o5iMi8tz.cjs → openapi-react-query-B-sNWHFU.cjs} +5 -5
  27. package/dist/openapi-react-query-B-sNWHFU.cjs.map +1 -0
  28. package/dist/{openapi-react-query-CcciaVu5.mjs → openapi-react-query-B6XTeGqS.mjs} +5 -5
  29. package/dist/openapi-react-query-B6XTeGqS.mjs.map +1 -0
  30. package/dist/openapi-react-query.cjs +1 -1
  31. package/dist/openapi-react-query.d.cts.map +1 -1
  32. package/dist/openapi-react-query.d.mts.map +1 -1
  33. package/dist/openapi-react-query.mjs +1 -1
  34. package/dist/openapi.cjs +2 -2
  35. package/dist/openapi.d.cts +1 -1
  36. package/dist/openapi.d.cts.map +1 -1
  37. package/dist/openapi.d.mts +1 -1
  38. package/dist/openapi.d.mts.map +1 -1
  39. package/dist/openapi.mjs +2 -2
  40. package/dist/storage-BOOpAF8N.cjs +5 -0
  41. package/dist/storage-Bj1E26lU.cjs +187 -0
  42. package/dist/storage-Bj1E26lU.cjs.map +1 -0
  43. package/dist/storage-kSxTjkNb.mjs +133 -0
  44. package/dist/storage-kSxTjkNb.mjs.map +1 -0
  45. package/dist/storage-tgZSUnKl.mjs +3 -0
  46. package/dist/{types-b-vwGpqc.d.cts → types-BR0M2v_c.d.mts} +100 -1
  47. package/dist/types-BR0M2v_c.d.mts.map +1 -0
  48. package/dist/{types-DXgiA1sF.d.mts → types-BhkZc-vm.d.cts} +100 -1
  49. package/dist/types-BhkZc-vm.d.cts.map +1 -0
  50. package/examples/cron-example.ts +27 -27
  51. package/examples/env.ts +27 -27
  52. package/examples/function-example.ts +31 -31
  53. package/examples/gkm.config.json +20 -20
  54. package/examples/gkm.config.ts +8 -8
  55. package/examples/gkm.minimal.config.json +5 -5
  56. package/examples/gkm.production.config.json +25 -25
  57. package/examples/logger.ts +2 -2
  58. package/package.json +6 -6
  59. package/src/__tests__/EndpointGenerator.hooks.spec.ts +191 -191
  60. package/src/__tests__/config.spec.ts +55 -55
  61. package/src/__tests__/loadEnvFiles.spec.ts +93 -93
  62. package/src/__tests__/normalizeHooksConfig.spec.ts +58 -58
  63. package/src/__tests__/openapi-react-query.spec.ts +497 -497
  64. package/src/__tests__/openapi.spec.ts +428 -428
  65. package/src/__tests__/test-helpers.ts +76 -76
  66. package/src/auth/__tests__/credentials.spec.ts +204 -0
  67. package/src/auth/__tests__/index.spec.ts +168 -0
  68. package/src/auth/credentials.ts +187 -0
  69. package/src/auth/index.ts +226 -0
  70. package/src/build/__tests__/bundler.spec.ts +444 -0
  71. package/src/build/__tests__/index-new.spec.ts +474 -474
  72. package/src/build/__tests__/manifests.spec.ts +333 -333
  73. package/src/build/bundler.ts +210 -0
  74. package/src/build/endpoint-analyzer.ts +236 -0
  75. package/src/build/handler-templates.ts +1253 -0
  76. package/src/build/index.ts +260 -179
  77. package/src/build/manifests.ts +52 -52
  78. package/src/build/providerResolver.ts +145 -145
  79. package/src/build/types.ts +64 -43
  80. package/src/config.ts +39 -39
  81. package/src/deploy/__tests__/docker.spec.ts +111 -0
  82. package/src/deploy/__tests__/dokploy.spec.ts +245 -0
  83. package/src/deploy/__tests__/init.spec.ts +662 -0
  84. package/src/deploy/docker.ts +128 -0
  85. package/src/deploy/dokploy.ts +204 -0
  86. package/src/deploy/index.ts +136 -0
  87. package/src/deploy/init.ts +484 -0
  88. package/src/deploy/types.ts +48 -0
  89. package/src/dev/__tests__/index.spec.ts +266 -266
  90. package/src/dev/index.ts +647 -601
  91. package/src/docker/__tests__/compose.spec.ts +531 -0
  92. package/src/docker/__tests__/templates.spec.ts +280 -0
  93. package/src/docker/compose.ts +273 -0
  94. package/src/docker/index.ts +230 -0
  95. package/src/docker/templates.ts +446 -0
  96. package/src/generators/CronGenerator.ts +72 -72
  97. package/src/generators/EndpointGenerator.ts +699 -398
  98. package/src/generators/FunctionGenerator.ts +84 -84
  99. package/src/generators/Generator.ts +72 -72
  100. package/src/generators/OpenApiTsGenerator.ts +577 -577
  101. package/src/generators/SubscriberGenerator.ts +124 -124
  102. package/src/generators/__tests__/CronGenerator.spec.ts +433 -433
  103. package/src/generators/__tests__/EndpointGenerator.spec.ts +532 -382
  104. package/src/generators/__tests__/FunctionGenerator.spec.ts +244 -244
  105. package/src/generators/__tests__/SubscriberGenerator.spec.ts +397 -382
  106. package/src/generators/index.ts +4 -4
  107. package/src/index.ts +623 -201
  108. package/src/init/__tests__/generators.spec.ts +334 -334
  109. package/src/init/__tests__/init.spec.ts +332 -332
  110. package/src/init/__tests__/utils.spec.ts +89 -89
  111. package/src/init/generators/config.ts +175 -175
  112. package/src/init/generators/docker.ts +41 -41
  113. package/src/init/generators/env.ts +72 -72
  114. package/src/init/generators/index.ts +1 -1
  115. package/src/init/generators/models.ts +64 -64
  116. package/src/init/generators/monorepo.ts +161 -161
  117. package/src/init/generators/package.ts +71 -71
  118. package/src/init/generators/source.ts +6 -6
  119. package/src/init/index.ts +203 -208
  120. package/src/init/templates/api.ts +115 -115
  121. package/src/init/templates/index.ts +75 -75
  122. package/src/init/templates/minimal.ts +98 -98
  123. package/src/init/templates/serverless.ts +89 -89
  124. package/src/init/templates/worker.ts +98 -98
  125. package/src/init/utils.ts +54 -56
  126. package/src/openapi-react-query.ts +194 -194
  127. package/src/openapi.ts +63 -63
  128. package/src/secrets/__tests__/encryption.spec.ts +226 -0
  129. package/src/secrets/__tests__/generator.spec.ts +319 -0
  130. package/src/secrets/__tests__/index.spec.ts +91 -0
  131. package/src/secrets/__tests__/storage.spec.ts +611 -0
  132. package/src/secrets/encryption.ts +91 -0
  133. package/src/secrets/generator.ts +164 -0
  134. package/src/secrets/index.ts +383 -0
  135. package/src/secrets/storage.ts +192 -0
  136. package/src/secrets/types.ts +53 -0
  137. package/src/types.ts +295 -176
  138. package/tsdown.config.ts +11 -8
  139. package/dist/config-BrkUalUh.mjs.map +0 -1
  140. package/dist/config-C9aXOHBe.cjs.map +0 -1
  141. package/dist/openapi-BeHLKcwP.cjs.map +0 -1
  142. package/dist/openapi-CZLI4QTr.mjs.map +0 -1
  143. package/dist/openapi-react-query-CcciaVu5.mjs.map +0 -1
  144. package/dist/openapi-react-query-o5iMi8tz.cjs.map +0 -1
  145. package/dist/types-DXgiA1sF.d.mts.map +0 -1
  146. package/dist/types-b-vwGpqc.d.cts.map +0 -1
@@ -3,102 +3,102 @@ import { tmpdir } from 'node:os';
3
3
  import { join } from 'node:path';
4
4
  import { afterEach, beforeEach, describe, expect, it } from 'vitest';
5
5
  import {
6
- checkDirectoryExists,
7
- detectPackageManager,
8
- validateProjectName,
6
+ checkDirectoryExists,
7
+ detectPackageManager,
8
+ validateProjectName,
9
9
  } from '../utils.js';
10
10
 
11
11
  describe('validateProjectName', () => {
12
- it('should accept valid project names', () => {
13
- expect(validateProjectName('my-project')).toBe(true);
14
- expect(validateProjectName('my_project')).toBe(true);
15
- expect(validateProjectName('myProject')).toBe(true);
16
- expect(validateProjectName('my-project-123')).toBe(true);
17
- expect(validateProjectName('project')).toBe(true);
18
- });
19
-
20
- it('should reject empty names', () => {
21
- expect(validateProjectName('')).toBe('Project name is required');
22
- });
23
-
24
- it('should reject names with invalid characters', () => {
25
- const result = validateProjectName('my project');
26
- expect(result).toContain('can only contain');
27
- });
28
-
29
- it('should accept scoped package names', () => {
30
- // @ / . are valid for scoped npm packages
31
- expect(validateProjectName('@my/project')).toBe(true);
32
- expect(validateProjectName('my.project')).toBe(true);
33
- });
34
-
35
- it('should reject names with other special characters', () => {
36
- expect(validateProjectName('my$project')).toContain('can only contain');
37
- expect(validateProjectName('my#project')).toContain('can only contain');
38
- expect(validateProjectName('my!project')).toContain('can only contain');
39
- });
12
+ it('should accept valid project names', () => {
13
+ expect(validateProjectName('my-project')).toBe(true);
14
+ expect(validateProjectName('my_project')).toBe(true);
15
+ expect(validateProjectName('myProject')).toBe(true);
16
+ expect(validateProjectName('my-project-123')).toBe(true);
17
+ expect(validateProjectName('project')).toBe(true);
18
+ });
19
+
20
+ it('should reject empty names', () => {
21
+ expect(validateProjectName('')).toBe('Project name is required');
22
+ });
23
+
24
+ it('should reject names with invalid characters', () => {
25
+ const result = validateProjectName('my project');
26
+ expect(result).toContain('can only contain');
27
+ });
28
+
29
+ it('should accept scoped package names', () => {
30
+ // @ / . are valid for scoped npm packages
31
+ expect(validateProjectName('@my/project')).toBe(true);
32
+ expect(validateProjectName('my.project')).toBe(true);
33
+ });
34
+
35
+ it('should reject names with other special characters', () => {
36
+ expect(validateProjectName('my$project')).toContain('can only contain');
37
+ expect(validateProjectName('my#project')).toContain('can only contain');
38
+ expect(validateProjectName('my!project')).toContain('can only contain');
39
+ });
40
40
  });
41
41
 
42
42
  describe('checkDirectoryExists', () => {
43
- let tempDir: string;
44
-
45
- beforeEach(async () => {
46
- tempDir = join(tmpdir(), `cli-test-${Date.now()}`);
47
- await mkdir(tempDir, { recursive: true });
48
- });
49
-
50
- afterEach(async () => {
51
- await rm(tempDir, { recursive: true, force: true });
52
- });
53
-
54
- it('should return true for non-existent directory', () => {
55
- expect(checkDirectoryExists('non-existent-dir', tempDir)).toBe(true);
56
- });
57
-
58
- it('should return error for existing directory', async () => {
59
- const existingDir = 'existing-dir';
60
- await mkdir(join(tempDir, existingDir));
61
- const result = checkDirectoryExists(existingDir, tempDir);
62
- expect(result).toContain('already exists');
63
- });
43
+ let tempDir: string;
44
+
45
+ beforeEach(async () => {
46
+ tempDir = join(tmpdir(), `cli-test-${Date.now()}`);
47
+ await mkdir(tempDir, { recursive: true });
48
+ });
49
+
50
+ afterEach(async () => {
51
+ await rm(tempDir, { recursive: true, force: true });
52
+ });
53
+
54
+ it('should return true for non-existent directory', () => {
55
+ expect(checkDirectoryExists('non-existent-dir', tempDir)).toBe(true);
56
+ });
57
+
58
+ it('should return error for existing directory', async () => {
59
+ const existingDir = 'existing-dir';
60
+ await mkdir(join(tempDir, existingDir));
61
+ const result = checkDirectoryExists(existingDir, tempDir);
62
+ expect(result).toContain('already exists');
63
+ });
64
64
  });
65
65
 
66
66
  describe('detectPackageManager', () => {
67
- let tempDir: string;
68
- let originalEnv: string | undefined;
69
-
70
- beforeEach(async () => {
71
- tempDir = join(tmpdir(), `cli-test-${Date.now()}`);
72
- await mkdir(tempDir, { recursive: true });
73
- originalEnv = process.env.npm_config_user_agent;
74
- });
75
-
76
- afterEach(async () => {
77
- await rm(tempDir, { recursive: true, force: true });
78
- if (originalEnv !== undefined) {
79
- process.env.npm_config_user_agent = originalEnv;
80
- } else {
81
- delete process.env.npm_config_user_agent;
82
- }
83
- });
84
-
85
- it('should detect pnpm from user agent', () => {
86
- process.env.npm_config_user_agent = 'pnpm/8.0.0';
87
- expect(detectPackageManager(tempDir)).toBe('pnpm');
88
- });
89
-
90
- it('should detect yarn from user agent', () => {
91
- process.env.npm_config_user_agent = 'yarn/4.0.0';
92
- expect(detectPackageManager(tempDir)).toBe('yarn');
93
- });
94
-
95
- it('should detect bun from user agent', () => {
96
- process.env.npm_config_user_agent = 'bun/1.0.0';
97
- expect(detectPackageManager(tempDir)).toBe('bun');
98
- });
99
-
100
- it('should default to npm when no lockfile or user agent', () => {
101
- delete process.env.npm_config_user_agent;
102
- expect(detectPackageManager(tempDir)).toBe('npm');
103
- });
67
+ let tempDir: string;
68
+ let originalEnv: string | undefined;
69
+
70
+ beforeEach(async () => {
71
+ tempDir = join(tmpdir(), `cli-test-${Date.now()}`);
72
+ await mkdir(tempDir, { recursive: true });
73
+ originalEnv = process.env.npm_config_user_agent;
74
+ });
75
+
76
+ afterEach(async () => {
77
+ await rm(tempDir, { recursive: true, force: true });
78
+ if (originalEnv !== undefined) {
79
+ process.env.npm_config_user_agent = originalEnv;
80
+ } else {
81
+ delete process.env.npm_config_user_agent;
82
+ }
83
+ });
84
+
85
+ it('should detect pnpm from user agent', () => {
86
+ process.env.npm_config_user_agent = 'pnpm/8.0.0';
87
+ expect(detectPackageManager(tempDir)).toBe('pnpm');
88
+ });
89
+
90
+ it('should detect yarn from user agent', () => {
91
+ process.env.npm_config_user_agent = 'yarn/4.0.0';
92
+ expect(detectPackageManager(tempDir)).toBe('yarn');
93
+ });
94
+
95
+ it('should detect bun from user agent', () => {
96
+ process.env.npm_config_user_agent = 'bun/1.0.0';
97
+ expect(detectPackageManager(tempDir)).toBe('bun');
98
+ });
99
+
100
+ it('should default to npm when no lockfile or user agent', () => {
101
+ delete process.env.npm_config_user_agent;
102
+ expect(detectPackageManager(tempDir)).toBe('npm');
103
+ });
104
104
  });
@@ -1,215 +1,215 @@
1
1
  import type {
2
- GeneratedFile,
3
- TemplateConfig,
4
- TemplateOptions,
2
+ GeneratedFile,
3
+ TemplateConfig,
4
+ TemplateOptions,
5
5
  } from '../templates/index.js';
6
6
 
7
7
  /**
8
8
  * Generate configuration files (gkm.config.ts, tsconfig.json, biome.json, turbo.json)
9
9
  */
10
10
  export function generateConfigFiles(
11
- options: TemplateOptions,
12
- template: TemplateConfig,
11
+ options: TemplateOptions,
12
+ template: TemplateConfig,
13
13
  ): GeneratedFile[] {
14
- const { telescope, studio, routesStructure } = options;
15
- const isServerless = template.name === 'serverless';
16
- const hasWorker = template.name === 'worker';
14
+ const { telescope, studio, routesStructure } = options;
15
+ const isServerless = template.name === 'serverless';
16
+ const hasWorker = template.name === 'worker';
17
17
 
18
- // Get routes glob pattern based on structure
19
- const getRoutesGlob = () => {
20
- switch (routesStructure) {
21
- case 'centralized-endpoints':
22
- return './src/endpoints/**/*.ts';
23
- case 'centralized-routes':
24
- return './src/routes/**/*.ts';
25
- case 'domain-based':
26
- return './src/**/routes/*.ts';
27
- }
28
- };
18
+ // Get routes glob pattern based on structure
19
+ const getRoutesGlob = () => {
20
+ switch (routesStructure) {
21
+ case 'centralized-endpoints':
22
+ return './src/endpoints/**/*.ts';
23
+ case 'centralized-routes':
24
+ return './src/routes/**/*.ts';
25
+ case 'domain-based':
26
+ return './src/**/routes/*.ts';
27
+ }
28
+ };
29
29
 
30
- // Build gkm.config.ts
31
- let gkmConfig = `import { defineConfig } from '@geekmidas/cli/config';
30
+ // Build gkm.config.ts
31
+ let gkmConfig = `import { defineConfig } from '@geekmidas/cli/config';
32
32
 
33
33
  export default defineConfig({
34
34
  routes: '${getRoutesGlob()}',
35
35
  envParser: './src/config/env#envParser',
36
36
  logger: './src/config/logger#logger',`;
37
37
 
38
- if (isServerless || hasWorker) {
39
- gkmConfig += `
38
+ if (isServerless || hasWorker) {
39
+ gkmConfig += `
40
40
  functions: './src/functions/**/*.ts',`;
41
- }
41
+ }
42
42
 
43
- if (hasWorker) {
44
- gkmConfig += `
43
+ if (hasWorker) {
44
+ gkmConfig += `
45
45
  crons: './src/crons/**/*.ts',
46
46
  subscribers: './src/subscribers/**/*.ts',`;
47
- }
47
+ }
48
48
 
49
- if (telescope) {
50
- gkmConfig += `
49
+ if (telescope) {
50
+ gkmConfig += `
51
51
  telescope: {
52
52
  enabled: true,
53
53
  path: '/__telescope',
54
54
  },`;
55
- }
55
+ }
56
56
 
57
- if (studio) {
58
- gkmConfig += `
57
+ if (studio) {
58
+ gkmConfig += `
59
59
  studio: './src/config/studio#studio',`;
60
- }
60
+ }
61
61
 
62
- // Always add openapi config (output path is fixed to .gkm/openapi.ts)
63
- gkmConfig += `
62
+ // Always add openapi config (output path is fixed to .gkm/openapi.ts)
63
+ gkmConfig += `
64
64
  openapi: {
65
65
  enabled: true,
66
66
  },`;
67
67
 
68
- gkmConfig += `
68
+ gkmConfig += `
69
69
  });
70
70
  `;
71
71
 
72
- // Build tsconfig.json - extends root for monorepo, standalone for non-monorepo
73
- const tsConfig = options.monorepo
74
- ? {
75
- extends: '../../tsconfig.json',
76
- compilerOptions: {
77
- outDir: './dist',
78
- rootDir: './src',
79
- baseUrl: '.',
80
- paths: {
81
- [`@${options.name}/*`]: ['../../packages/*/src'],
82
- },
83
- },
84
- include: ['src/**/*.ts'],
85
- exclude: ['node_modules', 'dist'],
86
- }
87
- : {
88
- compilerOptions: {
89
- target: 'ES2022',
90
- module: 'NodeNext',
91
- moduleResolution: 'NodeNext',
92
- lib: ['ES2022'],
93
- strict: true,
94
- esModuleInterop: true,
95
- skipLibCheck: true,
96
- forceConsistentCasingInFileNames: true,
97
- resolveJsonModule: true,
98
- declaration: true,
99
- declarationMap: true,
100
- outDir: './dist',
101
- rootDir: './src',
102
- },
103
- include: ['src/**/*.ts'],
104
- exclude: ['node_modules', 'dist'],
105
- };
72
+ // Build tsconfig.json - extends root for monorepo, standalone for non-monorepo
73
+ const tsConfig = options.monorepo
74
+ ? {
75
+ extends: '../../tsconfig.json',
76
+ compilerOptions: {
77
+ outDir: './dist',
78
+ rootDir: './src',
79
+ baseUrl: '.',
80
+ paths: {
81
+ [`@${options.name}/*`]: ['../../packages/*/src'],
82
+ },
83
+ },
84
+ include: ['src/**/*.ts'],
85
+ exclude: ['node_modules', 'dist'],
86
+ }
87
+ : {
88
+ compilerOptions: {
89
+ target: 'ES2022',
90
+ module: 'NodeNext',
91
+ moduleResolution: 'NodeNext',
92
+ lib: ['ES2022'],
93
+ strict: true,
94
+ esModuleInterop: true,
95
+ skipLibCheck: true,
96
+ forceConsistentCasingInFileNames: true,
97
+ resolveJsonModule: true,
98
+ declaration: true,
99
+ declarationMap: true,
100
+ outDir: './dist',
101
+ rootDir: './src',
102
+ },
103
+ include: ['src/**/*.ts'],
104
+ exclude: ['node_modules', 'dist'],
105
+ };
106
106
 
107
- // Skip biome.json and turbo.json for monorepo (they're at root)
108
- if (options.monorepo) {
109
- return [
110
- {
111
- path: 'gkm.config.ts',
112
- content: gkmConfig,
113
- },
114
- {
115
- path: 'tsconfig.json',
116
- content: JSON.stringify(tsConfig, null, 2) + '\n',
117
- },
118
- ];
119
- }
107
+ // Skip biome.json and turbo.json for monorepo (they're at root)
108
+ if (options.monorepo) {
109
+ return [
110
+ {
111
+ path: 'gkm.config.ts',
112
+ content: gkmConfig,
113
+ },
114
+ {
115
+ path: 'tsconfig.json',
116
+ content: `${JSON.stringify(tsConfig, null, 2)}\n`,
117
+ },
118
+ ];
119
+ }
120
120
 
121
- // Build biome.json
122
- const biomeConfig = {
123
- $schema: 'https://biomejs.dev/schemas/1.9.4/schema.json',
124
- vcs: {
125
- enabled: true,
126
- clientKind: 'git',
127
- useIgnoreFile: true,
128
- },
129
- organizeImports: {
130
- enabled: true,
131
- },
132
- formatter: {
133
- enabled: true,
134
- indentStyle: 'space',
135
- indentWidth: 2,
136
- lineWidth: 80,
137
- },
138
- javascript: {
139
- formatter: {
140
- quoteStyle: 'single',
141
- trailingCommas: 'all',
142
- semicolons: 'always',
143
- arrowParentheses: 'always',
144
- },
145
- },
146
- linter: {
147
- enabled: true,
148
- rules: {
149
- recommended: true,
150
- correctness: {
151
- noUnusedImports: 'error',
152
- noUnusedVariables: 'error',
153
- },
154
- style: {
155
- noNonNullAssertion: 'off',
156
- },
157
- },
158
- },
159
- files: {
160
- ignore: ['node_modules', 'dist', '.gkm', 'coverage'],
161
- },
162
- };
121
+ // Build biome.json
122
+ const biomeConfig = {
123
+ $schema: 'https://biomejs.dev/schemas/1.9.4/schema.json',
124
+ vcs: {
125
+ enabled: true,
126
+ clientKind: 'git',
127
+ useIgnoreFile: true,
128
+ },
129
+ organizeImports: {
130
+ enabled: true,
131
+ },
132
+ formatter: {
133
+ enabled: true,
134
+ indentStyle: 'space',
135
+ indentWidth: 2,
136
+ lineWidth: 80,
137
+ },
138
+ javascript: {
139
+ formatter: {
140
+ quoteStyle: 'single',
141
+ trailingCommas: 'all',
142
+ semicolons: 'always',
143
+ arrowParentheses: 'always',
144
+ },
145
+ },
146
+ linter: {
147
+ enabled: true,
148
+ rules: {
149
+ recommended: true,
150
+ correctness: {
151
+ noUnusedImports: 'error',
152
+ noUnusedVariables: 'error',
153
+ },
154
+ style: {
155
+ noNonNullAssertion: 'off',
156
+ },
157
+ },
158
+ },
159
+ files: {
160
+ ignore: ['node_modules', 'dist', '.gkm', 'coverage'],
161
+ },
162
+ };
163
163
 
164
- // Build turbo.json
165
- const turboConfig = {
166
- $schema: 'https://turbo.build/schema.json',
167
- tasks: {
168
- build: {
169
- dependsOn: ['^build'],
170
- outputs: ['dist/**'],
171
- },
172
- dev: {
173
- cache: false,
174
- persistent: true,
175
- },
176
- test: {
177
- dependsOn: ['^build'],
178
- cache: false,
179
- },
180
- 'test:once': {
181
- dependsOn: ['^build'],
182
- outputs: ['coverage/**'],
183
- },
184
- typecheck: {
185
- dependsOn: ['^build'],
186
- outputs: [],
187
- },
188
- lint: {
189
- outputs: [],
190
- },
191
- fmt: {
192
- outputs: [],
193
- },
194
- },
195
- };
164
+ // Build turbo.json
165
+ const turboConfig = {
166
+ $schema: 'https://turbo.build/schema.json',
167
+ tasks: {
168
+ build: {
169
+ dependsOn: ['^build'],
170
+ outputs: ['dist/**'],
171
+ },
172
+ dev: {
173
+ cache: false,
174
+ persistent: true,
175
+ },
176
+ test: {
177
+ dependsOn: ['^build'],
178
+ cache: false,
179
+ },
180
+ 'test:once': {
181
+ dependsOn: ['^build'],
182
+ outputs: ['coverage/**'],
183
+ },
184
+ typecheck: {
185
+ dependsOn: ['^build'],
186
+ outputs: [],
187
+ },
188
+ lint: {
189
+ outputs: [],
190
+ },
191
+ fmt: {
192
+ outputs: [],
193
+ },
194
+ },
195
+ };
196
196
 
197
- return [
198
- {
199
- path: 'gkm.config.ts',
200
- content: gkmConfig,
201
- },
202
- {
203
- path: 'tsconfig.json',
204
- content: JSON.stringify(tsConfig, null, 2) + '\n',
205
- },
206
- {
207
- path: 'biome.json',
208
- content: JSON.stringify(biomeConfig, null, 2) + '\n',
209
- },
210
- {
211
- path: 'turbo.json',
212
- content: JSON.stringify(turboConfig, null, 2) + '\n',
213
- },
214
- ];
197
+ return [
198
+ {
199
+ path: 'gkm.config.ts',
200
+ content: gkmConfig,
201
+ },
202
+ {
203
+ path: 'tsconfig.json',
204
+ content: `${JSON.stringify(tsConfig, null, 2)}\n`,
205
+ },
206
+ {
207
+ path: 'biome.json',
208
+ content: `${JSON.stringify(biomeConfig, null, 2)}\n`,
209
+ },
210
+ {
211
+ path: 'turbo.json',
212
+ content: `${JSON.stringify(turboConfig, null, 2)}\n`,
213
+ },
214
+ ];
215
215
  }