@livestore/utils-dev 0.0.0-snapshot-9d8807d2c51c95b4df3556744702cea55dc7ded3 → 0.0.0-snapshot-764973e035e7322a3936e1c79f2e78820e04f73c

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 (82) hide show
  1. package/dist/.tsbuildinfo.json +1 -1
  2. package/dist/node/DockerComposeService/DockerComposeService.d.ts +58 -0
  3. package/dist/node/DockerComposeService/DockerComposeService.d.ts.map +1 -0
  4. package/dist/node/DockerComposeService/DockerComposeService.js +144 -0
  5. package/dist/node/DockerComposeService/DockerComposeService.js.map +1 -0
  6. package/dist/node/DockerComposeService/DockerComposeService.test.d.ts +2 -0
  7. package/dist/node/DockerComposeService/DockerComposeService.test.d.ts.map +1 -0
  8. package/dist/node/DockerComposeService/DockerComposeService.test.js +64 -0
  9. package/dist/node/DockerComposeService/DockerComposeService.test.js.map +1 -0
  10. package/dist/node/FileLogger.d.ts +14 -0
  11. package/dist/node/FileLogger.d.ts.map +1 -0
  12. package/dist/node/FileLogger.js +151 -0
  13. package/dist/node/FileLogger.js.map +1 -0
  14. package/dist/node/cmd-log.d.ts +21 -0
  15. package/dist/node/cmd-log.d.ts.map +1 -0
  16. package/dist/node/cmd-log.js +50 -0
  17. package/dist/node/cmd-log.js.map +1 -0
  18. package/dist/node/cmd.d.ts +36 -0
  19. package/dist/node/cmd.d.ts.map +1 -0
  20. package/dist/node/cmd.js +234 -0
  21. package/dist/node/cmd.js.map +1 -0
  22. package/dist/node/cmd.test.d.ts +2 -0
  23. package/dist/node/cmd.test.d.ts.map +1 -0
  24. package/dist/node/cmd.test.js +101 -0
  25. package/dist/node/cmd.test.js.map +1 -0
  26. package/dist/node/mod.d.ts +5 -13
  27. package/dist/node/mod.d.ts.map +1 -1
  28. package/dist/node/mod.js +67 -42
  29. package/dist/node/mod.js.map +1 -1
  30. package/dist/node-vitest/Vitest.d.ts +52 -0
  31. package/dist/node-vitest/Vitest.d.ts.map +1 -0
  32. package/dist/node-vitest/Vitest.js +98 -0
  33. package/dist/node-vitest/Vitest.js.map +1 -0
  34. package/dist/node-vitest/Vitest.test.d.ts +2 -0
  35. package/dist/node-vitest/Vitest.test.d.ts.map +1 -0
  36. package/dist/node-vitest/Vitest.test.js +70 -0
  37. package/dist/node-vitest/Vitest.test.js.map +1 -0
  38. package/dist/node-vitest/global.d.ts +2 -0
  39. package/dist/node-vitest/global.d.ts.map +1 -0
  40. package/dist/node-vitest/{polyfill.js → global.js} +1 -1
  41. package/dist/node-vitest/global.js.map +1 -0
  42. package/dist/node-vitest/mod.d.ts +2 -1
  43. package/dist/node-vitest/mod.d.ts.map +1 -1
  44. package/dist/node-vitest/mod.js +2 -1
  45. package/dist/node-vitest/mod.js.map +1 -1
  46. package/dist/wrangler/WranglerDevServer.d.ts +52 -0
  47. package/dist/wrangler/WranglerDevServer.d.ts.map +1 -0
  48. package/dist/wrangler/WranglerDevServer.js +90 -0
  49. package/dist/wrangler/WranglerDevServer.js.map +1 -0
  50. package/dist/wrangler/WranglerDevServer.test.d.ts +2 -0
  51. package/dist/wrangler/WranglerDevServer.test.d.ts.map +1 -0
  52. package/dist/wrangler/WranglerDevServer.test.js +77 -0
  53. package/dist/wrangler/WranglerDevServer.test.js.map +1 -0
  54. package/dist/wrangler/fixtures/cf-worker.d.ts +8 -0
  55. package/dist/wrangler/fixtures/cf-worker.d.ts.map +1 -0
  56. package/dist/wrangler/fixtures/cf-worker.js +11 -0
  57. package/dist/wrangler/fixtures/cf-worker.js.map +1 -0
  58. package/dist/wrangler/mod.d.ts +2 -0
  59. package/dist/wrangler/mod.d.ts.map +1 -0
  60. package/dist/wrangler/mod.js +2 -0
  61. package/dist/wrangler/mod.js.map +1 -0
  62. package/package.json +16 -22
  63. package/src/node/DockerComposeService/DockerComposeService.test.ts +91 -0
  64. package/src/node/DockerComposeService/DockerComposeService.ts +328 -0
  65. package/src/node/DockerComposeService/test-fixtures/docker-compose.yml +4 -0
  66. package/src/node/FileLogger.ts +206 -0
  67. package/src/node/cmd-log.ts +92 -0
  68. package/src/node/cmd.test.ts +129 -0
  69. package/src/node/cmd.ts +419 -0
  70. package/src/node/mod.ts +81 -82
  71. package/src/node-vitest/Vitest.test.ts +112 -0
  72. package/src/node-vitest/Vitest.ts +238 -0
  73. package/src/node-vitest/mod.ts +3 -1
  74. package/src/wrangler/WranglerDevServer.test.ts +133 -0
  75. package/src/wrangler/WranglerDevServer.ts +180 -0
  76. package/src/wrangler/fixtures/cf-worker.ts +11 -0
  77. package/src/wrangler/fixtures/wrangler.toml +11 -0
  78. package/src/wrangler/mod.ts +6 -0
  79. package/dist/node-vitest/polyfill.d.ts +0 -2
  80. package/dist/node-vitest/polyfill.d.ts.map +0 -1
  81. package/dist/node-vitest/polyfill.js.map +0 -1
  82. /package/src/node-vitest/{polyfill.ts → global.ts} +0 -0
@@ -0,0 +1,133 @@
1
+ import { Effect, FetchHttpClient, Layer } from '@livestore/utils/effect'
2
+ import { getFreePort, PlatformNode } from '@livestore/utils/node'
3
+ import { Vitest } from '@livestore/utils-dev/node-vitest'
4
+ import { expect } from 'vitest'
5
+ import {
6
+ type StartWranglerDevServerArgs,
7
+ WranglerDevServerError,
8
+ WranglerDevServerService,
9
+ } from './WranglerDevServer.ts'
10
+
11
+ const testTimeout = 60_000
12
+
13
+ const WranglerDevServerTest = (args: Partial<StartWranglerDevServerArgs> = {}) =>
14
+ WranglerDevServerService.Default({
15
+ cwd: `${import.meta.dirname}/fixtures`,
16
+ ...args,
17
+ }).pipe(Layer.provide(FetchHttpClient.layer))
18
+
19
+ Vitest.describe('WranglerDevServer', { timeout: testTimeout }, () => {
20
+ Vitest.describe('Basic Operations', () => {
21
+ const withBasicTest = (args: Partial<StartWranglerDevServerArgs> = {}) =>
22
+ Vitest.makeWithTestCtx({
23
+ timeout: testTimeout,
24
+ makeLayer: () => WranglerDevServerTest(args).pipe(Layer.provide(PlatformNode.NodeContext.layer)),
25
+ })
26
+
27
+ Vitest.scopedLive('should start wrangler dev server and return port', (test) =>
28
+ Effect.gen(function* () {
29
+ const server = yield* WranglerDevServerService
30
+
31
+ expect(server.port).toBeGreaterThan(0)
32
+ expect(server.url).toMatch(/http:\/\/127.0.0.1:\d+/)
33
+ }).pipe(withBasicTest()(test)),
34
+ )
35
+
36
+ Vitest.scopedLive('should use specified port when provided', (test) =>
37
+ Effect.andThen(getFreePort, (port) =>
38
+ Effect.gen(function* () {
39
+ const server = yield* WranglerDevServerService
40
+
41
+ expect(server.port).toBe(port)
42
+ expect(server.url).toBe(`http://127.0.0.1:${port}`)
43
+ }).pipe(withBasicTest({ preferredPort: port })(test)),
44
+ ),
45
+ )
46
+ })
47
+
48
+ Vitest.describe('Error Handling', () => {
49
+ const withErrorTest = (args: Partial<StartWranglerDevServerArgs> = {}) =>
50
+ Vitest.makeWithTestCtx({
51
+ timeout: testTimeout,
52
+ makeLayer: () => WranglerDevServerTest(args).pipe(Layer.provide(PlatformNode.NodeContext.layer)),
53
+ })
54
+
55
+ Vitest.scopedLive('should handle missing wrangler.toml but should timeout', (test) =>
56
+ Effect.gen(function* () {
57
+ const error = yield* WranglerDevServerService.pipe(
58
+ Effect.provide(
59
+ WranglerDevServerTest({
60
+ cwd: '/tmp',
61
+ wranglerConfigPath: '/dev/null',
62
+ connectTimeout: '500 millis',
63
+ }).pipe(Layer.provide(PlatformNode.NodeContext.layer)),
64
+ ),
65
+ Effect.flip,
66
+ )
67
+
68
+ expect(error).toBeInstanceOf(WranglerDevServerError)
69
+ }).pipe(Vitest.withTestCtx(test)),
70
+ )
71
+
72
+ Vitest.scopedLive('should handle invalid working directory', (test) =>
73
+ Effect.gen(function* () {
74
+ const result = yield* WranglerDevServerService.pipe(
75
+ Effect.provide(
76
+ WranglerDevServerTest({
77
+ cwd: '/completely/nonexistent/directory',
78
+ }).pipe(Layer.provide(PlatformNode.NodeContext.layer)),
79
+ ),
80
+ Effect.either,
81
+ )
82
+
83
+ expect(result._tag).toBe('Left')
84
+ if (result._tag === 'Left') {
85
+ expect(result.left).toBeInstanceOf(WranglerDevServerError)
86
+ }
87
+ }).pipe(Vitest.withTestCtx(test)),
88
+ )
89
+
90
+ Vitest.scopedLive('should timeout if server fails to start', (test) =>
91
+ Effect.gen(function* () {
92
+ // Create a command that will never output "Ready on"
93
+ const result = yield* WranglerDevServerService.pipe(
94
+ // Override the timeout for this test to be shorter
95
+ Effect.timeout('5 seconds'),
96
+ Effect.either,
97
+ )
98
+
99
+ // This might succeed or fail depending on actual wrangler behavior
100
+ // The main point is testing timeout functionality
101
+ expect(['Left', 'Right']).toContain(result._tag)
102
+ }).pipe(withErrorTest()(test)),
103
+ )
104
+ })
105
+
106
+ Vitest.describe('Service Pattern', () => {
107
+ const withServiceTest = (args: Partial<StartWranglerDevServerArgs> = {}) =>
108
+ Vitest.makeWithTestCtx({
109
+ timeout: testTimeout,
110
+ makeLayer: () => WranglerDevServerTest(args).pipe(Layer.provide(PlatformNode.NodeContext.layer)),
111
+ })
112
+
113
+ Vitest.scopedLive('should work with service pattern', (test) =>
114
+ Effect.gen(function* () {
115
+ const server = yield* WranglerDevServerService
116
+
117
+ expect(server.port).toBeGreaterThan(0)
118
+ expect(server.url).toMatch(/http:\/\/127.0.0.1:\d+/)
119
+ }).pipe(withServiceTest()(test)),
120
+ )
121
+
122
+ Vitest.scopedLive('should work with custom port via service', (test) =>
123
+ Effect.andThen(getFreePort, (port) =>
124
+ Effect.gen(function* () {
125
+ const server = yield* WranglerDevServerService
126
+
127
+ expect(server.port).toBe(port)
128
+ expect(server.url).toBe(`http://127.0.0.1:${port}`)
129
+ }).pipe(withServiceTest({ preferredPort: port })(test)),
130
+ ),
131
+ )
132
+ })
133
+ })
@@ -0,0 +1,180 @@
1
+ import * as path from 'node:path'
2
+ import * as Toml from '@iarna/toml'
3
+ import { IS_CI } from '@livestore/utils'
4
+ import { Cause, Duration, Effect, FileSystem, HttpClient, Schedule, Schema } from '@livestore/utils/effect'
5
+ import { getFreePort } from '@livestore/utils/node'
6
+ import * as wrangler from 'wrangler'
7
+
8
+ /**
9
+ * Error type for WranglerDevServer operations
10
+ */
11
+ export class WranglerDevServerError extends Schema.TaggedError<WranglerDevServerError>()('WranglerDevServerError', {
12
+ cause: Schema.Unknown,
13
+ message: Schema.String,
14
+ port: Schema.Number,
15
+ }) {}
16
+
17
+ /**
18
+ * WranglerDevServer instance interface
19
+ */
20
+ export interface WranglerDevServer {
21
+ readonly port: number
22
+ readonly url: string
23
+ // readonly processId: number
24
+ }
25
+
26
+ /**
27
+ * Configuration for starting WranglerDevServer
28
+ */
29
+ export interface StartWranglerDevServerArgs {
30
+ wranglerConfigPath?: string
31
+ cwd: string
32
+ /** The port to try first. The dev server may bind a different port if unavailable. */
33
+ preferredPort?: number
34
+ /** @default false */
35
+ showLogs?: boolean
36
+ inspectorPort?: number
37
+ connectTimeout?: Duration.DurationInput
38
+ }
39
+
40
+ /**
41
+ * WranglerDevServer as an Effect.Service.
42
+ *
43
+ * This service provides the WranglerDevServer properties and can be accessed
44
+ * directly to get port and url.
45
+ *
46
+ * TODO: Allow for config to be passed in via code instead of `wrangler.toml` file
47
+ * (would need to be placed in temporary file as wrangler only accepts files as config)
48
+ */
49
+ export class WranglerDevServerService extends Effect.Service<WranglerDevServerService>()('WranglerDevServerService', {
50
+ scoped: (args: StartWranglerDevServerArgs) =>
51
+ Effect.gen(function* () {
52
+ const showLogs = args.showLogs ?? false
53
+
54
+ // Allocate preferred port (Wrangler may bind a different one if unavailable)
55
+ const preferredPort =
56
+ args.preferredPort ??
57
+ (yield* getFreePort.pipe(
58
+ Effect.mapError(
59
+ (cause) => new WranglerDevServerError({ cause, message: 'Failed to get free port', port: -1 }),
60
+ ),
61
+ ))
62
+
63
+ yield* Effect.annotateCurrentSpan({ preferredPort })
64
+
65
+ const configPath = path.resolve(args.wranglerConfigPath ?? path.join(args.cwd, 'wrangler.toml'))
66
+
67
+ const fs = yield* FileSystem.FileSystem
68
+ const configContent = yield* fs.readFileString(configPath)
69
+ const parsedConfig = yield* Effect.try(() => Toml.parse(configContent)).pipe(
70
+ Effect.andThen(Schema.decodeUnknown(Schema.Struct({ main: Schema.String }))),
71
+ Effect.mapError(
72
+ (error) => new WranglerDevServerError({ cause: error, message: 'Failed to parse wrangler config', port: -1 }),
73
+ ),
74
+ )
75
+ const resolvedMainPath = yield* Effect.try(() => path.resolve(args.cwd, parsedConfig.main))
76
+
77
+ const devServer = yield* Effect.promise(() =>
78
+ wrangler.unstable_dev(resolvedMainPath, {
79
+ config: configPath,
80
+ port: preferredPort,
81
+ inspectorPort: args.inspectorPort ?? 0,
82
+ persistTo: path.join(args.cwd, '.wrangler/state'),
83
+ logLevel: showLogs ? 'debug' : 'none',
84
+ experimental: {
85
+ disableExperimentalWarning: true,
86
+ },
87
+ }),
88
+ )
89
+
90
+ yield* Effect.addFinalizer(
91
+ Effect.fn(
92
+ function* (exit) {
93
+ if (exit._tag === 'Failure' && Cause.isInterruptedOnly(exit.cause) === false) {
94
+ yield* Effect.logError('Closing wrangler dev server on failure', exit.cause)
95
+ }
96
+
97
+ yield* Effect.tryPromise(async () => {
98
+ await devServer.stop()
99
+ // TODO investigate whether we need to wait until exit (see workers-sdk repo/talk to Cloudflare team)
100
+ // await devServer.waitUntilExit()
101
+ })
102
+ },
103
+ Effect.timeout('5 seconds'),
104
+ Effect.orDie,
105
+ Effect.tapCauseLogPretty,
106
+ Effect.withSpan('WranglerDevServerService:stopDevServer'),
107
+ ),
108
+ )
109
+
110
+ const actualPort = devServer.port
111
+ const actualHost = devServer.address
112
+ const url = `http://${actualHost}:${actualPort}`
113
+
114
+ // Use longer timeout in CI environments to account for slower startup times
115
+ const defaultTimeout = Duration.seconds(IS_CI ? 30 : 5)
116
+
117
+ yield* verifyHttpConnectivity({ url, showLogs, connectTimeout: args.connectTimeout ?? defaultTimeout })
118
+
119
+ if (showLogs) {
120
+ yield* Effect.logDebug(
121
+ `Wrangler dev server ready and accepting connections on port ${actualPort} (preferred: ${preferredPort})`,
122
+ )
123
+ }
124
+
125
+ return {
126
+ port: actualPort,
127
+ url,
128
+ } satisfies WranglerDevServer
129
+ }).pipe(
130
+ Effect.mapError(
131
+ (error) =>
132
+ new WranglerDevServerError({ cause: error, message: 'Failed to start wrangler dev server', port: -1 }),
133
+ ),
134
+ Effect.withSpan('WranglerDevServerService', {
135
+ attributes: { preferredPort: args.preferredPort ?? 'auto', cwd: args.cwd },
136
+ }),
137
+ ),
138
+ }) {}
139
+
140
+ /**
141
+ * Verifies the server is actually accepting HTTP connections by making a test request
142
+ */
143
+ const verifyHttpConnectivity = ({
144
+ url,
145
+ showLogs,
146
+ connectTimeout,
147
+ }: {
148
+ url: string
149
+ showLogs: boolean
150
+ connectTimeout: Duration.DurationInput
151
+ }): Effect.Effect<void, WranglerDevServerError, HttpClient.HttpClient> =>
152
+ Effect.gen(function* () {
153
+ const client = yield* HttpClient.HttpClient
154
+
155
+ if (showLogs) {
156
+ yield* Effect.logDebug(`Verifying HTTP connectivity to ${url}`)
157
+ }
158
+
159
+ // Try to connect with retries using exponential backoff
160
+ yield* client.get(url).pipe(
161
+ Effect.retryOrElse(
162
+ Schedule.exponential('50 millis', 2).pipe(
163
+ Schedule.jittered,
164
+ Schedule.intersect(Schedule.elapsed.pipe(Schedule.whileOutput(Duration.lessThanOrEqualTo(connectTimeout)))),
165
+ Schedule.compose(Schedule.count),
166
+ ),
167
+ (error, attemptCount) =>
168
+ Effect.fail(
169
+ new WranglerDevServerError({
170
+ cause: error,
171
+ message: `Failed to establish HTTP connection to Wrangler server at ${url} after ${attemptCount} attempts (timeout: ${Duration.toMillis(connectTimeout)}ms)`,
172
+ port: 0,
173
+ }),
174
+ ),
175
+ ),
176
+ Effect.tap(() => (showLogs ? Effect.logDebug(`HTTP connectivity verified for ${url}`) : Effect.void)),
177
+ Effect.asVoid,
178
+ Effect.withSpan('verifyHttpConnectivity'),
179
+ )
180
+ })
@@ -0,0 +1,11 @@
1
+ export default {
2
+ async fetch(_request: Request): Promise<Response> {
3
+ return new Response('Hello from Wrangler Dev Server test worker!')
4
+ },
5
+ }
6
+
7
+ export class TestDO {
8
+ async fetch(_request: Request): Promise<Response> {
9
+ return new Response('Hello from Test Durable Object!')
10
+ }
11
+ }
@@ -0,0 +1,11 @@
1
+ name = "wrangler-dev-server-test"
2
+ main = "./cf-worker.ts"
3
+ compatibility_date = "2024-05-12"
4
+
5
+ [[durable_objects.bindings]]
6
+ name = "TEST_DO"
7
+ class_name = "TestDO"
8
+
9
+ [[migrations]]
10
+ tag = "v1"
11
+ new_sqlite_classes = ["TestDO"]
@@ -0,0 +1,6 @@
1
+ export {
2
+ type StartWranglerDevServerArgs,
3
+ type WranglerDevServer,
4
+ WranglerDevServerError,
5
+ WranglerDevServerService,
6
+ } from './WranglerDevServer.ts'
@@ -1,2 +0,0 @@
1
- export {};
2
- //# sourceMappingURL=polyfill.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"polyfill.d.ts","sourceRoot":"","sources":["../../src/node-vitest/polyfill.ts"],"names":[],"mappings":""}
@@ -1 +0,0 @@
1
- {"version":3,"file":"polyfill.js","sourceRoot":"","sources":["../../src/node-vitest/polyfill.ts"],"names":[],"mappings":"AAAA,OAAO,CAAC,MAAM,CAAC,KAAK,GAAG,IAAI,CAAA"}
File without changes