Huge News!Announcing our $40M Series B led by Abstract Ventures.Learn More
Socket
Sign inDemoInstall
Socket

@eventual/core-runtime

Package Overview
Dependencies
Maintainers
2
Versions
100
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

@eventual/core-runtime - npm Package Compare versions

Comparing version 0.25.0 to 0.25.1

70

lib/cjs/handlers/command-worker.js

@@ -81,38 +81,40 @@ "use strict";

const shouldValidate = command.validate !== false;
// RPC route takes a POST request and passes the parsed JSON body as input to the input
router.post((0, core_1.commandRpcPath)(command), withMiddleware(async (request, context) => {
if (command.passThrough) {
// if passthrough is enabled, just proxy the request-response to the handler
return command.handler(request, context);
}
let input = await request.tryJson();
if (command.input && shouldValidate) {
try {
input = command.input.parse(input);
if (!command.passThrough) {
// RPC route takes a POST request and passes the parsed JSON body as input to the input
router.post((0, core_1.commandRpcPath)(command), withMiddleware(async (request, context) => {
if (command.passThrough) {
// if passthrough is enabled, just proxy the request-response to the handler
return command.handler(request, context);
}
catch (err) {
console.error("Invalid input", err, input);
return new core_1.HttpResponse(JSON.stringify(err), {
status: 400,
statusText: "Invalid input",
});
let input = await request.tryJson();
if (command.input && shouldValidate) {
try {
input = command.input.parse(input);
}
catch (err) {
console.error("Invalid input", err, input);
return new core_1.HttpResponse(JSON.stringify(err), {
status: 400,
statusText: "Invalid input",
});
}
}
}
let output = await command.handler(input, context);
if (command.output && shouldValidate) {
try {
output = command.output.parse(output);
let output = await command.handler(input, context);
if (command.output && shouldValidate) {
try {
output = command.output.parse(output);
}
catch (err) {
console.error("RPC output did not match schema", output, err);
return new core_1.HttpResponse(JSON.stringify(err), {
status: 500,
statusText: "RPC output did not match schema",
});
}
}
catch (err) {
console.error("RPC output did not match schema", output, err);
return new core_1.HttpResponse(JSON.stringify(err), {
status: 500,
statusText: "RPC output did not match schema",
});
}
}
return new core_1.HttpResponse(JSON.stringify(output, jsonReplacer), {
status: 200,
});
}));
return new core_1.HttpResponse(JSON.stringify(output, jsonReplacer), {
status: 200,
});
}));
}
const path = command.path;

@@ -226,2 +228,2 @@ if (path) {

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"command-worker.js","sourceRoot":"","sources":["../../../src/handlers/command-worker.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  EventualServiceClient,\n  isHttpError,\n  HttpRequest,\n  HttpResponse,\n  RestParamSpec,\n  commandRpcPath,\n} from \"@eventual/core\";\nimport {\n  registerServiceClient,\n  serviceTypeScope,\n  ServiceType,\n  commands,\n} from \"@eventual/core/internal\";\nimport itty from \"itty-router\";\n\nexport interface ApiHandlerDependencies {\n  serviceClient?: EventualServiceClient;\n}\n\nexport interface CommandWorker {\n  (request: HttpRequest): Promise<HttpResponse>;\n}\n\n/**\n * Creates a generic function for handling inbound API requests\n * that can be used in runtime implementations. This implementation is\n * decoupled from a runtime's specifics by the clients. A runtime must\n * inject its own client implementations designed for that platform.\n */\nexport function createCommandWorker({\n  serviceClient,\n}: ApiHandlerDependencies): CommandWorker {\n  // make the service client available to web hooks\n  if (serviceClient) {\n    registerServiceClient(serviceClient);\n  }\n\n  const router = initRouter();\n\n  /**\n   * Handle inbound webhook API requests.\n   *\n   * Each webhook registers routes on the central {@link router} which\n   * then handles the request.\n   */\n  return function (request) {\n    console.log(\"request\", request);\n    return serviceTypeScope(ServiceType.CommandWorker, async () => {\n      try {\n        const response = await router.handle(request);\n        if (response === undefined) {\n          if (request.method === \"OPTIONS\") {\n            return new HttpResponse(undefined, {\n              // CORS expects a 204 or 200, using 204 to match API Gateway\n              // and accurately reflect NO CONTENT\n              status: 204,\n            });\n          }\n          return new HttpResponse(\n            `Not Found: ${request.method} ${request.url}`,\n            {\n              status: 404,\n              statusText: \"Not Found\",\n            }\n          );\n        }\n        return response;\n      } catch (err) {\n        if (isHttpError(err)) {\n          return new HttpResponse(\n            JSON.stringify({\n              message: err.message,\n              data: err.data,\n            }),\n            {\n              status: err.code,\n            }\n          );\n        } else if (err instanceof Error) {\n          console.error(err);\n          return new HttpResponse(err.message, {\n            status: 500,\n            statusText: \"Internal Server Error\",\n          });\n        } else {\n          return new HttpResponse(\"Internal Server Error\", {\n            status: 500,\n          });\n        }\n      }\n    });\n  };\n}\n\nfunction initRouter() {\n  const router: Router = itty.Router<HttpRequest, Router>({\n    // paths always start with slash, the router will remove double slashes\n    base: \"/\",\n  });\n\n  for (const command of commands) {\n    const shouldValidate = command.validate !== false;\n\n    // RPC route takes a POST request and passes the parsed JSON body as input to the input\n    router.post(\n      commandRpcPath(command),\n      withMiddleware(async (request, context) => {\n        if (command.passThrough) {\n          // if passthrough is enabled, just proxy the request-response to the handler\n          return command.handler(request, context);\n        }\n\n        let input: any = await request.tryJson();\n        if (command.input && shouldValidate) {\n          try {\n            input = command.input.parse(input);\n          } catch (err) {\n            console.error(\"Invalid input\", err, input);\n            return new HttpResponse(JSON.stringify(err), {\n              status: 400,\n              statusText: \"Invalid input\",\n            });\n          }\n        }\n\n        let output: any = await command.handler(input, context);\n        if (command.output && shouldValidate) {\n          try {\n            output = command.output.parse(output);\n          } catch (err) {\n            console.error(\"RPC output did not match schema\", output, err);\n            return new HttpResponse(JSON.stringify(err), {\n              status: 500,\n              statusText: \"RPC output did not match schema\",\n            });\n          }\n        }\n        return new HttpResponse(JSON.stringify(output, jsonReplacer), {\n          status: 200,\n        });\n      })\n    );\n\n    const path = command.path;\n\n    if (path) {\n      const method = (command.method?.toLocaleLowerCase() ??\n        \"all\") as keyof Router;\n\n      // REST routes parse the request according to the command's path/method/params configuration\n      router[method](\n        path,\n        withMiddleware(async (request: HttpRequest, context) => {\n          if (command.passThrough) {\n            // if passthrough is enabled, just proxy the request-response to the handler\n            return command.handler(request, context);\n          }\n\n          // first, get the body as pure JSON - assume it's an object\n          const body = await request.tryJson();\n          let input: any = {\n            ...request.params,\n            ...(body && typeof body === \"object\" ? body : {}),\n          };\n\n          // parse headers/params/queries/body into the RPC interface\n          if (command.params) {\n            Object.entries(command.params as Record<string, RestParamSpec>).map(\n              ([name, spec]) => {\n                input[name] = resolveInput(name, spec);\n              }\n            );\n          }\n\n          if (command.input && shouldValidate) {\n            // validate the zod input schema if one is specified\n            input = command.input.parse(input);\n          }\n\n          // call the command RPC handler\n          let output: any = await command.handler(input, context);\n\n          if (command.output && shouldValidate) {\n            // validate the output of the command handler against the schema if it's defined\n            output = command.output.parse(output);\n          }\n\n          // TODO: support mapping RPC output back to HTTP properties such as Headers\n          // TODO: support alternative status code https://github.com/functionless/eventual/issues/276\n\n          return new HttpResponse(JSON.stringify(output, jsonReplacer), {\n            status: 200,\n            headers: {\n              \"Content-Type\": \"application/json\",\n            },\n          });\n\n          function resolveInput(name: string, spec: RestParamSpec): any {\n            if (spec === \"body\") {\n              return body?.[name];\n            } else if (spec === \"query\") {\n              return request.query?.[name];\n            } else if (spec === \"header\") {\n              return request.headers.get(name);\n            } else if (spec === \"path\") {\n              return request.params?.[name];\n            } else {\n              return resolveInput(spec.name ?? name, spec.in);\n            }\n          }\n        })\n      );\n    }\n\n    /**\n     * Applies the chain of middleware callbacks to the request to build up\n     * context and pass it through the chain and finally to the handler.\n     *\n     * Each context can add to or completely replace the context. They can\n     * also break the chain at any time by returning a HttpResponse instead\n     * of calling `next`.\n     *\n     * @param handler\n     * @returns\n     */\n    function withMiddleware(\n      handler: (request: HttpRequest, context: any) => Promise<HttpResponse>\n    ) {\n      return async (request: HttpRequest): Promise<HttpResponse> => {\n        const chain = (command.middlewares ?? []).values();\n\n        return next(request, {});\n\n        async function next(\n          request: HttpRequest,\n          context: any\n        ): Promise<HttpResponse> {\n          let consumed = false;\n          const middleware = chain.next();\n          if (middleware.done) {\n            return handler(request, context);\n          } else {\n            return middleware.value({\n              request,\n              context,\n              next: async (context) => {\n                if (consumed) {\n                  consumed = true;\n                  throw new Error(\n                    `Middleware cannot call 'next' more than once`\n                  );\n                }\n                return next(request, context);\n              },\n            });\n          }\n        }\n      };\n    }\n  }\n\n  return router;\n}\n\ninterface Router {\n  handle: (request: HttpRequest, ...extra: any) => Promise<HttpResponse>;\n  get: RouteFactory;\n  head: RouteFactory;\n  post: RouteFactory;\n  put: RouteFactory;\n  delete: RouteFactory;\n  connect: RouteFactory;\n  options: RouteFactory;\n  trace: RouteFactory;\n  patch: RouteFactory;\n}\n\ninterface RouteFactory {\n  (path: string, handlers: RouteHandler): Router;\n}\n\ntype RouteHandler = (\n  request: HttpRequest,\n  ...args: any\n) => HttpResponse | Promise<HttpResponse>;\n\n/**\n * Implements JSON serialization for well known types.\n */\nfunction jsonReplacer(_key: string, value: any) {\n  if (value instanceof Date) {\n    return value.toISOString();\n  }\n  return value;\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"command-worker.js","sourceRoot":"","sources":["../../../src/handlers/command-worker.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  EventualServiceClient,\n  isHttpError,\n  HttpRequest,\n  HttpResponse,\n  RestParamSpec,\n  commandRpcPath,\n} from \"@eventual/core\";\nimport {\n  registerServiceClient,\n  serviceTypeScope,\n  ServiceType,\n  commands,\n} from \"@eventual/core/internal\";\nimport itty from \"itty-router\";\n\nexport interface ApiHandlerDependencies {\n  serviceClient?: EventualServiceClient;\n}\n\nexport interface CommandWorker {\n  (request: HttpRequest): Promise<HttpResponse>;\n}\n\n/**\n * Creates a generic function for handling inbound API requests\n * that can be used in runtime implementations. This implementation is\n * decoupled from a runtime's specifics by the clients. A runtime must\n * inject its own client implementations designed for that platform.\n */\nexport function createCommandWorker({\n  serviceClient,\n}: ApiHandlerDependencies): CommandWorker {\n  // make the service client available to web hooks\n  if (serviceClient) {\n    registerServiceClient(serviceClient);\n  }\n\n  const router = initRouter();\n\n  /**\n   * Handle inbound webhook API requests.\n   *\n   * Each webhook registers routes on the central {@link router} which\n   * then handles the request.\n   */\n  return function (request) {\n    console.log(\"request\", request);\n    return serviceTypeScope(ServiceType.CommandWorker, async () => {\n      try {\n        const response = await router.handle(request);\n        if (response === undefined) {\n          if (request.method === \"OPTIONS\") {\n            return new HttpResponse(undefined, {\n              // CORS expects a 204 or 200, using 204 to match API Gateway\n              // and accurately reflect NO CONTENT\n              status: 204,\n            });\n          }\n          return new HttpResponse(\n            `Not Found: ${request.method} ${request.url}`,\n            {\n              status: 404,\n              statusText: \"Not Found\",\n            }\n          );\n        }\n        return response;\n      } catch (err) {\n        if (isHttpError(err)) {\n          return new HttpResponse(\n            JSON.stringify({\n              message: err.message,\n              data: err.data,\n            }),\n            {\n              status: err.code,\n            }\n          );\n        } else if (err instanceof Error) {\n          console.error(err);\n          return new HttpResponse(err.message, {\n            status: 500,\n            statusText: \"Internal Server Error\",\n          });\n        } else {\n          return new HttpResponse(\"Internal Server Error\", {\n            status: 500,\n          });\n        }\n      }\n    });\n  };\n}\n\nfunction initRouter() {\n  const router: Router = itty.Router<HttpRequest, Router>({\n    // paths always start with slash, the router will remove double slashes\n    base: \"/\",\n  });\n\n  for (const command of commands) {\n    const shouldValidate = command.validate !== false;\n\n    if (!command.passThrough) {\n      // RPC route takes a POST request and passes the parsed JSON body as input to the input\n      router.post(\n        commandRpcPath(command),\n        withMiddleware(async (request, context) => {\n          if (command.passThrough) {\n            // if passthrough is enabled, just proxy the request-response to the handler\n            return command.handler(request, context);\n          }\n\n          let input: any = await request.tryJson();\n          if (command.input && shouldValidate) {\n            try {\n              input = command.input.parse(input);\n            } catch (err) {\n              console.error(\"Invalid input\", err, input);\n              return new HttpResponse(JSON.stringify(err), {\n                status: 400,\n                statusText: \"Invalid input\",\n              });\n            }\n          }\n\n          let output: any = await command.handler(input, context);\n          if (command.output && shouldValidate) {\n            try {\n              output = command.output.parse(output);\n            } catch (err) {\n              console.error(\"RPC output did not match schema\", output, err);\n              return new HttpResponse(JSON.stringify(err), {\n                status: 500,\n                statusText: \"RPC output did not match schema\",\n              });\n            }\n          }\n          return new HttpResponse(JSON.stringify(output, jsonReplacer), {\n            status: 200,\n          });\n        })\n      );\n    }\n\n    const path = command.path;\n\n    if (path) {\n      const method = (command.method?.toLocaleLowerCase() ??\n        \"all\") as keyof Router;\n\n      // REST routes parse the request according to the command's path/method/params configuration\n      router[method](\n        path,\n        withMiddleware(async (request: HttpRequest, context) => {\n          if (command.passThrough) {\n            // if passthrough is enabled, just proxy the request-response to the handler\n            return command.handler(request, context);\n          }\n\n          // first, get the body as pure JSON - assume it's an object\n          const body = await request.tryJson();\n          let input: any = {\n            ...request.params,\n            ...(body && typeof body === \"object\" ? body : {}),\n          };\n\n          // parse headers/params/queries/body into the RPC interface\n          if (command.params) {\n            Object.entries(command.params as Record<string, RestParamSpec>).map(\n              ([name, spec]) => {\n                input[name] = resolveInput(name, spec);\n              }\n            );\n          }\n\n          if (command.input && shouldValidate) {\n            // validate the zod input schema if one is specified\n            input = command.input.parse(input);\n          }\n\n          // call the command RPC handler\n          let output: any = await command.handler(input, context);\n\n          if (command.output && shouldValidate) {\n            // validate the output of the command handler against the schema if it's defined\n            output = command.output.parse(output);\n          }\n\n          // TODO: support mapping RPC output back to HTTP properties such as Headers\n          // TODO: support alternative status code https://github.com/functionless/eventual/issues/276\n\n          return new HttpResponse(JSON.stringify(output, jsonReplacer), {\n            status: 200,\n            headers: {\n              \"Content-Type\": \"application/json\",\n            },\n          });\n\n          function resolveInput(name: string, spec: RestParamSpec): any {\n            if (spec === \"body\") {\n              return body?.[name];\n            } else if (spec === \"query\") {\n              return request.query?.[name];\n            } else if (spec === \"header\") {\n              return request.headers.get(name);\n            } else if (spec === \"path\") {\n              return request.params?.[name];\n            } else {\n              return resolveInput(spec.name ?? name, spec.in);\n            }\n          }\n        })\n      );\n    }\n\n    /**\n     * Applies the chain of middleware callbacks to the request to build up\n     * context and pass it through the chain and finally to the handler.\n     *\n     * Each context can add to or completely replace the context. They can\n     * also break the chain at any time by returning a HttpResponse instead\n     * of calling `next`.\n     *\n     * @param handler\n     * @returns\n     */\n    function withMiddleware(\n      handler: (request: HttpRequest, context: any) => Promise<HttpResponse>\n    ) {\n      return async (request: HttpRequest): Promise<HttpResponse> => {\n        const chain = (command.middlewares ?? []).values();\n\n        return next(request, {});\n\n        async function next(\n          request: HttpRequest,\n          context: any\n        ): Promise<HttpResponse> {\n          let consumed = false;\n          const middleware = chain.next();\n          if (middleware.done) {\n            return handler(request, context);\n          } else {\n            return middleware.value({\n              request,\n              context,\n              next: async (context) => {\n                if (consumed) {\n                  consumed = true;\n                  throw new Error(\n                    `Middleware cannot call 'next' more than once`\n                  );\n                }\n                return next(request, context);\n              },\n            });\n          }\n        }\n      };\n    }\n  }\n\n  return router;\n}\n\ninterface Router {\n  handle: (request: HttpRequest, ...extra: any) => Promise<HttpResponse>;\n  get: RouteFactory;\n  head: RouteFactory;\n  post: RouteFactory;\n  put: RouteFactory;\n  delete: RouteFactory;\n  connect: RouteFactory;\n  options: RouteFactory;\n  trace: RouteFactory;\n  patch: RouteFactory;\n}\n\ninterface RouteFactory {\n  (path: string, handlers: RouteHandler): Router;\n}\n\ntype RouteHandler = (\n  request: HttpRequest,\n  ...args: any\n) => HttpResponse | Promise<HttpResponse>;\n\n/**\n * Implements JSON serialization for well known types.\n */\nfunction jsonReplacer(_key: string, value: any) {\n  if (value instanceof Date) {\n    return value.toISOString();\n  }\n  return value;\n}\n"]}

@@ -74,38 +74,40 @@ import { isHttpError, HttpResponse, commandRpcPath, } from "@eventual/core";

const shouldValidate = command.validate !== false;
// RPC route takes a POST request and passes the parsed JSON body as input to the input
router.post(commandRpcPath(command), withMiddleware(async (request, context) => {
if (command.passThrough) {
// if passthrough is enabled, just proxy the request-response to the handler
return command.handler(request, context);
}
let input = await request.tryJson();
if (command.input && shouldValidate) {
try {
input = command.input.parse(input);
if (!command.passThrough) {
// RPC route takes a POST request and passes the parsed JSON body as input to the input
router.post(commandRpcPath(command), withMiddleware(async (request, context) => {
if (command.passThrough) {
// if passthrough is enabled, just proxy the request-response to the handler
return command.handler(request, context);
}
catch (err) {
console.error("Invalid input", err, input);
return new HttpResponse(JSON.stringify(err), {
status: 400,
statusText: "Invalid input",
});
let input = await request.tryJson();
if (command.input && shouldValidate) {
try {
input = command.input.parse(input);
}
catch (err) {
console.error("Invalid input", err, input);
return new HttpResponse(JSON.stringify(err), {
status: 400,
statusText: "Invalid input",
});
}
}
}
let output = await command.handler(input, context);
if (command.output && shouldValidate) {
try {
output = command.output.parse(output);
let output = await command.handler(input, context);
if (command.output && shouldValidate) {
try {
output = command.output.parse(output);
}
catch (err) {
console.error("RPC output did not match schema", output, err);
return new HttpResponse(JSON.stringify(err), {
status: 500,
statusText: "RPC output did not match schema",
});
}
}
catch (err) {
console.error("RPC output did not match schema", output, err);
return new HttpResponse(JSON.stringify(err), {
status: 500,
statusText: "RPC output did not match schema",
});
}
}
return new HttpResponse(JSON.stringify(output, jsonReplacer), {
status: 200,
});
}));
return new HttpResponse(JSON.stringify(output, jsonReplacer), {
status: 200,
});
}));
}
const path = command.path;

@@ -219,2 +221,2 @@ if (path) {

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"command-worker.js","sourceRoot":"","sources":["../../../src/handlers/command-worker.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  EventualServiceClient,\n  isHttpError,\n  HttpRequest,\n  HttpResponse,\n  RestParamSpec,\n  commandRpcPath,\n} from \"@eventual/core\";\nimport {\n  registerServiceClient,\n  serviceTypeScope,\n  ServiceType,\n  commands,\n} from \"@eventual/core/internal\";\nimport itty from \"itty-router\";\n\nexport interface ApiHandlerDependencies {\n  serviceClient?: EventualServiceClient;\n}\n\nexport interface CommandWorker {\n  (request: HttpRequest): Promise<HttpResponse>;\n}\n\n/**\n * Creates a generic function for handling inbound API requests\n * that can be used in runtime implementations. This implementation is\n * decoupled from a runtime's specifics by the clients. A runtime must\n * inject its own client implementations designed for that platform.\n */\nexport function createCommandWorker({\n  serviceClient,\n}: ApiHandlerDependencies): CommandWorker {\n  // make the service client available to web hooks\n  if (serviceClient) {\n    registerServiceClient(serviceClient);\n  }\n\n  const router = initRouter();\n\n  /**\n   * Handle inbound webhook API requests.\n   *\n   * Each webhook registers routes on the central {@link router} which\n   * then handles the request.\n   */\n  return function (request) {\n    console.log(\"request\", request);\n    return serviceTypeScope(ServiceType.CommandWorker, async () => {\n      try {\n        const response = await router.handle(request);\n        if (response === undefined) {\n          if (request.method === \"OPTIONS\") {\n            return new HttpResponse(undefined, {\n              // CORS expects a 204 or 200, using 204 to match API Gateway\n              // and accurately reflect NO CONTENT\n              status: 204,\n            });\n          }\n          return new HttpResponse(\n            `Not Found: ${request.method} ${request.url}`,\n            {\n              status: 404,\n              statusText: \"Not Found\",\n            }\n          );\n        }\n        return response;\n      } catch (err) {\n        if (isHttpError(err)) {\n          return new HttpResponse(\n            JSON.stringify({\n              message: err.message,\n              data: err.data,\n            }),\n            {\n              status: err.code,\n            }\n          );\n        } else if (err instanceof Error) {\n          console.error(err);\n          return new HttpResponse(err.message, {\n            status: 500,\n            statusText: \"Internal Server Error\",\n          });\n        } else {\n          return new HttpResponse(\"Internal Server Error\", {\n            status: 500,\n          });\n        }\n      }\n    });\n  };\n}\n\nfunction initRouter() {\n  const router: Router = itty.Router<HttpRequest, Router>({\n    // paths always start with slash, the router will remove double slashes\n    base: \"/\",\n  });\n\n  for (const command of commands) {\n    const shouldValidate = command.validate !== false;\n\n    // RPC route takes a POST request and passes the parsed JSON body as input to the input\n    router.post(\n      commandRpcPath(command),\n      withMiddleware(async (request, context) => {\n        if (command.passThrough) {\n          // if passthrough is enabled, just proxy the request-response to the handler\n          return command.handler(request, context);\n        }\n\n        let input: any = await request.tryJson();\n        if (command.input && shouldValidate) {\n          try {\n            input = command.input.parse(input);\n          } catch (err) {\n            console.error(\"Invalid input\", err, input);\n            return new HttpResponse(JSON.stringify(err), {\n              status: 400,\n              statusText: \"Invalid input\",\n            });\n          }\n        }\n\n        let output: any = await command.handler(input, context);\n        if (command.output && shouldValidate) {\n          try {\n            output = command.output.parse(output);\n          } catch (err) {\n            console.error(\"RPC output did not match schema\", output, err);\n            return new HttpResponse(JSON.stringify(err), {\n              status: 500,\n              statusText: \"RPC output did not match schema\",\n            });\n          }\n        }\n        return new HttpResponse(JSON.stringify(output, jsonReplacer), {\n          status: 200,\n        });\n      })\n    );\n\n    const path = command.path;\n\n    if (path) {\n      const method = (command.method?.toLocaleLowerCase() ??\n        \"all\") as keyof Router;\n\n      // REST routes parse the request according to the command's path/method/params configuration\n      router[method](\n        path,\n        withMiddleware(async (request: HttpRequest, context) => {\n          if (command.passThrough) {\n            // if passthrough is enabled, just proxy the request-response to the handler\n            return command.handler(request, context);\n          }\n\n          // first, get the body as pure JSON - assume it's an object\n          const body = await request.tryJson();\n          let input: any = {\n            ...request.params,\n            ...(body && typeof body === \"object\" ? body : {}),\n          };\n\n          // parse headers/params/queries/body into the RPC interface\n          if (command.params) {\n            Object.entries(command.params as Record<string, RestParamSpec>).map(\n              ([name, spec]) => {\n                input[name] = resolveInput(name, spec);\n              }\n            );\n          }\n\n          if (command.input && shouldValidate) {\n            // validate the zod input schema if one is specified\n            input = command.input.parse(input);\n          }\n\n          // call the command RPC handler\n          let output: any = await command.handler(input, context);\n\n          if (command.output && shouldValidate) {\n            // validate the output of the command handler against the schema if it's defined\n            output = command.output.parse(output);\n          }\n\n          // TODO: support mapping RPC output back to HTTP properties such as Headers\n          // TODO: support alternative status code https://github.com/functionless/eventual/issues/276\n\n          return new HttpResponse(JSON.stringify(output, jsonReplacer), {\n            status: 200,\n            headers: {\n              \"Content-Type\": \"application/json\",\n            },\n          });\n\n          function resolveInput(name: string, spec: RestParamSpec): any {\n            if (spec === \"body\") {\n              return body?.[name];\n            } else if (spec === \"query\") {\n              return request.query?.[name];\n            } else if (spec === \"header\") {\n              return request.headers.get(name);\n            } else if (spec === \"path\") {\n              return request.params?.[name];\n            } else {\n              return resolveInput(spec.name ?? name, spec.in);\n            }\n          }\n        })\n      );\n    }\n\n    /**\n     * Applies the chain of middleware callbacks to the request to build up\n     * context and pass it through the chain and finally to the handler.\n     *\n     * Each context can add to or completely replace the context. They can\n     * also break the chain at any time by returning a HttpResponse instead\n     * of calling `next`.\n     *\n     * @param handler\n     * @returns\n     */\n    function withMiddleware(\n      handler: (request: HttpRequest, context: any) => Promise<HttpResponse>\n    ) {\n      return async (request: HttpRequest): Promise<HttpResponse> => {\n        const chain = (command.middlewares ?? []).values();\n\n        return next(request, {});\n\n        async function next(\n          request: HttpRequest,\n          context: any\n        ): Promise<HttpResponse> {\n          let consumed = false;\n          const middleware = chain.next();\n          if (middleware.done) {\n            return handler(request, context);\n          } else {\n            return middleware.value({\n              request,\n              context,\n              next: async (context) => {\n                if (consumed) {\n                  consumed = true;\n                  throw new Error(\n                    `Middleware cannot call 'next' more than once`\n                  );\n                }\n                return next(request, context);\n              },\n            });\n          }\n        }\n      };\n    }\n  }\n\n  return router;\n}\n\ninterface Router {\n  handle: (request: HttpRequest, ...extra: any) => Promise<HttpResponse>;\n  get: RouteFactory;\n  head: RouteFactory;\n  post: RouteFactory;\n  put: RouteFactory;\n  delete: RouteFactory;\n  connect: RouteFactory;\n  options: RouteFactory;\n  trace: RouteFactory;\n  patch: RouteFactory;\n}\n\ninterface RouteFactory {\n  (path: string, handlers: RouteHandler): Router;\n}\n\ntype RouteHandler = (\n  request: HttpRequest,\n  ...args: any\n) => HttpResponse | Promise<HttpResponse>;\n\n/**\n * Implements JSON serialization for well known types.\n */\nfunction jsonReplacer(_key: string, value: any) {\n  if (value instanceof Date) {\n    return value.toISOString();\n  }\n  return value;\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"command-worker.js","sourceRoot":"","sources":["../../../src/handlers/command-worker.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  EventualServiceClient,\n  isHttpError,\n  HttpRequest,\n  HttpResponse,\n  RestParamSpec,\n  commandRpcPath,\n} from \"@eventual/core\";\nimport {\n  registerServiceClient,\n  serviceTypeScope,\n  ServiceType,\n  commands,\n} from \"@eventual/core/internal\";\nimport itty from \"itty-router\";\n\nexport interface ApiHandlerDependencies {\n  serviceClient?: EventualServiceClient;\n}\n\nexport interface CommandWorker {\n  (request: HttpRequest): Promise<HttpResponse>;\n}\n\n/**\n * Creates a generic function for handling inbound API requests\n * that can be used in runtime implementations. This implementation is\n * decoupled from a runtime's specifics by the clients. A runtime must\n * inject its own client implementations designed for that platform.\n */\nexport function createCommandWorker({\n  serviceClient,\n}: ApiHandlerDependencies): CommandWorker {\n  // make the service client available to web hooks\n  if (serviceClient) {\n    registerServiceClient(serviceClient);\n  }\n\n  const router = initRouter();\n\n  /**\n   * Handle inbound webhook API requests.\n   *\n   * Each webhook registers routes on the central {@link router} which\n   * then handles the request.\n   */\n  return function (request) {\n    console.log(\"request\", request);\n    return serviceTypeScope(ServiceType.CommandWorker, async () => {\n      try {\n        const response = await router.handle(request);\n        if (response === undefined) {\n          if (request.method === \"OPTIONS\") {\n            return new HttpResponse(undefined, {\n              // CORS expects a 204 or 200, using 204 to match API Gateway\n              // and accurately reflect NO CONTENT\n              status: 204,\n            });\n          }\n          return new HttpResponse(\n            `Not Found: ${request.method} ${request.url}`,\n            {\n              status: 404,\n              statusText: \"Not Found\",\n            }\n          );\n        }\n        return response;\n      } catch (err) {\n        if (isHttpError(err)) {\n          return new HttpResponse(\n            JSON.stringify({\n              message: err.message,\n              data: err.data,\n            }),\n            {\n              status: err.code,\n            }\n          );\n        } else if (err instanceof Error) {\n          console.error(err);\n          return new HttpResponse(err.message, {\n            status: 500,\n            statusText: \"Internal Server Error\",\n          });\n        } else {\n          return new HttpResponse(\"Internal Server Error\", {\n            status: 500,\n          });\n        }\n      }\n    });\n  };\n}\n\nfunction initRouter() {\n  const router: Router = itty.Router<HttpRequest, Router>({\n    // paths always start with slash, the router will remove double slashes\n    base: \"/\",\n  });\n\n  for (const command of commands) {\n    const shouldValidate = command.validate !== false;\n\n    if (!command.passThrough) {\n      // RPC route takes a POST request and passes the parsed JSON body as input to the input\n      router.post(\n        commandRpcPath(command),\n        withMiddleware(async (request, context) => {\n          if (command.passThrough) {\n            // if passthrough is enabled, just proxy the request-response to the handler\n            return command.handler(request, context);\n          }\n\n          let input: any = await request.tryJson();\n          if (command.input && shouldValidate) {\n            try {\n              input = command.input.parse(input);\n            } catch (err) {\n              console.error(\"Invalid input\", err, input);\n              return new HttpResponse(JSON.stringify(err), {\n                status: 400,\n                statusText: \"Invalid input\",\n              });\n            }\n          }\n\n          let output: any = await command.handler(input, context);\n          if (command.output && shouldValidate) {\n            try {\n              output = command.output.parse(output);\n            } catch (err) {\n              console.error(\"RPC output did not match schema\", output, err);\n              return new HttpResponse(JSON.stringify(err), {\n                status: 500,\n                statusText: \"RPC output did not match schema\",\n              });\n            }\n          }\n          return new HttpResponse(JSON.stringify(output, jsonReplacer), {\n            status: 200,\n          });\n        })\n      );\n    }\n\n    const path = command.path;\n\n    if (path) {\n      const method = (command.method?.toLocaleLowerCase() ??\n        \"all\") as keyof Router;\n\n      // REST routes parse the request according to the command's path/method/params configuration\n      router[method](\n        path,\n        withMiddleware(async (request: HttpRequest, context) => {\n          if (command.passThrough) {\n            // if passthrough is enabled, just proxy the request-response to the handler\n            return command.handler(request, context);\n          }\n\n          // first, get the body as pure JSON - assume it's an object\n          const body = await request.tryJson();\n          let input: any = {\n            ...request.params,\n            ...(body && typeof body === \"object\" ? body : {}),\n          };\n\n          // parse headers/params/queries/body into the RPC interface\n          if (command.params) {\n            Object.entries(command.params as Record<string, RestParamSpec>).map(\n              ([name, spec]) => {\n                input[name] = resolveInput(name, spec);\n              }\n            );\n          }\n\n          if (command.input && shouldValidate) {\n            // validate the zod input schema if one is specified\n            input = command.input.parse(input);\n          }\n\n          // call the command RPC handler\n          let output: any = await command.handler(input, context);\n\n          if (command.output && shouldValidate) {\n            // validate the output of the command handler against the schema if it's defined\n            output = command.output.parse(output);\n          }\n\n          // TODO: support mapping RPC output back to HTTP properties such as Headers\n          // TODO: support alternative status code https://github.com/functionless/eventual/issues/276\n\n          return new HttpResponse(JSON.stringify(output, jsonReplacer), {\n            status: 200,\n            headers: {\n              \"Content-Type\": \"application/json\",\n            },\n          });\n\n          function resolveInput(name: string, spec: RestParamSpec): any {\n            if (spec === \"body\") {\n              return body?.[name];\n            } else if (spec === \"query\") {\n              return request.query?.[name];\n            } else if (spec === \"header\") {\n              return request.headers.get(name);\n            } else if (spec === \"path\") {\n              return request.params?.[name];\n            } else {\n              return resolveInput(spec.name ?? name, spec.in);\n            }\n          }\n        })\n      );\n    }\n\n    /**\n     * Applies the chain of middleware callbacks to the request to build up\n     * context and pass it through the chain and finally to the handler.\n     *\n     * Each context can add to or completely replace the context. They can\n     * also break the chain at any time by returning a HttpResponse instead\n     * of calling `next`.\n     *\n     * @param handler\n     * @returns\n     */\n    function withMiddleware(\n      handler: (request: HttpRequest, context: any) => Promise<HttpResponse>\n    ) {\n      return async (request: HttpRequest): Promise<HttpResponse> => {\n        const chain = (command.middlewares ?? []).values();\n\n        return next(request, {});\n\n        async function next(\n          request: HttpRequest,\n          context: any\n        ): Promise<HttpResponse> {\n          let consumed = false;\n          const middleware = chain.next();\n          if (middleware.done) {\n            return handler(request, context);\n          } else {\n            return middleware.value({\n              request,\n              context,\n              next: async (context) => {\n                if (consumed) {\n                  consumed = true;\n                  throw new Error(\n                    `Middleware cannot call 'next' more than once`\n                  );\n                }\n                return next(request, context);\n              },\n            });\n          }\n        }\n      };\n    }\n  }\n\n  return router;\n}\n\ninterface Router {\n  handle: (request: HttpRequest, ...extra: any) => Promise<HttpResponse>;\n  get: RouteFactory;\n  head: RouteFactory;\n  post: RouteFactory;\n  put: RouteFactory;\n  delete: RouteFactory;\n  connect: RouteFactory;\n  options: RouteFactory;\n  trace: RouteFactory;\n  patch: RouteFactory;\n}\n\ninterface RouteFactory {\n  (path: string, handlers: RouteHandler): Router;\n}\n\ntype RouteHandler = (\n  request: HttpRequest,\n  ...args: any\n) => HttpResponse | Promise<HttpResponse>;\n\n/**\n * Implements JSON serialization for well known types.\n */\nfunction jsonReplacer(_key: string, value: any) {\n  if (value instanceof Date) {\n    return value.toISOString();\n  }\n  return value;\n}\n"]}
{
"name": "@eventual/core-runtime",
"version": "0.25.0",
"version": "0.25.1",
"exports": {

@@ -16,3 +16,3 @@ ".": {

"dependencies": {
"@eventual/core": "^0.25.0",
"@eventual/core": "^0.25.1",
"itty-router": "^2.6.6",

@@ -19,0 +19,0 @@ "ulidx": "^0.3.0",

Sorry, the diff of this file is not supported yet

Sorry, the diff of this file is not supported yet

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc