wrangler 2.20.0 → 3.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.
- package/README.md +4 -4
- package/bin/wrangler.js +9 -75
- package/package.json +5 -13
- package/templates/__tests__/tsconfig.tsbuildinfo +1 -1
- package/templates/checked-fetch.js +1 -1
- package/templates/first-party-worker-module-facade.ts +2 -2
- package/templates/middleware/common.ts +9 -4
- package/templates/middleware/loader-sw.ts +2 -7
- package/templates/new-worker-scheduled.ts +1 -1
- package/templates/new-worker.ts +1 -1
- package/templates/pages-dev-util.ts +4 -1
- package/templates/pages-shim.ts +0 -3
- package/templates/tsconfig.tsbuildinfo +1 -1
- package/wrangler-dist/cli.d.ts +149 -75
- package/wrangler-dist/cli.js +60062 -64338
- package/import_meta_url.js +0 -3
- package/miniflare-config-stubs/.env.empty +0 -0
- package/miniflare-config-stubs/package.empty.json +0 -1
- package/miniflare-config-stubs/wrangler.empty.toml +0 -0
- package/miniflare-dist/index.mjs +0 -6442
- package/src/__tests__/access.test.ts +0 -25
- package/src/__tests__/api-dev.test.ts +0 -238
- package/src/__tests__/api-devregistry.test.ts +0 -121
- package/src/__tests__/api.test.ts +0 -102
- package/src/__tests__/config-cache-without-cache-dir.test.ts +0 -38
- package/src/__tests__/config-cache.test.ts +0 -42
- package/src/__tests__/configuration.test.ts +0 -4517
- package/src/__tests__/constellation.test.ts +0 -371
- package/src/__tests__/d1/d1.test.ts +0 -82
- package/src/__tests__/d1/execute.test.ts +0 -66
- package/src/__tests__/d1/migrate.test.ts +0 -257
- package/src/__tests__/d1/splitter.test.ts +0 -255
- package/src/__tests__/delete.test.ts +0 -272
- package/src/__tests__/deployments.test.ts +0 -369
- package/src/__tests__/dev.test.tsx +0 -1617
- package/src/__tests__/generate.test.ts +0 -237
- package/src/__tests__/get-host-from-url.test.ts +0 -16
- package/src/__tests__/guess-worker-format.test.ts +0 -120
- package/src/__tests__/helpers/clipboardy-mock.js +0 -4
- package/src/__tests__/helpers/cmd-shim.d.ts +0 -11
- package/src/__tests__/helpers/end-event-loop.ts +0 -6
- package/src/__tests__/helpers/mock-account-id.ts +0 -48
- package/src/__tests__/helpers/mock-auth-domain.ts +0 -20
- package/src/__tests__/helpers/mock-bin.ts +0 -36
- package/src/__tests__/helpers/mock-console.ts +0 -112
- package/src/__tests__/helpers/mock-dialogs.ts +0 -139
- package/src/__tests__/helpers/mock-get-pages-upload-token.ts +0 -25
- package/src/__tests__/helpers/mock-get-zone-from-host.ts +0 -11
- package/src/__tests__/helpers/mock-http-server.ts +0 -46
- package/src/__tests__/helpers/mock-istty.ts +0 -74
- package/src/__tests__/helpers/mock-known-routes.ts +0 -12
- package/src/__tests__/helpers/mock-kv.ts +0 -46
- package/src/__tests__/helpers/mock-oauth-flow.ts +0 -263
- package/src/__tests__/helpers/mock-process.ts +0 -34
- package/src/__tests__/helpers/mock-set-timeout.ts +0 -16
- package/src/__tests__/helpers/mock-stdin.ts +0 -108
- package/src/__tests__/helpers/mock-web-socket.ts +0 -29
- package/src/__tests__/helpers/msw/blob-worker.cjs +0 -19
- package/src/__tests__/helpers/msw/handlers/access.ts +0 -13
- package/src/__tests__/helpers/msw/handlers/deployments.ts +0 -160
- package/src/__tests__/helpers/msw/handlers/namespaces.ts +0 -81
- package/src/__tests__/helpers/msw/handlers/oauth.ts +0 -31
- package/src/__tests__/helpers/msw/handlers/r2.ts +0 -60
- package/src/__tests__/helpers/msw/handlers/script.ts +0 -56
- package/src/__tests__/helpers/msw/handlers/user.ts +0 -52
- package/src/__tests__/helpers/msw/handlers/zones.ts +0 -20
- package/src/__tests__/helpers/msw/index.ts +0 -52
- package/src/__tests__/helpers/msw/read-file-sync.js +0 -61
- package/src/__tests__/helpers/run-in-tmp.ts +0 -38
- package/src/__tests__/helpers/run-wrangler.ts +0 -16
- package/src/__tests__/helpers/string-dynamic-values-matcher.ts +0 -28
- package/src/__tests__/helpers/worker-scripts/child-wrangler.toml +0 -1
- package/src/__tests__/helpers/worker-scripts/hello-world-worker.js +0 -5
- package/src/__tests__/helpers/worker-scripts/hello-world-wrangler.toml +0 -1
- package/src/__tests__/helpers/worker-scripts/parent-worker.js +0 -11
- package/src/__tests__/helpers/worker-scripts/parent-wrangler.toml +0 -5
- package/src/__tests__/helpers/write-worker-source.ts +0 -31
- package/src/__tests__/helpers/write-wrangler-toml.ts +0 -17
- package/src/__tests__/https-options.test.ts +0 -163
- package/src/__tests__/index.test.ts +0 -282
- package/src/__tests__/init.test.ts +0 -3196
- package/src/__tests__/jest.setup.ts +0 -179
- package/src/__tests__/kv.test.ts +0 -1799
- package/src/__tests__/logger.test.ts +0 -207
- package/src/__tests__/logout.test.ts +0 -47
- package/src/__tests__/metrics.test.ts +0 -493
- package/src/__tests__/middleware.scheduled.test.ts +0 -145
- package/src/__tests__/middleware.test.ts +0 -816
- package/src/__tests__/mtls-certificates.test.ts +0 -589
- package/src/__tests__/package-manager.test.ts +0 -353
- package/src/__tests__/pages/deployment-list.test.ts +0 -80
- package/src/__tests__/pages/functions-build.test.ts +0 -528
- package/src/__tests__/pages/pages.test.ts +0 -81
- package/src/__tests__/pages/project-create.test.ts +0 -63
- package/src/__tests__/pages/project-list.test.ts +0 -110
- package/src/__tests__/pages/project-upload.test.ts +0 -500
- package/src/__tests__/pages/publish.test.ts +0 -2864
- package/src/__tests__/pages-deployment-tail.test.ts +0 -957
- package/src/__tests__/parse.test.ts +0 -436
- package/src/__tests__/paths.test.ts +0 -39
- package/src/__tests__/publish.test.ts +0 -8849
- package/src/__tests__/pubsub.test.ts +0 -496
- package/src/__tests__/queues.test.ts +0 -532
- package/src/__tests__/r2.test.ts +0 -374
- package/src/__tests__/route.test.ts +0 -45
- package/src/__tests__/secret.test.ts +0 -693
- package/src/__tests__/tail.test.ts +0 -989
- package/src/__tests__/test-old-node-version.js +0 -31
- package/src/__tests__/traverse-module-graph.test.ts +0 -220
- package/src/__tests__/tsconfig-sanity.ts +0 -12
- package/src/__tests__/tsconfig.json +0 -8
- package/src/__tests__/tsconfig.tsbuildinfo +0 -1
- package/src/__tests__/type-generation.test.ts +0 -234
- package/src/__tests__/user.test.ts +0 -118
- package/src/__tests__/utils-collectKeyValues.test.ts +0 -47
- package/src/__tests__/validate-dev-props.test.ts +0 -56
- package/src/__tests__/version.test.ts +0 -35
- package/src/__tests__/whoami.test.tsx +0 -172
- package/src/__tests__/worker-namespace.test.ts +0 -340
- package/src/abort.d.ts +0 -3
- package/src/api/dev.ts +0 -321
- package/src/api/index.ts +0 -11
- package/src/api/mtls-certificate.ts +0 -148
- package/src/api/pages/create-worker-bundle-contents.ts +0 -77
- package/src/api/pages/index.ts +0 -5
- package/src/api/pages/publish.tsx +0 -371
- package/src/bundle-reporter.ts +0 -68
- package/src/bundle.ts +0 -929
- package/src/cfetch/index.ts +0 -158
- package/src/cfetch/internal.ts +0 -258
- package/src/cli.ts +0 -28
- package/src/config/README.md +0 -107
- package/src/config/config.ts +0 -282
- package/src/config/diagnostics.ts +0 -80
- package/src/config/environment.ts +0 -625
- package/src/config/index.ts +0 -403
- package/src/config/validation-helpers.ts +0 -597
- package/src/config/validation.ts +0 -2369
- package/src/config-cache.ts +0 -85
- package/src/constellation/createProject.tsx +0 -51
- package/src/constellation/deleteProject.ts +0 -51
- package/src/constellation/deleteProjectModel.ts +0 -68
- package/src/constellation/index.ts +0 -75
- package/src/constellation/listCatalog.tsx +0 -35
- package/src/constellation/listModel.tsx +0 -41
- package/src/constellation/listProject.tsx +0 -28
- package/src/constellation/listRuntime.tsx +0 -28
- package/src/constellation/options.ts +0 -17
- package/src/constellation/types.ts +0 -17
- package/src/constellation/uploadModel.tsx +0 -64
- package/src/constellation/utils.ts +0 -90
- package/src/create-worker-preview.ts +0 -293
- package/src/create-worker-upload-form.ts +0 -363
- package/src/d1/backups.tsx +0 -219
- package/src/d1/constants.ts +0 -2
- package/src/d1/create.tsx +0 -70
- package/src/d1/delete.ts +0 -53
- package/src/d1/execute.tsx +0 -357
- package/src/d1/formatTimeAgo.ts +0 -14
- package/src/d1/index.ts +0 -100
- package/src/d1/list.tsx +0 -62
- package/src/d1/migrations/apply.tsx +0 -212
- package/src/d1/migrations/create.tsx +0 -79
- package/src/d1/migrations/helpers.ts +0 -169
- package/src/d1/migrations/index.ts +0 -3
- package/src/d1/migrations/list.tsx +0 -95
- package/src/d1/migrations/options.ts +0 -23
- package/src/d1/options.ts +0 -22
- package/src/d1/splitter.ts +0 -161
- package/src/d1/types.ts +0 -25
- package/src/d1/utils.ts +0 -49
- package/src/delete.ts +0 -100
- package/src/deployments.ts +0 -368
- package/src/deprecated/index.ts +0 -144
- package/src/dev/dev-vars.ts +0 -39
- package/src/dev/dev.tsx +0 -605
- package/src/dev/get-local-persistence-path.ts +0 -31
- package/src/dev/local.tsx +0 -952
- package/src/dev/remote.tsx +0 -635
- package/src/dev/start-server.ts +0 -545
- package/src/dev/use-esbuild.ts +0 -215
- package/src/dev/validate-dev-props.ts +0 -40
- package/src/dev-registry.ts +0 -202
- package/src/dev.tsx +0 -934
- package/src/dialogs.ts +0 -136
- package/src/dispatch-namespace.ts +0 -211
- package/src/docs/helpers.ts +0 -50
- package/src/docs/index.ts +0 -54
- package/src/durable.ts +0 -102
- package/src/entry.ts +0 -344
- package/src/environment-variables/factory.ts +0 -89
- package/src/environment-variables/misc-variables.ts +0 -30
- package/src/errors.ts +0 -11
- package/src/generate/index.ts +0 -298
- package/src/git-client.ts +0 -135
- package/src/global-wrangler-config-path.ts +0 -26
- package/src/https-options.ts +0 -127
- package/src/index.ts +0 -768
- package/src/init.ts +0 -1037
- package/src/inspect.ts +0 -883
- package/src/intl-polyfill.d.ts +0 -139
- package/src/is-ci.ts +0 -14
- package/src/is-interactive.ts +0 -16
- package/src/jest.d.ts +0 -4
- package/src/kv/helpers.ts +0 -433
- package/src/kv/index.ts +0 -594
- package/src/logger.ts +0 -123
- package/src/metrics/index.ts +0 -5
- package/src/metrics/metrics-config.ts +0 -239
- package/src/metrics/metrics-dispatcher.ts +0 -96
- package/src/metrics/metrics-usage-headers.ts +0 -24
- package/src/metrics/send-event.ts +0 -99
- package/src/miniflare-cli/README.md +0 -30
- package/src/miniflare-cli/assets.ts +0 -251
- package/src/miniflare-cli/index.ts +0 -210
- package/src/miniflare-cli/request-context.ts +0 -40
- package/src/miniflare-cli/tsconfig.json +0 -9
- package/src/miniflare-cli/tsconfig.tsbuildinfo +0 -1
- package/src/miniflare-cli/types.ts +0 -11
- package/src/module-collection.ts +0 -333
- package/src/mtls-certificate/cli.ts +0 -155
- package/src/open-in-browser.ts +0 -17
- package/src/package-manager.ts +0 -219
- package/src/pages/build.ts +0 -423
- package/src/pages/buildFunctions.ts +0 -140
- package/src/pages/constants.ts +0 -18
- package/src/pages/deployment-tails.ts +0 -281
- package/src/pages/deployments.tsx +0 -84
- package/src/pages/dev.ts +0 -734
- package/src/pages/errors.ts +0 -67
- package/src/pages/functions/buildPlugin.ts +0 -114
- package/src/pages/functions/buildWorker.ts +0 -350
- package/src/pages/functions/filepath-routing.test.ts +0 -234
- package/src/pages/functions/filepath-routing.ts +0 -189
- package/src/pages/functions/identifiers.ts +0 -78
- package/src/pages/functions/routes-consolidation.test.ts +0 -250
- package/src/pages/functions/routes-consolidation.ts +0 -73
- package/src/pages/functions/routes-transformation.test.ts +0 -282
- package/src/pages/functions/routes-transformation.ts +0 -115
- package/src/pages/functions/routes-validation.test.ts +0 -403
- package/src/pages/functions/routes-validation.ts +0 -202
- package/src/pages/functions/routes.ts +0 -151
- package/src/pages/functions/tsconfig.json +0 -8
- package/src/pages/functions/tsconfig.tsbuildinfo +0 -1
- package/src/pages/functions.ts +0 -86
- package/src/pages/hash.ts +0 -13
- package/src/pages/index.ts +0 -102
- package/src/pages/projects.tsx +0 -159
- package/src/pages/prompt-select-project.tsx +0 -31
- package/src/pages/publish.tsx +0 -267
- package/src/pages/types.ts +0 -46
- package/src/pages/upload.tsx +0 -469
- package/src/pages/utils.ts +0 -23
- package/src/parse.ts +0 -308
- package/src/paths.ts +0 -71
- package/src/proxy.ts +0 -694
- package/src/publish/index.ts +0 -274
- package/src/publish/publish.ts +0 -1065
- package/src/pubsub/index.ts +0 -286
- package/src/pubsub/pubsub-commands.ts +0 -623
- package/src/queues/cli/commands/consumer/add.ts +0 -71
- package/src/queues/cli/commands/consumer/index.ts +0 -19
- package/src/queues/cli/commands/consumer/remove.ts +0 -31
- package/src/queues/cli/commands/create.ts +0 -25
- package/src/queues/cli/commands/delete.ts +0 -26
- package/src/queues/cli/commands/index.ts +0 -35
- package/src/queues/cli/commands/list.ts +0 -25
- package/src/queues/client.ts +0 -136
- package/src/queues/utils.ts +0 -18
- package/src/r2/constants.ts +0 -4
- package/src/r2/helpers.ts +0 -132
- package/src/r2/index.ts +0 -289
- package/src/routes.ts +0 -140
- package/src/secret/index.ts +0 -377
- package/src/selfsigned.d.ts +0 -29
- package/src/sites.ts +0 -484
- package/src/tail/createTail.ts +0 -415
- package/src/tail/filters.ts +0 -277
- package/src/tail/index.ts +0 -211
- package/src/tail/printing.ts +0 -132
- package/src/traverse-module-graph.ts +0 -54
- package/src/tsconfig-sanity.ts +0 -16
- package/src/type-generation.ts +0 -181
- package/src/update-check.ts +0 -19
- package/src/user/access.ts +0 -68
- package/src/user/auth-variables.ts +0 -113
- package/src/user/choose-account.tsx +0 -39
- package/src/user/generate-auth-url.ts +0 -33
- package/src/user/generate-random-state.ts +0 -16
- package/src/user/index.ts +0 -2
- package/src/user/user.ts +0 -1234
- package/src/utils/collectKeyValues.ts +0 -14
- package/src/utils/render.ts +0 -93
- package/src/whoami.ts +0 -135
- package/src/worker.ts +0 -279
- package/src/yargs-types.ts +0 -37
- package/src/zones.ts +0 -191
package/src/dev/dev.tsx
DELETED
|
@@ -1,605 +0,0 @@
|
|
|
1
|
-
import { spawn } from "node:child_process";
|
|
2
|
-
import * as path from "node:path";
|
|
3
|
-
import * as util from "node:util";
|
|
4
|
-
import { watch } from "chokidar";
|
|
5
|
-
import clipboardy from "clipboardy";
|
|
6
|
-
import commandExists from "command-exists";
|
|
7
|
-
import { Box, Text, useApp, useInput, useStdin } from "ink";
|
|
8
|
-
import React, { useState, useEffect, useRef, useMemo } from "react";
|
|
9
|
-
import { withErrorBoundary, useErrorHandler } from "react-error-boundary";
|
|
10
|
-
import onExit from "signal-exit";
|
|
11
|
-
import tmp from "tmp-promise";
|
|
12
|
-
import { fetch } from "undici";
|
|
13
|
-
import {
|
|
14
|
-
getBoundRegisteredWorkers,
|
|
15
|
-
startWorkerRegistry,
|
|
16
|
-
stopWorkerRegistry,
|
|
17
|
-
unregisterWorker,
|
|
18
|
-
} from "../dev-registry";
|
|
19
|
-
import { runCustomBuild } from "../entry";
|
|
20
|
-
import { openInspector } from "../inspect";
|
|
21
|
-
import { logger } from "../logger";
|
|
22
|
-
import openInBrowser from "../open-in-browser";
|
|
23
|
-
import { Local } from "./local";
|
|
24
|
-
import { Remote } from "./remote";
|
|
25
|
-
import { useEsbuild } from "./use-esbuild";
|
|
26
|
-
import { validateDevProps } from "./validate-dev-props";
|
|
27
|
-
import type { Config } from "../config";
|
|
28
|
-
import type { Route } from "../config/environment";
|
|
29
|
-
import type { WorkerRegistry } from "../dev-registry";
|
|
30
|
-
import type { Entry } from "../entry";
|
|
31
|
-
import type { EnablePagesAssetsServiceBindingOptions } from "../miniflare-cli/types";
|
|
32
|
-
import type { AssetPaths } from "../sites";
|
|
33
|
-
import type { CfWorkerInit } from "../worker";
|
|
34
|
-
|
|
35
|
-
/**
|
|
36
|
-
* This hooks establishes a connection with the dev registry,
|
|
37
|
-
* and periodically updates itself with details of workers currently
|
|
38
|
-
* running a dev session on this system.
|
|
39
|
-
*/
|
|
40
|
-
function useDevRegistry(
|
|
41
|
-
name: string | undefined,
|
|
42
|
-
services: Config["services"] | undefined,
|
|
43
|
-
durableObjects: Config["durable_objects"] | undefined,
|
|
44
|
-
mode: "local" | "remote"
|
|
45
|
-
): WorkerRegistry {
|
|
46
|
-
const [workers, setWorkers] = useState<WorkerRegistry>({});
|
|
47
|
-
|
|
48
|
-
useEffect(() => {
|
|
49
|
-
// Let's try to start registry
|
|
50
|
-
// TODO: we should probably call this in a loop
|
|
51
|
-
// in case the registry dies elsewhere
|
|
52
|
-
startWorkerRegistry().catch((err) => {
|
|
53
|
-
logger.error("failed to start worker registry", err);
|
|
54
|
-
});
|
|
55
|
-
|
|
56
|
-
const interval =
|
|
57
|
-
// TODO: enable this for remote mode as well
|
|
58
|
-
// https://github.com/cloudflare/workers-sdk/issues/1182
|
|
59
|
-
mode === "local"
|
|
60
|
-
? setInterval(() => {
|
|
61
|
-
getBoundRegisteredWorkers({
|
|
62
|
-
services,
|
|
63
|
-
durableObjects,
|
|
64
|
-
}).then(
|
|
65
|
-
(boundRegisteredWorkers: WorkerRegistry | undefined) => {
|
|
66
|
-
setWorkers((prevWorkers) => {
|
|
67
|
-
if (
|
|
68
|
-
!util.isDeepStrictEqual(boundRegisteredWorkers, prevWorkers)
|
|
69
|
-
) {
|
|
70
|
-
return boundRegisteredWorkers || {};
|
|
71
|
-
}
|
|
72
|
-
return prevWorkers;
|
|
73
|
-
});
|
|
74
|
-
},
|
|
75
|
-
(err) => {
|
|
76
|
-
logger.warn("Failed to get worker definitions", err);
|
|
77
|
-
}
|
|
78
|
-
);
|
|
79
|
-
}, 300)
|
|
80
|
-
: undefined;
|
|
81
|
-
|
|
82
|
-
return () => {
|
|
83
|
-
interval && clearInterval(interval);
|
|
84
|
-
Promise.allSettled([
|
|
85
|
-
name ? unregisterWorker(name) : Promise.resolve(),
|
|
86
|
-
stopWorkerRegistry(),
|
|
87
|
-
]).then(
|
|
88
|
-
([unregisterResult, stopRegistryResult]) => {
|
|
89
|
-
if (unregisterResult.status === "rejected") {
|
|
90
|
-
logger.error(
|
|
91
|
-
"Failed to unregister worker",
|
|
92
|
-
unregisterResult.reason
|
|
93
|
-
);
|
|
94
|
-
}
|
|
95
|
-
if (stopRegistryResult.status === "rejected") {
|
|
96
|
-
logger.error(
|
|
97
|
-
"Failed to stop worker registry",
|
|
98
|
-
stopRegistryResult.reason
|
|
99
|
-
);
|
|
100
|
-
}
|
|
101
|
-
},
|
|
102
|
-
(err) => {
|
|
103
|
-
logger.error("Failed to clear dev registry effect", err);
|
|
104
|
-
}
|
|
105
|
-
);
|
|
106
|
-
};
|
|
107
|
-
}, [name, services, durableObjects, mode]);
|
|
108
|
-
|
|
109
|
-
return workers;
|
|
110
|
-
}
|
|
111
|
-
|
|
112
|
-
export type DevProps = {
|
|
113
|
-
name: string | undefined;
|
|
114
|
-
noBundle: boolean;
|
|
115
|
-
entry: Entry;
|
|
116
|
-
initialPort: number;
|
|
117
|
-
initialIp: string;
|
|
118
|
-
inspectorPort: number;
|
|
119
|
-
processEntrypoint: boolean;
|
|
120
|
-
rules: Config["rules"];
|
|
121
|
-
accountId: string | undefined;
|
|
122
|
-
initialMode: "local" | "remote";
|
|
123
|
-
jsxFactory: string | undefined;
|
|
124
|
-
jsxFragment: string | undefined;
|
|
125
|
-
tsconfig: string | undefined;
|
|
126
|
-
upstreamProtocol: "https" | "http";
|
|
127
|
-
localProtocol: "https" | "http";
|
|
128
|
-
localUpstream: string | undefined;
|
|
129
|
-
localPersistencePath: string | null;
|
|
130
|
-
liveReload: boolean;
|
|
131
|
-
bindings: CfWorkerInit["bindings"];
|
|
132
|
-
define: Config["define"];
|
|
133
|
-
crons: Config["triggers"]["crons"];
|
|
134
|
-
queueConsumers: Config["queues"]["consumers"];
|
|
135
|
-
isWorkersSite: boolean;
|
|
136
|
-
assetPaths: AssetPaths | undefined;
|
|
137
|
-
assetsConfig: Config["assets"];
|
|
138
|
-
compatibilityDate: string;
|
|
139
|
-
compatibilityFlags: string[] | undefined;
|
|
140
|
-
usageModel: "bundled" | "unbound" | undefined;
|
|
141
|
-
minify: boolean | undefined;
|
|
142
|
-
legacyNodeCompat: boolean | undefined;
|
|
143
|
-
nodejsCompat: boolean | undefined;
|
|
144
|
-
build: Config["build"];
|
|
145
|
-
env: string | undefined;
|
|
146
|
-
legacyEnv: boolean;
|
|
147
|
-
zone: string | undefined;
|
|
148
|
-
host: string | undefined;
|
|
149
|
-
routes: Route[] | undefined;
|
|
150
|
-
inspect: boolean;
|
|
151
|
-
onReady: ((ip: string, port: number) => void) | undefined;
|
|
152
|
-
showInteractiveDevSession: boolean | undefined;
|
|
153
|
-
forceLocal: boolean | undefined;
|
|
154
|
-
enablePagesAssetsServiceBinding?: EnablePagesAssetsServiceBindingOptions;
|
|
155
|
-
firstPartyWorker: boolean | undefined;
|
|
156
|
-
sendMetrics: boolean | undefined;
|
|
157
|
-
testScheduled: boolean | undefined;
|
|
158
|
-
experimentalLocal: boolean | undefined;
|
|
159
|
-
experimentalLocalRemoteKv: boolean | undefined;
|
|
160
|
-
};
|
|
161
|
-
|
|
162
|
-
export function DevImplementation(props: DevProps): JSX.Element {
|
|
163
|
-
validateDevProps(props);
|
|
164
|
-
|
|
165
|
-
// only load the UI if we're running in a supported environment
|
|
166
|
-
const { isRawModeSupported } = useStdin();
|
|
167
|
-
|
|
168
|
-
return props.showInteractiveDevSession ?? isRawModeSupported ? (
|
|
169
|
-
<InteractiveDevSession {...props} />
|
|
170
|
-
) : (
|
|
171
|
-
<DevSession {...props} local={props.initialMode === "local"} />
|
|
172
|
-
);
|
|
173
|
-
}
|
|
174
|
-
|
|
175
|
-
// This is a nasty hack to allow `useHotkeys` and its "[b] open a browser" feature to read these values
|
|
176
|
-
// without triggering a re-render loop when `onReady()` updates them.
|
|
177
|
-
// The initially requested port can be different than what's actually used, if, for example, you request port 0.
|
|
178
|
-
let ip: string;
|
|
179
|
-
let port: number;
|
|
180
|
-
|
|
181
|
-
function InteractiveDevSession(props: DevProps) {
|
|
182
|
-
const toggles = useHotkeys({
|
|
183
|
-
initial: {
|
|
184
|
-
local: props.initialMode === "local",
|
|
185
|
-
tunnel: false,
|
|
186
|
-
},
|
|
187
|
-
inspectorPort: props.inspectorPort,
|
|
188
|
-
inspect: props.inspect,
|
|
189
|
-
localProtocol: props.localProtocol,
|
|
190
|
-
forceLocal: props.forceLocal,
|
|
191
|
-
worker: props.name,
|
|
192
|
-
});
|
|
193
|
-
|
|
194
|
-
ip = props.initialIp;
|
|
195
|
-
port = props.initialPort;
|
|
196
|
-
|
|
197
|
-
useTunnel(toggles.tunnel);
|
|
198
|
-
|
|
199
|
-
const onReady = (newIp: string, newPort: number) => {
|
|
200
|
-
if (newIp !== props.initialIp || newPort !== props.initialPort) {
|
|
201
|
-
ip = newIp;
|
|
202
|
-
port = newPort;
|
|
203
|
-
if (props.onReady) {
|
|
204
|
-
props.onReady(newIp, newPort);
|
|
205
|
-
}
|
|
206
|
-
}
|
|
207
|
-
};
|
|
208
|
-
|
|
209
|
-
return (
|
|
210
|
-
<>
|
|
211
|
-
<DevSession {...props} local={toggles.local} onReady={onReady} />
|
|
212
|
-
<Box borderStyle="round" paddingLeft={1} paddingRight={1}>
|
|
213
|
-
<Text bold={true}>[b]</Text>
|
|
214
|
-
<Text> open a browser, </Text>
|
|
215
|
-
{props.inspect ? (
|
|
216
|
-
<>
|
|
217
|
-
<Text bold={true}>[d]</Text>
|
|
218
|
-
<Text> open Devtools, </Text>
|
|
219
|
-
</>
|
|
220
|
-
) : null}
|
|
221
|
-
{!props.forceLocal ? (
|
|
222
|
-
<>
|
|
223
|
-
<Text bold={true}>[l]</Text>
|
|
224
|
-
<Text> {toggles.local ? "turn off" : "turn on"} local mode, </Text>
|
|
225
|
-
</>
|
|
226
|
-
) : null}
|
|
227
|
-
<Text bold={true}>[c]</Text>
|
|
228
|
-
<Text> clear console, </Text>
|
|
229
|
-
<Text bold={true}>[x]</Text>
|
|
230
|
-
<Text> to exit</Text>
|
|
231
|
-
</Box>
|
|
232
|
-
</>
|
|
233
|
-
);
|
|
234
|
-
}
|
|
235
|
-
|
|
236
|
-
type DevSessionProps = DevProps & {
|
|
237
|
-
local: boolean;
|
|
238
|
-
experimentalLocal?: boolean;
|
|
239
|
-
};
|
|
240
|
-
|
|
241
|
-
function DevSession(props: DevSessionProps) {
|
|
242
|
-
useCustomBuild(props.entry, props.build);
|
|
243
|
-
|
|
244
|
-
const directory = useTmpDir();
|
|
245
|
-
const handleError = useErrorHandler();
|
|
246
|
-
|
|
247
|
-
// Note: when D1 is out of beta, this (and all instances of `betaD1Shims`) can be removed.
|
|
248
|
-
// Additionally, useMemo is used so that new arrays aren't created on every render
|
|
249
|
-
// cause re-rendering further down.
|
|
250
|
-
const betaD1Shims = useMemo(
|
|
251
|
-
() => props.bindings.d1_databases?.map((db) => db.binding),
|
|
252
|
-
[props.bindings.d1_databases]
|
|
253
|
-
);
|
|
254
|
-
|
|
255
|
-
// If we are using d1 bindings, and are not bundling the worker
|
|
256
|
-
// we should error here as the d1 shim won't be added
|
|
257
|
-
if (Array.isArray(betaD1Shims) && betaD1Shims.length > 0 && props.noBundle) {
|
|
258
|
-
handleError(
|
|
259
|
-
new Error(
|
|
260
|
-
"While in beta, you cannot use D1 bindings without bundling your worker. Please remove `no_bundle` from your wrangler.toml file or remove the `--no-bundle` flag to access D1 bindings."
|
|
261
|
-
)
|
|
262
|
-
);
|
|
263
|
-
}
|
|
264
|
-
|
|
265
|
-
const workerDefinitions = useDevRegistry(
|
|
266
|
-
props.name,
|
|
267
|
-
props.bindings.services,
|
|
268
|
-
props.bindings.durable_objects,
|
|
269
|
-
props.local ? "local" : "remote"
|
|
270
|
-
);
|
|
271
|
-
|
|
272
|
-
const bundle = useEsbuild({
|
|
273
|
-
entry: props.entry,
|
|
274
|
-
destination: directory,
|
|
275
|
-
jsxFactory: props.jsxFactory,
|
|
276
|
-
processEntrypoint: props.processEntrypoint,
|
|
277
|
-
rules: props.rules,
|
|
278
|
-
jsxFragment: props.jsxFragment,
|
|
279
|
-
serveAssetsFromWorker: Boolean(
|
|
280
|
-
props.assetPaths && !props.isWorkersSite && props.local
|
|
281
|
-
),
|
|
282
|
-
tsconfig: props.tsconfig,
|
|
283
|
-
minify: props.minify,
|
|
284
|
-
legacyNodeCompat: props.legacyNodeCompat,
|
|
285
|
-
nodejsCompat: props.nodejsCompat,
|
|
286
|
-
betaD1Shims,
|
|
287
|
-
define: props.define,
|
|
288
|
-
noBundle: props.noBundle,
|
|
289
|
-
assets: props.assetsConfig,
|
|
290
|
-
workerDefinitions,
|
|
291
|
-
services: props.bindings.services,
|
|
292
|
-
durableObjects: props.bindings.durable_objects || { bindings: [] },
|
|
293
|
-
firstPartyWorkerDevFacade: props.firstPartyWorker,
|
|
294
|
-
local: props.local,
|
|
295
|
-
// Enable the bundling to know whether we are using dev or publish
|
|
296
|
-
targetConsumer: "dev",
|
|
297
|
-
testScheduled: props.testScheduled ?? false,
|
|
298
|
-
experimentalLocal: props.experimentalLocal,
|
|
299
|
-
});
|
|
300
|
-
|
|
301
|
-
// TODO(queues) support remote wrangler dev
|
|
302
|
-
if (
|
|
303
|
-
!props.local &&
|
|
304
|
-
(props.bindings.queues?.length || props.queueConsumers?.length)
|
|
305
|
-
) {
|
|
306
|
-
logger.warn(
|
|
307
|
-
"Queues are currently in Beta and are not supported in wrangler dev remote mode."
|
|
308
|
-
);
|
|
309
|
-
}
|
|
310
|
-
|
|
311
|
-
const announceAndOnReady: typeof props.onReady = (finalIp, finalPort) => {
|
|
312
|
-
if (process.send) {
|
|
313
|
-
process.send(
|
|
314
|
-
JSON.stringify({
|
|
315
|
-
event: "DEV_SERVER_READY",
|
|
316
|
-
ip: finalIp,
|
|
317
|
-
port: finalPort,
|
|
318
|
-
})
|
|
319
|
-
);
|
|
320
|
-
}
|
|
321
|
-
|
|
322
|
-
if (props.onReady) {
|
|
323
|
-
props.onReady(finalIp, finalPort);
|
|
324
|
-
}
|
|
325
|
-
};
|
|
326
|
-
|
|
327
|
-
return props.local ? (
|
|
328
|
-
<Local
|
|
329
|
-
name={props.name}
|
|
330
|
-
bundle={bundle}
|
|
331
|
-
format={props.entry.format}
|
|
332
|
-
compatibilityDate={props.compatibilityDate}
|
|
333
|
-
compatibilityFlags={props.compatibilityFlags}
|
|
334
|
-
usageModel={props.usageModel}
|
|
335
|
-
bindings={props.bindings}
|
|
336
|
-
workerDefinitions={workerDefinitions}
|
|
337
|
-
assetPaths={props.assetPaths}
|
|
338
|
-
initialPort={props.initialPort}
|
|
339
|
-
initialIp={props.initialIp}
|
|
340
|
-
rules={props.rules}
|
|
341
|
-
inspectorPort={props.inspectorPort}
|
|
342
|
-
localPersistencePath={props.localPersistencePath}
|
|
343
|
-
liveReload={props.liveReload}
|
|
344
|
-
crons={props.crons}
|
|
345
|
-
queueConsumers={props.queueConsumers}
|
|
346
|
-
localProtocol={props.localProtocol}
|
|
347
|
-
localUpstream={props.localUpstream}
|
|
348
|
-
inspect={props.inspect}
|
|
349
|
-
onReady={announceAndOnReady}
|
|
350
|
-
enablePagesAssetsServiceBinding={props.enablePagesAssetsServiceBinding}
|
|
351
|
-
experimentalLocal={props.experimentalLocal}
|
|
352
|
-
accountId={props.accountId}
|
|
353
|
-
experimentalLocalRemoteKv={props.experimentalLocalRemoteKv}
|
|
354
|
-
sourceMapPath={bundle?.sourceMapPath}
|
|
355
|
-
/>
|
|
356
|
-
) : (
|
|
357
|
-
<Remote
|
|
358
|
-
name={props.name}
|
|
359
|
-
bundle={bundle}
|
|
360
|
-
format={props.entry.format}
|
|
361
|
-
accountId={props.accountId}
|
|
362
|
-
bindings={props.bindings}
|
|
363
|
-
assetPaths={props.assetPaths}
|
|
364
|
-
isWorkersSite={props.isWorkersSite}
|
|
365
|
-
port={props.initialPort}
|
|
366
|
-
ip={props.initialIp}
|
|
367
|
-
localProtocol={props.localProtocol}
|
|
368
|
-
inspectorPort={props.inspectorPort}
|
|
369
|
-
// TODO: @threepointone #1167
|
|
370
|
-
// liveReload={props.liveReload}
|
|
371
|
-
inspect={props.inspect}
|
|
372
|
-
compatibilityDate={props.compatibilityDate}
|
|
373
|
-
compatibilityFlags={props.compatibilityFlags}
|
|
374
|
-
usageModel={props.usageModel}
|
|
375
|
-
env={props.env}
|
|
376
|
-
legacyEnv={props.legacyEnv}
|
|
377
|
-
zone={props.zone}
|
|
378
|
-
host={props.host}
|
|
379
|
-
routes={props.routes}
|
|
380
|
-
onReady={announceAndOnReady}
|
|
381
|
-
sourceMapPath={bundle?.sourceMapPath}
|
|
382
|
-
sendMetrics={props.sendMetrics}
|
|
383
|
-
/>
|
|
384
|
-
);
|
|
385
|
-
}
|
|
386
|
-
|
|
387
|
-
export interface DirectorySyncResult {
|
|
388
|
-
name: string;
|
|
389
|
-
removeCallback: () => void;
|
|
390
|
-
}
|
|
391
|
-
|
|
392
|
-
function useTmpDir(): string | undefined {
|
|
393
|
-
const [directory, setDirectory] = useState<DirectorySyncResult>();
|
|
394
|
-
const handleError = useErrorHandler();
|
|
395
|
-
useEffect(() => {
|
|
396
|
-
let dir: DirectorySyncResult | undefined;
|
|
397
|
-
try {
|
|
398
|
-
dir = tmp.dirSync({ unsafeCleanup: true });
|
|
399
|
-
setDirectory(dir);
|
|
400
|
-
return;
|
|
401
|
-
} catch (err) {
|
|
402
|
-
logger.error(
|
|
403
|
-
"Failed to create temporary directory to store built files."
|
|
404
|
-
);
|
|
405
|
-
handleError(err);
|
|
406
|
-
}
|
|
407
|
-
return () => {
|
|
408
|
-
dir?.removeCallback();
|
|
409
|
-
};
|
|
410
|
-
}, [handleError]);
|
|
411
|
-
return directory?.name;
|
|
412
|
-
}
|
|
413
|
-
|
|
414
|
-
function useCustomBuild(expectedEntry: Entry, build: Config["build"]): void {
|
|
415
|
-
useEffect(() => {
|
|
416
|
-
if (!build.command) return;
|
|
417
|
-
let watcher: ReturnType<typeof watch> | undefined;
|
|
418
|
-
if (build.watch_dir) {
|
|
419
|
-
watcher = watch(build.watch_dir, {
|
|
420
|
-
persistent: true,
|
|
421
|
-
ignoreInitial: true,
|
|
422
|
-
}).on("all", (_event, filePath) => {
|
|
423
|
-
const relativeFile =
|
|
424
|
-
path.relative(expectedEntry.directory, expectedEntry.file) || ".";
|
|
425
|
-
//TODO: we should buffer requests to the proxy until this completes
|
|
426
|
-
logger.log(`The file ${filePath} changed, restarting build...`);
|
|
427
|
-
runCustomBuild(expectedEntry.file, relativeFile, build).catch((err) => {
|
|
428
|
-
logger.error("Custom build failed:", err);
|
|
429
|
-
});
|
|
430
|
-
});
|
|
431
|
-
}
|
|
432
|
-
|
|
433
|
-
return () => {
|
|
434
|
-
void watcher?.close();
|
|
435
|
-
};
|
|
436
|
-
}, [build, expectedEntry]);
|
|
437
|
-
}
|
|
438
|
-
|
|
439
|
-
function sleep(period: number) {
|
|
440
|
-
return new Promise((resolve) => setTimeout(resolve, period));
|
|
441
|
-
}
|
|
442
|
-
const SLEEP_DURATION = 2000;
|
|
443
|
-
// really need a first class api for this
|
|
444
|
-
const hostNameRegex = /userHostname="(.*)"/g;
|
|
445
|
-
async function findTunnelHostname() {
|
|
446
|
-
let hostName: string | undefined;
|
|
447
|
-
while (!hostName) {
|
|
448
|
-
try {
|
|
449
|
-
const resp = await fetch("http://localhost:8789/metrics");
|
|
450
|
-
const data = await resp.text();
|
|
451
|
-
const matches = Array.from(data.matchAll(hostNameRegex));
|
|
452
|
-
hostName = matches[0][1];
|
|
453
|
-
} catch (err) {
|
|
454
|
-
await sleep(SLEEP_DURATION);
|
|
455
|
-
}
|
|
456
|
-
}
|
|
457
|
-
return hostName;
|
|
458
|
-
}
|
|
459
|
-
|
|
460
|
-
/**
|
|
461
|
-
* Create a tunnel to the remote worker.
|
|
462
|
-
* We've disabled this for now until we figure out a better user experience.
|
|
463
|
-
*/
|
|
464
|
-
function useTunnel(toggle: boolean) {
|
|
465
|
-
const tunnel = useRef<ReturnType<typeof spawn>>();
|
|
466
|
-
const removeSignalExitListener = useRef<() => void>();
|
|
467
|
-
// TODO: test if cloudflared is available, if not
|
|
468
|
-
// point them to a url where they can get docs to install it
|
|
469
|
-
useEffect(() => {
|
|
470
|
-
async function startTunnel() {
|
|
471
|
-
if (toggle) {
|
|
472
|
-
try {
|
|
473
|
-
await commandExists("cloudflared");
|
|
474
|
-
} catch (e) {
|
|
475
|
-
logger.warn(
|
|
476
|
-
"To share your worker on the Internet, please install `cloudflared` from https://developers.cloudflare.com/cloudflare-one/connections/connect-apps/install-and-setup/installation"
|
|
477
|
-
);
|
|
478
|
-
return;
|
|
479
|
-
}
|
|
480
|
-
logger.log("⎔ Starting a tunnel...");
|
|
481
|
-
tunnel.current = spawn("cloudflared", [
|
|
482
|
-
"tunnel",
|
|
483
|
-
"--url",
|
|
484
|
-
"http://localhost:8787",
|
|
485
|
-
"--metrics",
|
|
486
|
-
"localhost:8789",
|
|
487
|
-
]);
|
|
488
|
-
|
|
489
|
-
tunnel.current.on("close", (code) => {
|
|
490
|
-
if (code) {
|
|
491
|
-
logger.log(`Tunnel process exited with code ${code}`);
|
|
492
|
-
}
|
|
493
|
-
});
|
|
494
|
-
|
|
495
|
-
removeSignalExitListener.current = onExit((_code, _signal) => {
|
|
496
|
-
logger.log("⎔ Shutting down local tunnel.");
|
|
497
|
-
tunnel.current?.kill();
|
|
498
|
-
tunnel.current = undefined;
|
|
499
|
-
});
|
|
500
|
-
|
|
501
|
-
const hostName = await findTunnelHostname();
|
|
502
|
-
await clipboardy.write(hostName);
|
|
503
|
-
logger.log(`⬣ Sharing at ${hostName}, copied to clipboard.`);
|
|
504
|
-
}
|
|
505
|
-
}
|
|
506
|
-
|
|
507
|
-
startTunnel().catch(async (err) => {
|
|
508
|
-
logger.error("tunnel:", err);
|
|
509
|
-
});
|
|
510
|
-
|
|
511
|
-
return () => {
|
|
512
|
-
if (tunnel.current) {
|
|
513
|
-
logger.log("⎔ Shutting down tunnel.");
|
|
514
|
-
tunnel.current?.kill();
|
|
515
|
-
tunnel.current = undefined;
|
|
516
|
-
removeSignalExitListener.current && removeSignalExitListener.current();
|
|
517
|
-
removeSignalExitListener.current = undefined;
|
|
518
|
-
}
|
|
519
|
-
};
|
|
520
|
-
}, [toggle]);
|
|
521
|
-
}
|
|
522
|
-
|
|
523
|
-
type useHotkeysInitialState = {
|
|
524
|
-
local: boolean;
|
|
525
|
-
tunnel: boolean;
|
|
526
|
-
};
|
|
527
|
-
function useHotkeys(props: {
|
|
528
|
-
initial: useHotkeysInitialState;
|
|
529
|
-
inspectorPort: number;
|
|
530
|
-
inspect: boolean;
|
|
531
|
-
localProtocol: "http" | "https";
|
|
532
|
-
forceLocal: boolean | undefined;
|
|
533
|
-
worker: string | undefined;
|
|
534
|
-
}) {
|
|
535
|
-
const { initial, inspectorPort, inspect, localProtocol, forceLocal } = props;
|
|
536
|
-
// UGH, we should put port in context instead
|
|
537
|
-
const [toggles, setToggles] = useState(initial);
|
|
538
|
-
const { exit } = useApp();
|
|
539
|
-
|
|
540
|
-
useInput(
|
|
541
|
-
async (
|
|
542
|
-
input,
|
|
543
|
-
// eslint-disable-next-line unused-imports/no-unused-vars
|
|
544
|
-
key
|
|
545
|
-
) => {
|
|
546
|
-
switch (input.toLowerCase()) {
|
|
547
|
-
// clear console
|
|
548
|
-
case "c":
|
|
549
|
-
console.clear();
|
|
550
|
-
// This console.log causes Ink to re-render the `DevSession` component.
|
|
551
|
-
// Couldn't find a better way to tell it to do so...
|
|
552
|
-
console.log();
|
|
553
|
-
break;
|
|
554
|
-
// open browser
|
|
555
|
-
case "b": {
|
|
556
|
-
if (ip === "0.0.0.0") {
|
|
557
|
-
await openInBrowser(`${localProtocol}://127.0.0.1:${port}`);
|
|
558
|
-
return;
|
|
559
|
-
}
|
|
560
|
-
await openInBrowser(`${localProtocol}://${ip}:${port}`);
|
|
561
|
-
break;
|
|
562
|
-
}
|
|
563
|
-
// toggle inspector
|
|
564
|
-
case "d": {
|
|
565
|
-
if (inspect) {
|
|
566
|
-
await openInspector(inspectorPort, props.worker);
|
|
567
|
-
}
|
|
568
|
-
break;
|
|
569
|
-
}
|
|
570
|
-
// toggle local
|
|
571
|
-
case "l":
|
|
572
|
-
if (forceLocal) return;
|
|
573
|
-
setToggles((previousToggles) => ({
|
|
574
|
-
...previousToggles,
|
|
575
|
-
local: !previousToggles.local,
|
|
576
|
-
}));
|
|
577
|
-
break;
|
|
578
|
-
// shut down
|
|
579
|
-
case "q":
|
|
580
|
-
case "x":
|
|
581
|
-
exit();
|
|
582
|
-
break;
|
|
583
|
-
default:
|
|
584
|
-
// nothing?
|
|
585
|
-
break;
|
|
586
|
-
}
|
|
587
|
-
}
|
|
588
|
-
);
|
|
589
|
-
return toggles;
|
|
590
|
-
}
|
|
591
|
-
|
|
592
|
-
function ErrorFallback(props: { error: Error }) {
|
|
593
|
-
const { exit } = useApp();
|
|
594
|
-
useEffect(() => exit(props.error));
|
|
595
|
-
return (
|
|
596
|
-
<>
|
|
597
|
-
<Text>Something went wrong:</Text>
|
|
598
|
-
<Text>{props.error.stack}</Text>
|
|
599
|
-
</>
|
|
600
|
-
);
|
|
601
|
-
}
|
|
602
|
-
|
|
603
|
-
export default withErrorBoundary(DevImplementation, {
|
|
604
|
-
FallbackComponent: ErrorFallback,
|
|
605
|
-
});
|
|
@@ -1,31 +0,0 @@
|
|
|
1
|
-
import path from "node:path";
|
|
2
|
-
|
|
3
|
-
export function getLocalPersistencePath(
|
|
4
|
-
persistTo: string | undefined,
|
|
5
|
-
doPersist: true,
|
|
6
|
-
configPath: string | undefined
|
|
7
|
-
): string;
|
|
8
|
-
|
|
9
|
-
export function getLocalPersistencePath(
|
|
10
|
-
persistTo: string | undefined,
|
|
11
|
-
doPersist: boolean,
|
|
12
|
-
configPath: string | undefined
|
|
13
|
-
): string | null;
|
|
14
|
-
|
|
15
|
-
export function getLocalPersistencePath(
|
|
16
|
-
persistTo: string | undefined,
|
|
17
|
-
doPersist: boolean,
|
|
18
|
-
configPath: string | undefined
|
|
19
|
-
) {
|
|
20
|
-
return persistTo
|
|
21
|
-
? // If path specified, always treat it as relative to cwd()
|
|
22
|
-
path.resolve(process.cwd(), persistTo)
|
|
23
|
-
: doPersist
|
|
24
|
-
? // If just flagged on, treat it as relative to wrangler.toml,
|
|
25
|
-
// if one can be found, otherwise cwd()
|
|
26
|
-
path.resolve(
|
|
27
|
-
configPath ? path.dirname(configPath) : process.cwd(),
|
|
28
|
-
".wrangler/state"
|
|
29
|
-
)
|
|
30
|
-
: null;
|
|
31
|
-
}
|